Back to project page appengine-dashboard-app.
The source code is released under:
Apache License
If you think the Android project appengine-dashboard-app listed in this page is inappropriate, such as containing malicious code/tools or violating the copyright, please email info at java2s dot com, thanks.
/* * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at *// w w w . j a v a 2s .c o m * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.friedran.appengine.dashboard.gui; import android.accounts.Account; import android.content.Context; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.os.AsyncTask; import android.os.Bundle; import android.support.v4.util.LruCache; import android.util.DisplayMetrics; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.view.animation.AnimationUtils; import android.widget.AdapterView; import android.widget.ArrayAdapter; import android.widget.BaseAdapter; import android.widget.GridView; import android.widget.ImageView; import android.widget.LinearLayout; import android.widget.Spinner; import android.widget.TextView; import android.widget.Toast; import android.widget.ViewSwitcher; import com.actionbarsherlock.app.SherlockFragment; import com.friedran.appengine.dashboard.R; import com.friedran.appengine.dashboard.client.AppEngineDashboardAPI; import com.friedran.appengine.dashboard.client.AppEngineDashboardClient; import com.friedran.appengine.dashboard.utils.AnalyticsUtils; import com.friedran.appengine.dashboard.utils.LogUtils; import com.google.analytics.tracking.android.Tracker; import java.io.InputStream; import uk.co.senab.actionbarpulltorefresh.library.PullToRefreshAttacher; public class DashboardLoadFragment extends SherlockFragment implements AdapterView.OnItemSelectedListener, PullToRefreshAttacher.OnRefreshListener { public static final String CHART_URL_BACKGROUND_COLOR_SUFFIX = "&chf=bg,s,E8E8E8"; public static final int CHART_HEIGHT_PIXELS = 240; public static final int CHART_MAX_WIDTH_PIXELS = 1000; public static final String KEY_ACCOUNT = "KEY_ACCOUNT"; public static final String KEY_APPLICATION_ID = "KEY_APPLICATION_ID"; private DashboardActivity mActivity; private ChartAdapter mChartGridAdapter; private PullToRefreshAttacher mPullToRefreshAttacher; private DisplayMetrics mDisplayMetrics; private LruCache<String, Bitmap> mChartsMemoryCache; private Tracker mTracker; int mDisplayedTimeID; public static DashboardLoadFragment newInstance(Account account, String applicationID) { Bundle args = new Bundle(); args.putParcelable(KEY_ACCOUNT, account); args.putString(KEY_APPLICATION_ID, applicationID); DashboardLoadFragment fragment = new DashboardLoadFragment(); fragment.setArguments(args); return fragment; } @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { LinearLayout layout = (LinearLayout) inflater.inflate(R.layout.load_fragment, container, false); mActivity = (DashboardActivity) getSherlockActivity(); setSpinnerWithItems(layout, R.array.load_time_options, R.id.load_chart_time_spinner); mDisplayedTimeID = 0; mDisplayMetrics = new DisplayMetrics(); mActivity.getWindowManager().getDefaultDisplay().getMetrics(mDisplayMetrics); Account account = getArguments().getParcelable(KEY_ACCOUNT); String applicationID = getArguments().getString(KEY_APPLICATION_ID); mChartGridAdapter = new ChartAdapter(mActivity, AppEngineDashboardAPI.getInstance().getClient(account), applicationID); GridView chartsGridView = (GridView) layout.findViewById(R.id.load_charts_grid); chartsGridView.setAdapter(mChartGridAdapter); mChartsMemoryCache = initChartsMemoryCache(); mTracker = AnalyticsUtils.getTracker(mActivity); mPullToRefreshAttacher = mActivity.getPullToRefreshAttacher(); mPullToRefreshAttacher.setRefreshableView(chartsGridView, this); return layout; } private LruCache<String, Bitmap> initChartsMemoryCache() { final int maxMemoryInKBs = (int) (Runtime.getRuntime().maxMemory() / 1024); final int cacheSize = maxMemoryInKBs / 5; return new LruCache<String, Bitmap>(cacheSize) { @Override protected int sizeOf(String key, Bitmap bitmap) { // Measured in kilobytes return bitmap.getByteCount() / 1024; } }; } private Spinner setSpinnerWithItems(LinearLayout layout, int optionsListResourceID, int spinnerResourceID) { Spinner spinner = (Spinner) layout.findViewById(spinnerResourceID); // Set options list ArrayAdapter<CharSequence> adapter = ArrayAdapter.createFromResource(mActivity, optionsListResourceID, android.R.layout.simple_spinner_item); adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item); spinner.setAdapter(adapter); // Set listener spinner.setOnItemSelectedListener(this); return spinner; } @Override public void onResume() { super.onResume(); } /** * Called when the time spinner is selected */ @Override public void onItemSelected(AdapterView<?> parent, View view, int position, long id) { if (position != mDisplayedTimeID) { LogUtils.i("DashboardLoadFragment", "Time option selected: " + mDisplayedTimeID + " ==> " + position); AnalyticsUtils.sendEvent(mTracker, "ui_action", "spinner_click", "time_spinner_" + position, null); mDisplayedTimeID = position; mChartGridAdapter.notifyDataSetChanged(); } } @Override public void onNothingSelected(AdapterView<?> parent) { // Do Nothing } @Override public void onRefreshStarted(View view) { refresh(); } public void refresh() { mChartsMemoryCache.evictAll(); mChartGridAdapter.notifyDataSetChanged(); } private class ChartAdapter extends BaseAdapter { private Context mContext; private AppEngineDashboardClient mAppEngineClient; private String mApplicationID; private String[] mAppEngineMetrics; public ChartAdapter(Context c, AppEngineDashboardClient appEngineClient, String applicationID) { mContext = c; mAppEngineClient = appEngineClient; mApplicationID = applicationID; mAppEngineMetrics = getResources().getStringArray(R.array.load_metric_options); } // Disables highlighting items @Override public boolean areAllItemsEnabled() { return false; } // Disables highlighting items @Override public boolean isEnabled(int position) { return false; } @Override public int getCount() { return mAppEngineMetrics.length; } @Override public Object getItem(int position) { return mAppEngineMetrics[position]; } @Override public long getItemId(int position) { return position; } @Override public View getView(int position, View chartView, ViewGroup parent) { LayoutInflater inflater = (LayoutInflater) mContext.getSystemService(Context.LAYOUT_INFLATER_SERVICE); chartView = inflater.inflate(R.layout.load_charts_grid_item, null); TextView textView = (TextView) chartView.findViewById(R.id.load_chart_title); textView.setText(mAppEngineMetrics[position]); Bitmap chartImage = getChartImageFromCache(position, mDisplayedTimeID); if (chartImage != null) { LogUtils.i("DashboardLoadFragment", String.format("Updated chart from cache: %d, %d", position, mDisplayedTimeID)); updateChartImage(chartView, chartImage, false); } else { // Load the image asynchronously, while displaying the progress animation switchChartToProgress(chartView); executeGetAndDisplayChart(chartView, mDisplayedTimeID, position); } return chartView; } // Gets and loads the chart into chartView asynchronously private void executeGetAndDisplayChart(final View chartView, final int selectedTimeWindow, final int metricTypeID) { mAppEngineClient.executeGetChartUrl(mApplicationID, metricTypeID, selectedTimeWindow, new AppEngineDashboardClient.PostExecuteCallback() { @Override public void onPostExecute(Bundle result) { if (!result.getBoolean(AppEngineDashboardClient.KEY_RESULT)) { LogUtils.e("DashboardLoadFragment", "GetChartURL has failed"); updateChartImage(chartView, null, true); AnalyticsUtils.sendEvent(mTracker, "ui_event", "get_chart_url_failed", "get_chart_url_failed", null); return; } String chartUrl = result.getString(AppEngineDashboardClient.KEY_CHART_URL); chartUrl = chartUrl.replaceAll("chs=\\d+x\\d+", String.format("chs=%sx%s", Math.min(mDisplayMetrics.widthPixels, CHART_MAX_WIDTH_PIXELS), CHART_HEIGHT_PIXELS)); chartUrl += CHART_URL_BACKGROUND_COLOR_SUFFIX; new ChartDownloadTask(mActivity, chartView, selectedTimeWindow, metricTypeID, chartUrl).executeOnExecutor(AsyncTask.THREAD_POOL_EXECUTOR); } }); } } /** Downloads a chart image and displays it asynchronously */ private class ChartDownloadTask extends AsyncTask<String, Void, Bitmap> { Context mContext; View mChartView; int mTimeWindowID; int mMetricTypeID; String mUrl; public ChartDownloadTask(Context context, View chartView, int timeWindowID, int metricTypeID, String url) { mContext = context; mChartView = chartView; mTimeWindowID = timeWindowID; mMetricTypeID = metricTypeID; mUrl = url; } @Override protected Bitmap doInBackground(String... params) { Bitmap decodedBitmap = null; try { LogUtils.i("DashboardLoadFragment", String.format("Downloading chart (%s, %s) from: %s", mTimeWindowID, mMetricTypeID, mUrl)); InputStream in = new java.net.URL(mUrl).openStream(); decodedBitmap = BitmapFactory.decodeStream(in); updateChartImageInCache(mMetricTypeID, mTimeWindowID, decodedBitmap); } catch (Exception e) { LogUtils.e("DashboardLoadFragment", e.getMessage(), e); } return decodedBitmap; } @Override protected void onPostExecute(Bitmap result) { updateChartImage(mChartView, result, true); } } private Bitmap getChartImageFromCache(int metricID, int windowID) { return mChartsMemoryCache.get(String.format("%s.%s", metricID, windowID)); } private void updateChartImageInCache(int metricID, int windowID, Bitmap image) { mChartsMemoryCache.put(String.format("%s.%s", metricID, windowID), image); } private void switchChartToProgress(View chartView) { ViewSwitcher switcher = (ViewSwitcher) chartView.findViewById(R.id.load_chart_switcher); if (switcher.getDisplayedChild() != 0) { switcher.showPrevious(); } } private void updateChartImage(View chartView, Bitmap image, boolean animate) { ImageView chartImageView = (ImageView) chartView.findViewById(R.id.load_chart_image); if (image != null) { chartImageView.setImageBitmap(image); } else { chartImageView.setImageResource(android.R.color.transparent); Toast.makeText(mActivity, "Failed downloading charts, please make sure you have Internet connectivity and try refreshing", 2000).show(); } ViewSwitcher viewSwitcher = (ViewSwitcher) chartView.findViewById(R.id.load_chart_switcher); if (viewSwitcher.getDisplayedChild() != 1) { if (animate) { viewSwitcher.setAnimation(AnimationUtils.loadAnimation(mActivity, R.anim.fadein)); } else viewSwitcher.setAnimation(null); viewSwitcher.showNext(); } // Mark the refresh UI as complete once anyone of the charts has been loaded. mPullToRefreshAttacher.setRefreshComplete(); } }