1 /*
2  * Copyright (C) 2015 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.tv.tuner.setup;
18 
19 import android.app.Fragment;
20 import android.app.Notification;
21 import android.app.NotificationChannel;
22 import android.app.NotificationManager;
23 import android.app.PendingIntent;
24 import android.content.Context;
25 import android.content.Intent;
26 import android.content.pm.PackageManager;
27 import android.content.res.Resources;
28 import android.graphics.Bitmap;
29 import android.graphics.BitmapFactory;
30 import android.os.AsyncTask;
31 import android.os.Build;
32 import android.os.Bundle;
33 import android.support.annotation.MainThread;
34 import android.support.annotation.VisibleForTesting;
35 import android.support.annotation.WorkerThread;
36 import android.support.v4.app.NotificationCompat;
37 import android.text.TextUtils;
38 import android.util.Log;
39 import android.widget.Toast;
40 import com.android.tv.common.SoftPreconditions;
41 import com.android.tv.common.feature.CommonFeatures;
42 import com.android.tv.common.ui.setup.SetupActivity;
43 import com.android.tv.common.ui.setup.SetupFragment;
44 import com.android.tv.common.ui.setup.SetupMultiPaneFragment;
45 import com.android.tv.common.util.AutoCloseableUtils;
46 import com.android.tv.common.util.PostalCodeUtils;
47 import com.android.tv.tuner.R;
48 import com.android.tv.tuner.api.Tuner;
49 import com.android.tv.tuner.api.TunerFactory;
50 import com.android.tv.tuner.prefs.TunerPreferences;
51 import java.util.concurrent.Executor;
52 import javax.inject.Inject;
53 
54 /** The base setup activity class for tuner. */
55 public abstract class BaseTunerSetupActivity extends SetupActivity {
56     private static final String TAG = "BaseTunerSetupActivity";
57     private static final boolean DEBUG = false;
58 
59     /** Key for passing tuner type to sub-fragments. */
60     public static final String KEY_TUNER_TYPE = "TunerSetupActivity.tunerType";
61 
62     // For the notification.
63     protected static final String TUNER_SET_UP_NOTIFICATION_CHANNEL_ID = "tuner_setup_channel";
64     protected static final String NOTIFY_TAG = "TunerSetup";
65     protected static final int NOTIFY_ID = 1000;
66     protected static final String TAG_DRAWABLE = "drawable";
67     protected static final String TAG_ICON = "ic_launcher_s";
68     protected static final int PERMISSIONS_REQUEST_ACCESS_COARSE_LOCATION = 1;
69 
70     protected static final int[] CHANNEL_MAP_SCAN_FILE = {
71         R.raw.ut_us_atsc_center_frequencies_8vsb,
72         R.raw.ut_us_cable_standard_center_frequencies_qam256,
73         R.raw.ut_us_all,
74         R.raw.ut_kr_atsc_center_frequencies_8vsb,
75         R.raw.ut_kr_cable_standard_center_frequencies_qam256,
76         R.raw.ut_kr_all,
77         R.raw.ut_kr_dev_cj_cable_center_frequencies_qam256,
78         R.raw.ut_euro_dvbt_all
79         /* these two resource files are obsolete and removed, so comment them out
80         R.raw.ut_euro_all,
81         R.raw.ut_euro_all */
82     };
83 
84     protected final String mInputId;
85 
86     protected ScanFragment mLastScanFragment;
87     protected Integer mTunerType;
88     protected boolean mNeedToShowPostalCodeFragment;
89     protected String mPreviousPostalCode;
90     protected boolean mActivityStopped;
91     protected boolean mPendingShowInitialFragment;
92     @Inject protected TunerFactory mTunerFactory;
93 
94     private TunerHalCreator mTunerHalCreator;
95 
BaseTunerSetupActivity(String mInputId)96     protected BaseTunerSetupActivity(String mInputId) {
97         this.mInputId = mInputId;
98     }
99 
100     @Override
onCreate(Bundle savedInstanceState)101     protected void onCreate(Bundle savedInstanceState) {
102         if (DEBUG) {
103             Log.d(TAG, "onCreate");
104         }
105         super.onCreate(savedInstanceState);
106         mActivityStopped = false;
107         executeGetTunerTypeAndCountAsyncTask();
108         mTunerHalCreator =
109                 new TunerHalCreator(
110                         getApplicationContext(), AsyncTask.THREAD_POOL_EXECUTOR, mTunerFactory);
111         try {
112             // Updating postal code takes time, therefore we called it here for "warm-up".
113             mPreviousPostalCode = PostalCodeUtils.getLastPostalCode(this);
114             PostalCodeUtils.setLastPostalCode(this, null);
115             PostalCodeUtils.updatePostalCode(this);
116         } catch (Exception e) {
117             // Do nothing. If the last known postal code is null, we'll show guided fragment to
118             // prompt users to input postal code before ConnectionTypeFragment is shown.
119             Log.i(TAG, "Can't get postal code:" + e);
120         }
121     }
122 
executeGetTunerTypeAndCountAsyncTask()123     protected void executeGetTunerTypeAndCountAsyncTask() {}
124 
125     @Override
onStop()126     protected void onStop() {
127         mActivityStopped = true;
128         super.onStop();
129     }
130 
131     @Override
onResume()132     protected void onResume() {
133         super.onResume();
134         mActivityStopped = false;
135         if (mPendingShowInitialFragment) {
136             showInitialFragment();
137             mPendingShowInitialFragment = false;
138         }
139     }
140 
141     @Override
onCreateInitialFragment()142     protected Fragment onCreateInitialFragment() {
143         if (mTunerType != null) {
144             SetupFragment fragment = new WelcomeFragment();
145             Bundle args = new Bundle();
146             args.putInt(KEY_TUNER_TYPE, mTunerType);
147             fragment.setArguments(args);
148             fragment.setShortDistance(
149                     SetupFragment.FRAGMENT_EXIT_TRANSITION
150                             | SetupFragment.FRAGMENT_REENTER_TRANSITION);
151             return fragment;
152         } else {
153             return null;
154         }
155     }
156 
157     @Override
executeAction(String category, int actionId, Bundle params)158     protected boolean executeAction(String category, int actionId, Bundle params) {
159         switch (category) {
160             case WelcomeFragment.ACTION_CATEGORY:
161                 switch (actionId) {
162                     case SetupMultiPaneFragment.ACTION_DONE:
163                         // If the scan was performed, then the result should be OK.
164                         setResult(mLastScanFragment == null ? RESULT_CANCELED : RESULT_OK);
165                         finish();
166                         break;
167                     default:
168                         String postalCode = PostalCodeUtils.getLastPostalCode(this);
169                         boolean needLocation =
170                                 CommonFeatures.ENABLE_CLOUD_EPG_REGION.isEnabled(
171                                                 getApplicationContext())
172                                         && TextUtils.isEmpty(postalCode);
173                         if (needLocation
174                                 && checkSelfPermission(
175                                                 android.Manifest.permission.ACCESS_COARSE_LOCATION)
176                                         != PackageManager.PERMISSION_GRANTED) {
177                             showLocationFragment();
178                         } else if (mNeedToShowPostalCodeFragment || needLocation) {
179                             // We cannot get postal code automatically. Postal code input fragment
180                             // should always be shown even if users have input some valid postal
181                             // code in this activity before.
182                             mNeedToShowPostalCodeFragment = true;
183                             showPostalCodeFragment();
184                         } else {
185                             showConnectionTypeFragment();
186                         }
187                         break;
188                 }
189                 return true;
190             case LocationFragment.ACTION_CATEGORY:
191                 switch (actionId) {
192                     case LocationFragment.ACTION_ALLOW_PERMISSION:
193                         String postalCode =
194                                 params == null
195                                         ? null
196                                         : params.getString(LocationFragment.KEY_POSTAL_CODE);
197                         if (postalCode == null) {
198                             showPostalCodeFragment();
199                         } else {
200                             showConnectionTypeFragment();
201                         }
202                         break;
203                     default:
204                         showConnectionTypeFragment();
205                 }
206                 return true;
207             case PostalCodeFragment.ACTION_CATEGORY:
208                 switch (actionId) {
209                     case SetupMultiPaneFragment.ACTION_DONE:
210                         // fall through
211                     case SetupMultiPaneFragment.ACTION_SKIP:
212                         showConnectionTypeFragment();
213                         break;
214                     default: // fall out
215                 }
216                 return true;
217             case ConnectionTypeFragment.ACTION_CATEGORY:
218                 if (mTunerHalCreator.getOrCreate() == null) {
219                     finish();
220                     Toast.makeText(
221                                     getApplicationContext(),
222                                     R.string.ut_channel_scan_tuner_unavailable,
223                                     Toast.LENGTH_LONG)
224                             .show();
225                     return true;
226                 }
227                 mLastScanFragment = new ScanFragment();
228                 Bundle args1 = new Bundle();
229                 args1.putInt(
230                         ScanFragment.EXTRA_FOR_CHANNEL_SCAN_FILE, CHANNEL_MAP_SCAN_FILE[actionId]);
231                 args1.putInt(KEY_TUNER_TYPE, mTunerType);
232                 args1.putString(ScanFragment.EXTRA_FOR_INPUT_ID, mInputId);
233                 mLastScanFragment.setArguments(args1);
234                 showFragment(mLastScanFragment, true);
235                 return true;
236             case ScanFragment.ACTION_CATEGORY:
237                 switch (actionId) {
238                     case ScanFragment.ACTION_CANCEL:
239                         getFragmentManager().popBackStack();
240                         return true;
241                     case ScanFragment.ACTION_FINISH:
242                         mTunerHalCreator.clear();
243                         showScanResultFragment();
244                         return true;
245                     default: // fall out
246                 }
247                 break;
248             case ScanResultFragment.ACTION_CATEGORY:
249                 switch (actionId) {
250                     case SetupMultiPaneFragment.ACTION_DONE:
251                         setResult(RESULT_OK);
252                         finish();
253                         break;
254                     default:
255                         // scan again
256                         SetupFragment fragment = new ConnectionTypeFragment();
257                         fragment.setShortDistance(
258                                 SetupFragment.FRAGMENT_ENTER_TRANSITION
259                                         | SetupFragment.FRAGMENT_RETURN_TRANSITION);
260                         showFragment(fragment, true);
261                         break;
262                 }
263                 return true;
264             default: // fall out
265         }
266         return false;
267     }
268 
269     @Override
onDestroy()270     public void onDestroy() {
271         if (mPreviousPostalCode != null && PostalCodeUtils.getLastPostalCode(this) == null) {
272             PostalCodeUtils.setLastPostalCode(this, mPreviousPostalCode);
273         }
274         super.onDestroy();
275     }
276 
277     /** Gets the currently used tuner HAL. */
getTunerHal()278     Tuner getTunerHal() {
279         return mTunerHalCreator.getOrCreate();
280     }
281 
282     /** Generates tuner HAL. */
generateTunerHal()283     void generateTunerHal() {
284         mTunerHalCreator.generate();
285     }
286 
287     /** Clears the currently used tuner HAL. */
clearTunerHal()288     protected void clearTunerHal() {
289         mTunerHalCreator.clear();
290     }
291 
showLocationFragment()292     protected void showLocationFragment() {
293         SetupFragment fragment = new LocationFragment();
294         fragment.setShortDistance(
295                 SetupFragment.FRAGMENT_ENTER_TRANSITION | SetupFragment.FRAGMENT_RETURN_TRANSITION);
296         showFragment(fragment, true);
297     }
298 
showPostalCodeFragment()299     protected void showPostalCodeFragment() {
300         showPostalCodeFragment(null);
301     }
302 
showPostalCodeFragment(Bundle args)303     protected void showPostalCodeFragment(Bundle args) {
304         SetupFragment fragment = new PostalCodeFragment();
305         if (args != null) {
306             fragment.setArguments(args);
307         }
308         fragment.setShortDistance(
309                 SetupFragment.FRAGMENT_ENTER_TRANSITION | SetupFragment.FRAGMENT_RETURN_TRANSITION);
310         showFragment(fragment, true);
311     }
312 
showConnectionTypeFragment()313     protected void showConnectionTypeFragment() {
314         SetupFragment fragment = new ConnectionTypeFragment();
315         fragment.setShortDistance(
316                 SetupFragment.FRAGMENT_ENTER_TRANSITION | SetupFragment.FRAGMENT_RETURN_TRANSITION);
317         showFragment(fragment, true);
318     }
319 
showScanResultFragment()320     protected void showScanResultFragment() {
321         SetupFragment scanResultFragment = new ScanResultFragment();
322         Bundle args2 = new Bundle();
323         args2.putInt(KEY_TUNER_TYPE, mTunerType);
324         scanResultFragment.setShortDistance(
325                 SetupFragment.FRAGMENT_EXIT_TRANSITION | SetupFragment.FRAGMENT_REENTER_TRANSITION);
326         showFragment(scanResultFragment, true);
327     }
328 
329     /**
330      * Cancels the previously shown notification.
331      *
332      * @param context a {@link Context} instance
333      */
cancelNotification(Context context)334     public static void cancelNotification(Context context) {
335         NotificationManager notificationManager =
336                 (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
337         notificationManager.cancel(NOTIFY_TAG, NOTIFY_ID);
338     }
339 
340     /**
341      * A callback to be invoked when the TvInputService is enabled or disabled.
342      *
343      * @param tunerSetupIntent
344      * @param context a {@link Context} instance
345      * @param enabled {@code true} for the {@link TunerTvInputService} to be enabled; otherwise
346      *     {@code false}
347      */
onTvInputEnabled( Context context, boolean enabled, Integer tunerType, Intent tunerSetupIntent)348     public static void onTvInputEnabled(
349             Context context, boolean enabled, Integer tunerType, Intent tunerSetupIntent) {
350         // Send a notification for tuner setup if there's no channels and the tuner TV input
351         // setup has been not done.
352         boolean channelScanDoneOnPreference = TunerPreferences.isScanDone(context);
353         int channelCountOnPreference = TunerPreferences.getScannedChannelCount(context);
354         if (enabled && !channelScanDoneOnPreference && channelCountOnPreference == 0) {
355             TunerPreferences.setShouldShowSetupActivity(context, true);
356             sendNotification(context, tunerType, tunerSetupIntent);
357         } else {
358             TunerPreferences.setShouldShowSetupActivity(context, false);
359             cancelNotification(context);
360         }
361     }
362 
sendNotification( Context context, Integer tunerType, Intent tunerSetupIntent)363     private static void sendNotification(
364             Context context, Integer tunerType, Intent tunerSetupIntent) {
365         SoftPreconditions.checkState(
366                 tunerType != null, TAG, "tunerType is null when send notification");
367         if (tunerType == null) {
368             return;
369         }
370         Resources resources = context.getResources();
371         String contentTitle = resources.getString(R.string.ut_setup_notification_content_title);
372         int contentTextId = 0;
373         switch (tunerType) {
374             case Tuner.TUNER_TYPE_BUILT_IN:
375                 contentTextId = R.string.bt_setup_notification_content_text;
376                 break;
377             case Tuner.TUNER_TYPE_USB:
378                 contentTextId = R.string.ut_setup_notification_content_text;
379                 break;
380             case Tuner.TUNER_TYPE_NETWORK:
381                 contentTextId = R.string.nt_setup_notification_content_text;
382                 break;
383             default: // fall out
384         }
385         String contentText = resources.getString(contentTextId);
386         if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
387             sendNotificationInternal(context, contentTitle, contentText, tunerSetupIntent);
388         } else {
389             Bitmap largeIcon =
390                     BitmapFactory.decodeResource(resources, R.drawable.recommendation_antenna);
391             sendRecommendationCard(context, contentTitle, contentText, largeIcon, tunerSetupIntent);
392         }
393     }
394 
sendNotificationInternal( Context context, String contentTitle, String contentText, Intent tunerSetupIntent)395     private static void sendNotificationInternal(
396             Context context, String contentTitle, String contentText, Intent tunerSetupIntent) {
397         NotificationManager notificationManager =
398                 (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
399         notificationManager.createNotificationChannel(
400                 new NotificationChannel(
401                         TUNER_SET_UP_NOTIFICATION_CHANNEL_ID,
402                         context.getResources()
403                                 .getString(R.string.ut_setup_notification_channel_name),
404                         NotificationManager.IMPORTANCE_HIGH));
405         Notification notification =
406                 new Notification.Builder(context, TUNER_SET_UP_NOTIFICATION_CHANNEL_ID)
407                         .setContentTitle(contentTitle)
408                         .setContentText(contentText)
409                         .setSmallIcon(
410                                 context.getResources()
411                                         .getIdentifier(
412                                                 TAG_ICON, TAG_DRAWABLE, context.getPackageName()))
413                         .setContentIntent(
414                                 createPendingIntentForSetupActivity(context, tunerSetupIntent))
415                         .setVisibility(Notification.VISIBILITY_PUBLIC)
416                         .extend(new Notification.TvExtender())
417                         .build();
418         notificationManager.notify(NOTIFY_TAG, NOTIFY_ID, notification);
419     }
420 
421     /**
422      * Sends the recommendation card to start the tuner TV input setup activity.
423      *
424      * @param tunerSetupIntent
425      * @param context a {@link Context} instance
426      */
sendRecommendationCard( Context context, String contentTitle, String contentText, Bitmap largeIcon, Intent tunerSetupIntent)427     private static void sendRecommendationCard(
428             Context context,
429             String contentTitle,
430             String contentText,
431             Bitmap largeIcon,
432             Intent tunerSetupIntent) {
433         // Build and send the notification.
434         Notification notification =
435                 new NotificationCompat.BigPictureStyle(
436                                 new NotificationCompat.Builder(context)
437                                         .setAutoCancel(false)
438                                         .setContentTitle(contentTitle)
439                                         .setContentText(contentText)
440                                         .setContentInfo(contentText)
441                                         .setCategory(Notification.CATEGORY_RECOMMENDATION)
442                                         .setLargeIcon(largeIcon)
443                                         .setSmallIcon(
444                                                 context.getResources()
445                                                         .getIdentifier(
446                                                                 TAG_ICON,
447                                                                 TAG_DRAWABLE,
448                                                                 context.getPackageName()))
449                                         .setContentIntent(
450                                                 createPendingIntentForSetupActivity(
451                                                         context, tunerSetupIntent)))
452                         .build();
453         NotificationManager notificationManager =
454                 (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
455         notificationManager.notify(NOTIFY_TAG, NOTIFY_ID, notification);
456     }
457 
458     /**
459      * Returns a {@link PendingIntent} to launch the tuner TV input service.
460      *
461      * @param context a {@link Context} instance
462      * @param tunerSetupIntent
463      */
createPendingIntentForSetupActivity( Context context, Intent tunerSetupIntent)464     private static PendingIntent createPendingIntentForSetupActivity(
465             Context context, Intent tunerSetupIntent) {
466         return PendingIntent.getActivity(context, 0, tunerSetupIntent,
467                     PendingIntent.FLAG_UPDATE_CURRENT|PendingIntent.FLAG_IMMUTABLE);
468     }
469 
470     /** Creates {@link Tuner} instances in a worker thread * */
471     @VisibleForTesting
472     protected static class TunerHalCreator {
473         private Context mContext;
474         @VisibleForTesting Tuner mTunerHal;
475         private TunerHalCreator.GenerateTunerHalTask mGenerateTunerHalTask;
476         private final Executor mExecutor;
477         private final TunerFactory mTunerFactory;
478 
TunerHalCreator(Context context, Executor executor, TunerFactory tunerFactory)479         TunerHalCreator(Context context, Executor executor, TunerFactory tunerFactory) {
480             mContext = context;
481             mExecutor = executor;
482             mTunerFactory = tunerFactory;
483         }
484 
485         /**
486          * Returns tuner HAL currently used. If it's {@code null} and tuner HAL is not generated
487          * before, tries to generate it synchronously.
488          */
489         @WorkerThread
getOrCreate()490         Tuner getOrCreate() {
491             if (mGenerateTunerHalTask != null
492                     && mGenerateTunerHalTask.getStatus() != AsyncTask.Status.FINISHED) {
493                 try {
494                     return mGenerateTunerHalTask.get();
495                 } catch (Exception e) {
496                     Log.e(TAG, "Cannot get Tuner HAL: " + e);
497                 }
498             } else if (mGenerateTunerHalTask == null && mTunerHal == null) {
499                 mTunerHal = createInstance();
500             }
501             return mTunerHal;
502         }
503 
504         /** Generates tuner hal for scanning with asynchronous tasks. */
505         @MainThread
generate()506         void generate() {
507             if (mGenerateTunerHalTask == null && mTunerHal == null) {
508                 mGenerateTunerHalTask = new TunerHalCreator.GenerateTunerHalTask();
509                 mGenerateTunerHalTask.executeOnExecutor(mExecutor);
510             }
511         }
512 
513         /** Clears the currently used tuner hal. */
514         @MainThread
clear()515         void clear() {
516             if (mGenerateTunerHalTask != null) {
517                 mGenerateTunerHalTask.cancel(true);
518                 mGenerateTunerHalTask = null;
519             }
520             if (mTunerHal != null) {
521                 AutoCloseableUtils.closeQuietly(mTunerHal);
522                 mTunerHal = null;
523             }
524         }
525 
526         @WorkerThread
createInstance()527         protected Tuner createInstance() {
528             return mTunerFactory.createInstance(mContext);
529         }
530 
531         class GenerateTunerHalTask extends AsyncTask<Void, Void, Tuner> {
532             @Override
doInBackground(Void... args)533             protected Tuner doInBackground(Void... args) {
534                 return createInstance();
535             }
536 
537             @Override
onPostExecute(Tuner tunerHal)538             protected void onPostExecute(Tuner tunerHal) {
539                 mTunerHal = tunerHal;
540             }
541         }
542     }
543 }
544