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; 18 19 import android.app.Activity; 20 import android.content.ActivityNotFoundException; 21 import android.content.ComponentName; 22 import android.content.Intent; 23 import android.media.tv.TvInputInfo; 24 import android.os.Bundle; 25 import android.os.Handler; 26 import android.os.Looper; 27 import android.support.annotation.MainThread; 28 import android.util.Log; 29 30 import com.android.tv.common.CommonConstants; 31 import com.android.tv.common.SoftPreconditions; 32 import com.android.tv.common.actions.InputSetupActionUtils; 33 import com.android.tv.data.ChannelDataManager; 34 import com.android.tv.data.ChannelDataManager.Listener; 35 import com.android.tv.data.epg.EpgFetcher; 36 import com.android.tv.data.epg.EpgInputWhiteList; 37 import com.android.tv.features.TvFeatures; 38 import com.android.tv.util.SetupUtils; 39 import com.android.tv.util.TvInputManagerHelper; 40 import com.android.tv.util.Utils; 41 42 import com.google.android.tv.partner.support.EpgContract; 43 44 import dagger.android.AndroidInjection; 45 import dagger.android.ContributesAndroidInjector; 46 47 import java.util.concurrent.TimeUnit; 48 49 import javax.inject.Inject; 50 51 /** 52 * An activity to launch a TV input setup activity. 53 * 54 * <p>After setup activity is finished, all channels will be browsable. 55 */ 56 public class SetupPassthroughActivity extends Activity { 57 private static final String TAG = "SetupPassthroughAct"; 58 private static final boolean DEBUG = false; 59 60 private static final int REQUEST_START_SETUP_ACTIVITY = 200; 61 62 private static ScanTimeoutMonitor sScanTimeoutMonitor; 63 64 private TvInputInfo mTvInputInfo; 65 private Intent mActivityAfterCompletion; 66 private boolean mEpgFetcherDuringScan; 67 @Inject EpgInputWhiteList mEpgInputWhiteList; 68 @Inject TvInputManagerHelper mInputManager; 69 @Inject SetupUtils mSetupUtils; 70 @Inject ChannelDataManager mChannelDataManager; 71 @Inject EpgFetcher mEpgFetcher; 72 73 @Override onCreate(Bundle savedInstanceState)74 public void onCreate(Bundle savedInstanceState) { 75 if (DEBUG) Log.d(TAG, "onCreate"); 76 AndroidInjection.inject(this); 77 super.onCreate(savedInstanceState); 78 Intent intent = getIntent(); 79 String inputId = intent.getStringExtra(InputSetupActionUtils.EXTRA_INPUT_ID); 80 mTvInputInfo = mInputManager.getTvInputInfo(inputId); 81 mActivityAfterCompletion = InputSetupActionUtils.getExtraActivityAfter(intent); 82 boolean needToFetchEpg = 83 mTvInputInfo != null && Utils.isInternalTvInput(this, mTvInputInfo.getId()); 84 if (needToFetchEpg) { 85 // In case when the activity is restored, this flag should be restored as well. 86 mEpgFetcherDuringScan = true; 87 } 88 if (savedInstanceState == null) { 89 SoftPreconditions.checkArgument( 90 InputSetupActionUtils.hasInputSetupAction(intent), 91 TAG, 92 "Unsupported action %s", 93 intent.getAction()); 94 if (DEBUG) Log.d(TAG, "TvInputId " + inputId + " / TvInputInfo " + mTvInputInfo); 95 if (mTvInputInfo == null) { 96 Log.w(TAG, "There is no input with the ID " + inputId + "."); 97 finish(); 98 return; 99 } 100 if (intent.getExtras() == null) { 101 Log.w(TAG, "There is no extra info in the intent"); 102 finish(); 103 return; 104 } 105 Intent setupIntent = InputSetupActionUtils.getExtraSetupIntent(intent); 106 if (DEBUG) Log.d(TAG, "Setup activity launch intent: " + setupIntent); 107 if (setupIntent == null) { 108 Log.w(TAG, "The input (" + mTvInputInfo.getId() + ") doesn't have setup."); 109 finish(); 110 return; 111 } 112 if (DEBUG) Log.d(TAG, "Activity after completion " + mActivityAfterCompletion); 113 // If EXTRA_SETUP_INTENT is not removed, an infinite recursion happens during 114 // setupIntent.putExtras(intent.getExtras()). 115 Bundle extras = intent.getExtras(); 116 InputSetupActionUtils.removeSetupIntent(extras); 117 setupIntent.putExtras(extras); 118 try { 119 ComponentName callingActivity = getCallingActivity(); 120 if (callingActivity == null 121 || !callingActivity.getPackageName().equals(CommonConstants.BASE_PACKAGE)) { 122 String name = 123 callingActivity == null ? "null" : callingActivity.getPackageName(); 124 Log.w(TAG, 125 "Calling activity " + name + " is not trusted. Not forwarding intent."); 126 finish(); 127 return; 128 } 129 SetupUtils.grantEpgPermission(this, mTvInputInfo.getServiceInfo().packageName); 130 startActivityForResult(setupIntent, REQUEST_START_SETUP_ACTIVITY); 131 } catch (ActivityNotFoundException e) { 132 Log.e(TAG, "Can't find activity: " + setupIntent.getComponent()); 133 finish(); 134 return; 135 } 136 if (needToFetchEpg) { 137 if (sScanTimeoutMonitor == null) { 138 sScanTimeoutMonitor = new ScanTimeoutMonitor(mEpgFetcher, mChannelDataManager); 139 } 140 sScanTimeoutMonitor.startMonitoring(); 141 mEpgFetcher.onChannelScanStarted(); 142 } 143 } 144 } 145 146 @Override onActivityResult(int requestCode, final int resultCode, final Intent data)147 public void onActivityResult(int requestCode, final int resultCode, final Intent data) { 148 if (DEBUG) 149 Log.d(TAG, "onActivityResult(" + requestCode + ", " + resultCode + ", " + data + ")"); 150 if (sScanTimeoutMonitor != null) { 151 sScanTimeoutMonitor.stopMonitoring(); 152 } 153 // Note: It's not guaranteed that this method is always called after scanning. 154 boolean setupComplete = 155 requestCode == REQUEST_START_SETUP_ACTIVITY && resultCode == Activity.RESULT_OK; 156 // Tells EpgFetcher that channel source setup is finished. 157 158 if (mEpgFetcherDuringScan) { 159 mEpgFetcher.onChannelScanFinished(); 160 } 161 if (!setupComplete) { 162 setResult(resultCode, data); 163 finish(); 164 return; 165 } 166 if (TvFeatures.CLOUD_EPG_FOR_3RD_PARTY.isEnabled(this) 167 && data != null 168 && data.getBooleanExtra(EpgContract.EXTRA_USE_CLOUD_EPG, false)) { 169 if (DEBUG) Log.d(TAG, "extra " + data.getExtras()); 170 String inputId = data.getStringExtra(TvInputInfo.EXTRA_INPUT_ID); 171 if (mEpgInputWhiteList.isInputWhiteListed(inputId)) { 172 mEpgFetcher.fetchImmediately(); 173 } 174 } 175 176 if (mTvInputInfo == null) { 177 Log.w( 178 TAG, 179 "There is no input with ID " 180 + getIntent().getStringExtra(InputSetupActionUtils.EXTRA_INPUT_ID) 181 + "."); 182 setResult(resultCode, data); 183 finish(); 184 return; 185 } 186 mSetupUtils.onTvInputSetupFinished( 187 mTvInputInfo.getId(), 188 () -> { 189 if (mActivityAfterCompletion != null) { 190 try { 191 startActivity(mActivityAfterCompletion); 192 } catch (ActivityNotFoundException e) { 193 Log.w(TAG, "Activity launch failed", e); 194 } 195 } 196 setResult(resultCode, data); 197 finish(); 198 }); 199 } 200 201 /** 202 * Monitors the scan progress and notifies the timeout of the scanning. The purpose of this 203 * monitor is to call EpgFetcher.onChannelScanFinished() in case when 204 * SetupPassthroughActivity.onActivityResult() is not called properly. b/36008534 205 */ 206 @MainThread 207 private static class ScanTimeoutMonitor { 208 // Set timeout long enough. The message in Sony TV says the scanning takes about 30 minutes. 209 private static final long SCAN_TIMEOUT_MS = TimeUnit.MINUTES.toMillis(30); 210 211 private final EpgFetcher mEpgFetcher; 212 private final ChannelDataManager mChannelDataManager; 213 private final Handler mHandler = new Handler(Looper.getMainLooper()); 214 private final Runnable mScanTimeoutRunnable = 215 () -> { 216 Log.w( 217 TAG, 218 "No channels has been added for a while." 219 + " The scan might have finished unexpectedly."); 220 onScanTimedOut(); 221 }; 222 private final Listener mChannelDataManagerListener = 223 new Listener() { 224 @Override 225 public void onLoadFinished() { 226 setupTimer(); 227 } 228 229 @Override 230 public void onChannelListUpdated() { 231 setupTimer(); 232 } 233 234 @Override 235 public void onChannelBrowsableChanged() {} 236 }; 237 private boolean mStarted; 238 ScanTimeoutMonitor(EpgFetcher epgFetcher, ChannelDataManager mChannelDataManager)239 private ScanTimeoutMonitor(EpgFetcher epgFetcher, ChannelDataManager mChannelDataManager) { 240 mEpgFetcher = epgFetcher; 241 this.mChannelDataManager = mChannelDataManager; 242 } 243 startMonitoring()244 private void startMonitoring() { 245 if (!mStarted) { 246 mStarted = true; 247 mChannelDataManager.addListener(mChannelDataManagerListener); 248 } 249 if (mChannelDataManager.isDbLoadFinished()) { 250 setupTimer(); 251 } 252 } 253 stopMonitoring()254 private void stopMonitoring() { 255 if (mStarted) { 256 mStarted = false; 257 mHandler.removeCallbacks(mScanTimeoutRunnable); 258 mChannelDataManager.removeListener(mChannelDataManagerListener); 259 } 260 } 261 setupTimer()262 private void setupTimer() { 263 mHandler.removeCallbacks(mScanTimeoutRunnable); 264 mHandler.postDelayed(mScanTimeoutRunnable, SCAN_TIMEOUT_MS); 265 } 266 onScanTimedOut()267 private void onScanTimedOut() { 268 stopMonitoring(); 269 mEpgFetcher.onChannelScanFinished(); 270 } 271 } 272 273 /** Exports {@link MainActivity} for Dagger codegen to create the appropriate injector. */ 274 @dagger.Module 275 public abstract static class Module { 276 @ContributesAndroidInjector contributesSetupPassthroughActivity()277 abstract SetupPassthroughActivity contributesSetupPassthroughActivity(); 278 } 279 } 280