1*90c8c64dSAndroid Build Coastguard Worker /*
2*90c8c64dSAndroid Build Coastguard Worker * Copyright 2014 The Android Open Source Project
3*90c8c64dSAndroid Build Coastguard Worker *
4*90c8c64dSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*90c8c64dSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*90c8c64dSAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*90c8c64dSAndroid Build Coastguard Worker *
8*90c8c64dSAndroid Build Coastguard Worker *     http://www.apache.org/licenses/LICENSE-2.0
9*90c8c64dSAndroid Build Coastguard Worker *
10*90c8c64dSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*90c8c64dSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*90c8c64dSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*90c8c64dSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*90c8c64dSAndroid Build Coastguard Worker * limitations under the License.
15*90c8c64dSAndroid Build Coastguard Worker */
16*90c8c64dSAndroid Build Coastguard Worker 
17*90c8c64dSAndroid Build Coastguard Worker package com.example.android.appusagestatistics;
18*90c8c64dSAndroid Build Coastguard Worker 
19*90c8c64dSAndroid Build Coastguard Worker import android.app.usage.UsageStats;
20*90c8c64dSAndroid Build Coastguard Worker import android.app.usage.UsageStatsManager;
21*90c8c64dSAndroid Build Coastguard Worker import android.content.Intent;
22*90c8c64dSAndroid Build Coastguard Worker import android.content.pm.PackageManager;
23*90c8c64dSAndroid Build Coastguard Worker import android.graphics.drawable.Drawable;
24*90c8c64dSAndroid Build Coastguard Worker import android.os.Bundle;
25*90c8c64dSAndroid Build Coastguard Worker import android.provider.Settings;
26*90c8c64dSAndroid Build Coastguard Worker import android.support.v4.app.Fragment;
27*90c8c64dSAndroid Build Coastguard Worker import android.support.v7.widget.RecyclerView;
28*90c8c64dSAndroid Build Coastguard Worker import android.util.Log;
29*90c8c64dSAndroid Build Coastguard Worker import android.view.LayoutInflater;
30*90c8c64dSAndroid Build Coastguard Worker import android.view.View;
31*90c8c64dSAndroid Build Coastguard Worker import android.view.ViewGroup;
32*90c8c64dSAndroid Build Coastguard Worker import android.widget.AdapterView;
33*90c8c64dSAndroid Build Coastguard Worker import android.widget.ArrayAdapter;
34*90c8c64dSAndroid Build Coastguard Worker import android.widget.Button;
35*90c8c64dSAndroid Build Coastguard Worker import android.widget.Spinner;
36*90c8c64dSAndroid Build Coastguard Worker import android.widget.SpinnerAdapter;
37*90c8c64dSAndroid Build Coastguard Worker import android.widget.Toast;
38*90c8c64dSAndroid Build Coastguard Worker 
39*90c8c64dSAndroid Build Coastguard Worker import java.util.ArrayList;
40*90c8c64dSAndroid Build Coastguard Worker import java.util.Calendar;
41*90c8c64dSAndroid Build Coastguard Worker import java.util.Collections;
42*90c8c64dSAndroid Build Coastguard Worker import java.util.Comparator;
43*90c8c64dSAndroid Build Coastguard Worker import java.util.List;
44*90c8c64dSAndroid Build Coastguard Worker 
45*90c8c64dSAndroid Build Coastguard Worker /**
46*90c8c64dSAndroid Build Coastguard Worker  * Fragment that demonstrates how to use App Usage Statistics API.
47*90c8c64dSAndroid Build Coastguard Worker  */
48*90c8c64dSAndroid Build Coastguard Worker public class AppUsageStatisticsFragment extends Fragment {
49*90c8c64dSAndroid Build Coastguard Worker 
50*90c8c64dSAndroid Build Coastguard Worker     private static final String TAG = AppUsageStatisticsFragment.class.getSimpleName();
51*90c8c64dSAndroid Build Coastguard Worker 
52*90c8c64dSAndroid Build Coastguard Worker     //VisibleForTesting for variables below
53*90c8c64dSAndroid Build Coastguard Worker     UsageStatsManager mUsageStatsManager;
54*90c8c64dSAndroid Build Coastguard Worker     UsageListAdapter mUsageListAdapter;
55*90c8c64dSAndroid Build Coastguard Worker     RecyclerView mRecyclerView;
56*90c8c64dSAndroid Build Coastguard Worker     RecyclerView.LayoutManager mLayoutManager;
57*90c8c64dSAndroid Build Coastguard Worker     Button mOpenUsageSettingButton;
58*90c8c64dSAndroid Build Coastguard Worker     Spinner mSpinner;
59*90c8c64dSAndroid Build Coastguard Worker 
60*90c8c64dSAndroid Build Coastguard Worker     /**
61*90c8c64dSAndroid Build Coastguard Worker      * Use this factory method to create a new instance of
62*90c8c64dSAndroid Build Coastguard Worker      * this fragment using the provided parameters.
63*90c8c64dSAndroid Build Coastguard Worker      *
64*90c8c64dSAndroid Build Coastguard Worker      * @return A new instance of fragment {@link AppUsageStatisticsFragment}.
65*90c8c64dSAndroid Build Coastguard Worker      */
newInstance()66*90c8c64dSAndroid Build Coastguard Worker     public static AppUsageStatisticsFragment newInstance() {
67*90c8c64dSAndroid Build Coastguard Worker         AppUsageStatisticsFragment fragment = new AppUsageStatisticsFragment();
68*90c8c64dSAndroid Build Coastguard Worker         return fragment;
69*90c8c64dSAndroid Build Coastguard Worker     }
70*90c8c64dSAndroid Build Coastguard Worker 
AppUsageStatisticsFragment()71*90c8c64dSAndroid Build Coastguard Worker     public AppUsageStatisticsFragment() {
72*90c8c64dSAndroid Build Coastguard Worker         // Required empty public constructor
73*90c8c64dSAndroid Build Coastguard Worker     }
74*90c8c64dSAndroid Build Coastguard Worker 
75*90c8c64dSAndroid Build Coastguard Worker     @Override
onCreate(Bundle savedInstanceState)76*90c8c64dSAndroid Build Coastguard Worker     public void onCreate(Bundle savedInstanceState) {
77*90c8c64dSAndroid Build Coastguard Worker         super.onCreate(savedInstanceState);
78*90c8c64dSAndroid Build Coastguard Worker 
79*90c8c64dSAndroid Build Coastguard Worker         mUsageStatsManager = (UsageStatsManager) getActivity()
80*90c8c64dSAndroid Build Coastguard Worker                 .getSystemService("usagestats"); //Context.USAGE_STATS_SERVICE
81*90c8c64dSAndroid Build Coastguard Worker     }
82*90c8c64dSAndroid Build Coastguard Worker 
83*90c8c64dSAndroid Build Coastguard Worker     @Override
onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState)84*90c8c64dSAndroid Build Coastguard Worker     public View onCreateView(LayoutInflater inflater, ViewGroup container,
85*90c8c64dSAndroid Build Coastguard Worker             Bundle savedInstanceState) {
86*90c8c64dSAndroid Build Coastguard Worker         return inflater.inflate(R.layout.fragment_app_usage_statistics, container, false);
87*90c8c64dSAndroid Build Coastguard Worker     }
88*90c8c64dSAndroid Build Coastguard Worker 
89*90c8c64dSAndroid Build Coastguard Worker     @Override
onViewCreated(View rootView, Bundle savedInstanceState)90*90c8c64dSAndroid Build Coastguard Worker     public void onViewCreated(View rootView, Bundle savedInstanceState) {
91*90c8c64dSAndroid Build Coastguard Worker         super.onViewCreated(rootView, savedInstanceState);
92*90c8c64dSAndroid Build Coastguard Worker 
93*90c8c64dSAndroid Build Coastguard Worker         mUsageListAdapter = new UsageListAdapter();
94*90c8c64dSAndroid Build Coastguard Worker         mRecyclerView = (RecyclerView) rootView.findViewById(R.id.recyclerview_app_usage);
95*90c8c64dSAndroid Build Coastguard Worker         mLayoutManager = mRecyclerView.getLayoutManager();
96*90c8c64dSAndroid Build Coastguard Worker         mRecyclerView.scrollToPosition(0);
97*90c8c64dSAndroid Build Coastguard Worker         mRecyclerView.setAdapter(mUsageListAdapter);
98*90c8c64dSAndroid Build Coastguard Worker         mOpenUsageSettingButton = (Button) rootView.findViewById(R.id.button_open_usage_setting);
99*90c8c64dSAndroid Build Coastguard Worker         mSpinner = (Spinner) rootView.findViewById(R.id.spinner_time_span);
100*90c8c64dSAndroid Build Coastguard Worker         SpinnerAdapter spinnerAdapter = ArrayAdapter.createFromResource(getActivity(),
101*90c8c64dSAndroid Build Coastguard Worker                 R.array.action_list, android.R.layout.simple_spinner_dropdown_item);
102*90c8c64dSAndroid Build Coastguard Worker         mSpinner.setAdapter(spinnerAdapter);
103*90c8c64dSAndroid Build Coastguard Worker         mSpinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
104*90c8c64dSAndroid Build Coastguard Worker 
105*90c8c64dSAndroid Build Coastguard Worker             String[] strings = getResources().getStringArray(R.array.action_list);
106*90c8c64dSAndroid Build Coastguard Worker 
107*90c8c64dSAndroid Build Coastguard Worker             @Override
108*90c8c64dSAndroid Build Coastguard Worker             public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
109*90c8c64dSAndroid Build Coastguard Worker                 StatsUsageInterval statsUsageInterval = StatsUsageInterval
110*90c8c64dSAndroid Build Coastguard Worker                         .getValue(strings[position]);
111*90c8c64dSAndroid Build Coastguard Worker                 if (statsUsageInterval != null) {
112*90c8c64dSAndroid Build Coastguard Worker                     List<UsageStats> usageStatsList =
113*90c8c64dSAndroid Build Coastguard Worker                             getUsageStatistics(statsUsageInterval.mInterval);
114*90c8c64dSAndroid Build Coastguard Worker                     Collections.sort(usageStatsList, new LastTimeLaunchedComparatorDesc());
115*90c8c64dSAndroid Build Coastguard Worker                     updateAppsList(usageStatsList);
116*90c8c64dSAndroid Build Coastguard Worker                 }
117*90c8c64dSAndroid Build Coastguard Worker             }
118*90c8c64dSAndroid Build Coastguard Worker 
119*90c8c64dSAndroid Build Coastguard Worker             @Override
120*90c8c64dSAndroid Build Coastguard Worker             public void onNothingSelected(AdapterView<?> parent) {
121*90c8c64dSAndroid Build Coastguard Worker             }
122*90c8c64dSAndroid Build Coastguard Worker         });
123*90c8c64dSAndroid Build Coastguard Worker     }
124*90c8c64dSAndroid Build Coastguard Worker 
125*90c8c64dSAndroid Build Coastguard Worker     /**
126*90c8c64dSAndroid Build Coastguard Worker      * Returns the {@link #mRecyclerView} including the time span specified by the
127*90c8c64dSAndroid Build Coastguard Worker      * intervalType argument.
128*90c8c64dSAndroid Build Coastguard Worker      *
129*90c8c64dSAndroid Build Coastguard Worker      * @param intervalType The time interval by which the stats are aggregated.
130*90c8c64dSAndroid Build Coastguard Worker      *                     Corresponding to the value of {@link UsageStatsManager}.
131*90c8c64dSAndroid Build Coastguard Worker      *                     E.g. {@link UsageStatsManager#INTERVAL_DAILY}, {@link
132*90c8c64dSAndroid Build Coastguard Worker      *                     UsageStatsManager#INTERVAL_WEEKLY},
133*90c8c64dSAndroid Build Coastguard Worker      *
134*90c8c64dSAndroid Build Coastguard Worker      * @return A list of {@link android.app.usage.UsageStats}.
135*90c8c64dSAndroid Build Coastguard Worker      */
getUsageStatistics(int intervalType)136*90c8c64dSAndroid Build Coastguard Worker     public List<UsageStats> getUsageStatistics(int intervalType) {
137*90c8c64dSAndroid Build Coastguard Worker         // Get the app statistics since one year ago from the current time.
138*90c8c64dSAndroid Build Coastguard Worker         Calendar cal = Calendar.getInstance();
139*90c8c64dSAndroid Build Coastguard Worker         cal.add(Calendar.YEAR, -1);
140*90c8c64dSAndroid Build Coastguard Worker 
141*90c8c64dSAndroid Build Coastguard Worker         List<UsageStats> queryUsageStats = mUsageStatsManager
142*90c8c64dSAndroid Build Coastguard Worker                 .queryUsageStats(intervalType, cal.getTimeInMillis(),
143*90c8c64dSAndroid Build Coastguard Worker                         System.currentTimeMillis());
144*90c8c64dSAndroid Build Coastguard Worker 
145*90c8c64dSAndroid Build Coastguard Worker         if (queryUsageStats.size() == 0) {
146*90c8c64dSAndroid Build Coastguard Worker             Log.i(TAG, "The user may not allow the access to apps usage. ");
147*90c8c64dSAndroid Build Coastguard Worker             Toast.makeText(getActivity(),
148*90c8c64dSAndroid Build Coastguard Worker                     getString(R.string.explanation_access_to_appusage_is_not_enabled),
149*90c8c64dSAndroid Build Coastguard Worker                     Toast.LENGTH_LONG).show();
150*90c8c64dSAndroid Build Coastguard Worker             mOpenUsageSettingButton.setVisibility(View.VISIBLE);
151*90c8c64dSAndroid Build Coastguard Worker             mOpenUsageSettingButton.setOnClickListener(new View.OnClickListener() {
152*90c8c64dSAndroid Build Coastguard Worker                 @Override
153*90c8c64dSAndroid Build Coastguard Worker                 public void onClick(View v) {
154*90c8c64dSAndroid Build Coastguard Worker                     startActivity(new Intent(Settings.ACTION_USAGE_ACCESS_SETTINGS));
155*90c8c64dSAndroid Build Coastguard Worker                 }
156*90c8c64dSAndroid Build Coastguard Worker             });
157*90c8c64dSAndroid Build Coastguard Worker         }
158*90c8c64dSAndroid Build Coastguard Worker         return queryUsageStats;
159*90c8c64dSAndroid Build Coastguard Worker     }
160*90c8c64dSAndroid Build Coastguard Worker 
161*90c8c64dSAndroid Build Coastguard Worker     /**
162*90c8c64dSAndroid Build Coastguard Worker      * Updates the {@link #mRecyclerView} with the list of {@link UsageStats} passed as an argument.
163*90c8c64dSAndroid Build Coastguard Worker      *
164*90c8c64dSAndroid Build Coastguard Worker      * @param usageStatsList A list of {@link UsageStats} from which update the
165*90c8c64dSAndroid Build Coastguard Worker      *                       {@link #mRecyclerView}.
166*90c8c64dSAndroid Build Coastguard Worker      */
167*90c8c64dSAndroid Build Coastguard Worker     //VisibleForTesting
updateAppsList(List<UsageStats> usageStatsList)168*90c8c64dSAndroid Build Coastguard Worker     void updateAppsList(List<UsageStats> usageStatsList) {
169*90c8c64dSAndroid Build Coastguard Worker         List<CustomUsageStats> customUsageStatsList = new ArrayList<>();
170*90c8c64dSAndroid Build Coastguard Worker         for (int i = 0; i < usageStatsList.size(); i++) {
171*90c8c64dSAndroid Build Coastguard Worker             CustomUsageStats customUsageStats = new CustomUsageStats();
172*90c8c64dSAndroid Build Coastguard Worker             customUsageStats.usageStats = usageStatsList.get(i);
173*90c8c64dSAndroid Build Coastguard Worker             try {
174*90c8c64dSAndroid Build Coastguard Worker                 Drawable appIcon = getActivity().getPackageManager()
175*90c8c64dSAndroid Build Coastguard Worker                         .getApplicationIcon(customUsageStats.usageStats.getPackageName());
176*90c8c64dSAndroid Build Coastguard Worker                 customUsageStats.appIcon = appIcon;
177*90c8c64dSAndroid Build Coastguard Worker             } catch (PackageManager.NameNotFoundException e) {
178*90c8c64dSAndroid Build Coastguard Worker                 Log.w(TAG, String.format("App Icon is not found for %s",
179*90c8c64dSAndroid Build Coastguard Worker                         customUsageStats.usageStats.getPackageName()));
180*90c8c64dSAndroid Build Coastguard Worker                 customUsageStats.appIcon = getActivity()
181*90c8c64dSAndroid Build Coastguard Worker                         .getDrawable(R.drawable.ic_default_app_launcher);
182*90c8c64dSAndroid Build Coastguard Worker             }
183*90c8c64dSAndroid Build Coastguard Worker             customUsageStatsList.add(customUsageStats);
184*90c8c64dSAndroid Build Coastguard Worker         }
185*90c8c64dSAndroid Build Coastguard Worker         mUsageListAdapter.setCustomUsageStatsList(customUsageStatsList);
186*90c8c64dSAndroid Build Coastguard Worker         mUsageListAdapter.notifyDataSetChanged();
187*90c8c64dSAndroid Build Coastguard Worker         mRecyclerView.scrollToPosition(0);
188*90c8c64dSAndroid Build Coastguard Worker     }
189*90c8c64dSAndroid Build Coastguard Worker 
190*90c8c64dSAndroid Build Coastguard Worker     /**
191*90c8c64dSAndroid Build Coastguard Worker      * The {@link Comparator} to sort a collection of {@link UsageStats} sorted by the timestamp
192*90c8c64dSAndroid Build Coastguard Worker      * last time the app was used in the descendant order.
193*90c8c64dSAndroid Build Coastguard Worker      */
194*90c8c64dSAndroid Build Coastguard Worker     private static class LastTimeLaunchedComparatorDesc implements Comparator<UsageStats> {
195*90c8c64dSAndroid Build Coastguard Worker 
196*90c8c64dSAndroid Build Coastguard Worker         @Override
compare(UsageStats left, UsageStats right)197*90c8c64dSAndroid Build Coastguard Worker         public int compare(UsageStats left, UsageStats right) {
198*90c8c64dSAndroid Build Coastguard Worker             return Long.compare(right.getLastTimeUsed(), left.getLastTimeUsed());
199*90c8c64dSAndroid Build Coastguard Worker         }
200*90c8c64dSAndroid Build Coastguard Worker     }
201*90c8c64dSAndroid Build Coastguard Worker 
202*90c8c64dSAndroid Build Coastguard Worker     /**
203*90c8c64dSAndroid Build Coastguard Worker      * Enum represents the intervals for {@link android.app.usage.UsageStatsManager} so that
204*90c8c64dSAndroid Build Coastguard Worker      * values for intervals can be found by a String representation.
205*90c8c64dSAndroid Build Coastguard Worker      *
206*90c8c64dSAndroid Build Coastguard Worker      */
207*90c8c64dSAndroid Build Coastguard Worker     //VisibleForTesting
208*90c8c64dSAndroid Build Coastguard Worker     static enum StatsUsageInterval {
209*90c8c64dSAndroid Build Coastguard Worker         DAILY("Daily", UsageStatsManager.INTERVAL_DAILY),
210*90c8c64dSAndroid Build Coastguard Worker         WEEKLY("Weekly", UsageStatsManager.INTERVAL_WEEKLY),
211*90c8c64dSAndroid Build Coastguard Worker         MONTHLY("Monthly", UsageStatsManager.INTERVAL_MONTHLY),
212*90c8c64dSAndroid Build Coastguard Worker         YEARLY("Yearly", UsageStatsManager.INTERVAL_YEARLY);
213*90c8c64dSAndroid Build Coastguard Worker 
214*90c8c64dSAndroid Build Coastguard Worker         private int mInterval;
215*90c8c64dSAndroid Build Coastguard Worker         private String mStringRepresentation;
216*90c8c64dSAndroid Build Coastguard Worker 
StatsUsageInterval(String stringRepresentation, int interval)217*90c8c64dSAndroid Build Coastguard Worker         StatsUsageInterval(String stringRepresentation, int interval) {
218*90c8c64dSAndroid Build Coastguard Worker             mStringRepresentation = stringRepresentation;
219*90c8c64dSAndroid Build Coastguard Worker             mInterval = interval;
220*90c8c64dSAndroid Build Coastguard Worker         }
221*90c8c64dSAndroid Build Coastguard Worker 
getValue(String stringRepresentation)222*90c8c64dSAndroid Build Coastguard Worker         static StatsUsageInterval getValue(String stringRepresentation) {
223*90c8c64dSAndroid Build Coastguard Worker             for (StatsUsageInterval statsUsageInterval : values()) {
224*90c8c64dSAndroid Build Coastguard Worker                 if (statsUsageInterval.mStringRepresentation.equals(stringRepresentation)) {
225*90c8c64dSAndroid Build Coastguard Worker                     return statsUsageInterval;
226*90c8c64dSAndroid Build Coastguard Worker                 }
227*90c8c64dSAndroid Build Coastguard Worker             }
228*90c8c64dSAndroid Build Coastguard Worker             return null;
229*90c8c64dSAndroid Build Coastguard Worker         }
230*90c8c64dSAndroid Build Coastguard Worker     }
231*90c8c64dSAndroid Build Coastguard Worker }
232