/* * Copyright 2014 The Android Open Source Project * * 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 * * 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.example.android.appusagestatistics; import android.app.usage.UsageStats; import android.app.usage.UsageStatsManager; import android.content.Intent; import android.content.pm.PackageManager; import android.graphics.drawable.Drawable; import android.os.Bundle; import android.provider.Settings; import android.support.v4.app.Fragment; import android.support.v7.widget.RecyclerView; import android.util.Log; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.AdapterView; import android.widget.ArrayAdapter; import android.widget.Button; import android.widget.Spinner; import android.widget.SpinnerAdapter; import android.widget.Toast; import java.util.ArrayList; import java.util.Calendar; import java.util.Collections; import java.util.Comparator; import java.util.List; /** * Fragment that demonstrates how to use App Usage Statistics API. */ public class AppUsageStatisticsFragment extends Fragment { private static final String TAG = AppUsageStatisticsFragment.class.getSimpleName(); //VisibleForTesting for variables below UsageStatsManager mUsageStatsManager; UsageListAdapter mUsageListAdapter; RecyclerView mRecyclerView; RecyclerView.LayoutManager mLayoutManager; Button mOpenUsageSettingButton; Spinner mSpinner; /** * Use this factory method to create a new instance of * this fragment using the provided parameters. * * @return A new instance of fragment {@link AppUsageStatisticsFragment}. */ public static AppUsageStatisticsFragment newInstance() { AppUsageStatisticsFragment fragment = new AppUsageStatisticsFragment(); return fragment; } public AppUsageStatisticsFragment() { // Required empty public constructor } @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); mUsageStatsManager = (UsageStatsManager) getActivity() .getSystemService("usagestats"); //Context.USAGE_STATS_SERVICE } @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { return inflater.inflate(R.layout.fragment_app_usage_statistics, container, false); } @Override public void onViewCreated(View rootView, Bundle savedInstanceState) { super.onViewCreated(rootView, savedInstanceState); mUsageListAdapter = new UsageListAdapter(); mRecyclerView = (RecyclerView) rootView.findViewById(R.id.recyclerview_app_usage); mLayoutManager = mRecyclerView.getLayoutManager(); mRecyclerView.scrollToPosition(0); mRecyclerView.setAdapter(mUsageListAdapter); mOpenUsageSettingButton = (Button) rootView.findViewById(R.id.button_open_usage_setting); mSpinner = (Spinner) rootView.findViewById(R.id.spinner_time_span); SpinnerAdapter spinnerAdapter = ArrayAdapter.createFromResource(getActivity(), R.array.action_list, android.R.layout.simple_spinner_dropdown_item); mSpinner.setAdapter(spinnerAdapter); mSpinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() { String[] strings = getResources().getStringArray(R.array.action_list); @Override public void onItemSelected(AdapterView parent, View view, int position, long id) { StatsUsageInterval statsUsageInterval = StatsUsageInterval .getValue(strings[position]); if (statsUsageInterval != null) { List usageStatsList = getUsageStatistics(statsUsageInterval.mInterval); Collections.sort(usageStatsList, new LastTimeLaunchedComparatorDesc()); updateAppsList(usageStatsList); } } @Override public void onNothingSelected(AdapterView parent) { } }); } /** * Returns the {@link #mRecyclerView} including the time span specified by the * intervalType argument. * * @param intervalType The time interval by which the stats are aggregated. * Corresponding to the value of {@link UsageStatsManager}. * E.g. {@link UsageStatsManager#INTERVAL_DAILY}, {@link * UsageStatsManager#INTERVAL_WEEKLY}, * * @return A list of {@link android.app.usage.UsageStats}. */ public List getUsageStatistics(int intervalType) { // Get the app statistics since one year ago from the current time. Calendar cal = Calendar.getInstance(); cal.add(Calendar.YEAR, -1); List queryUsageStats = mUsageStatsManager .queryUsageStats(intervalType, cal.getTimeInMillis(), System.currentTimeMillis()); if (queryUsageStats.size() == 0) { Log.i(TAG, "The user may not allow the access to apps usage. "); Toast.makeText(getActivity(), getString(R.string.explanation_access_to_appusage_is_not_enabled), Toast.LENGTH_LONG).show(); mOpenUsageSettingButton.setVisibility(View.VISIBLE); mOpenUsageSettingButton.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { startActivity(new Intent(Settings.ACTION_USAGE_ACCESS_SETTINGS)); } }); } return queryUsageStats; } /** * Updates the {@link #mRecyclerView} with the list of {@link UsageStats} passed as an argument. * * @param usageStatsList A list of {@link UsageStats} from which update the * {@link #mRecyclerView}. */ //VisibleForTesting void updateAppsList(List usageStatsList) { List customUsageStatsList = new ArrayList<>(); for (int i = 0; i < usageStatsList.size(); i++) { CustomUsageStats customUsageStats = new CustomUsageStats(); customUsageStats.usageStats = usageStatsList.get(i); try { Drawable appIcon = getActivity().getPackageManager() .getApplicationIcon(customUsageStats.usageStats.getPackageName()); customUsageStats.appIcon = appIcon; } catch (PackageManager.NameNotFoundException e) { Log.w(TAG, String.format("App Icon is not found for %s", customUsageStats.usageStats.getPackageName())); customUsageStats.appIcon = getActivity() .getDrawable(R.drawable.ic_default_app_launcher); } customUsageStatsList.add(customUsageStats); } mUsageListAdapter.setCustomUsageStatsList(customUsageStatsList); mUsageListAdapter.notifyDataSetChanged(); mRecyclerView.scrollToPosition(0); } /** * The {@link Comparator} to sort a collection of {@link UsageStats} sorted by the timestamp * last time the app was used in the descendant order. */ private static class LastTimeLaunchedComparatorDesc implements Comparator { @Override public int compare(UsageStats left, UsageStats right) { return Long.compare(right.getLastTimeUsed(), left.getLastTimeUsed()); } } /** * Enum represents the intervals for {@link android.app.usage.UsageStatsManager} so that * values for intervals can be found by a String representation. * */ //VisibleForTesting static enum StatsUsageInterval { DAILY("Daily", UsageStatsManager.INTERVAL_DAILY), WEEKLY("Weekly", UsageStatsManager.INTERVAL_WEEKLY), MONTHLY("Monthly", UsageStatsManager.INTERVAL_MONTHLY), YEARLY("Yearly", UsageStatsManager.INTERVAL_YEARLY); private int mInterval; private String mStringRepresentation; StatsUsageInterval(String stringRepresentation, int interval) { mStringRepresentation = stringRepresentation; mInterval = interval; } static StatsUsageInterval getValue(String stringRepresentation) { for (StatsUsageInterval statsUsageInterval : values()) { if (statsUsageInterval.mStringRepresentation.equals(stringRepresentation)) { return statsUsageInterval; } } return null; } } }