• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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.dvr.recorder;
18 
19 import android.annotation.TargetApi;
20 import android.content.ContentUris;
21 import android.media.tv.TvContract;
22 import android.net.Uri;
23 import android.os.Build;
24 import android.os.Message;
25 import android.support.annotation.MainThread;
26 import android.support.annotation.NonNull;
27 import android.support.annotation.Nullable;
28 import android.util.ArraySet;
29 import android.util.Log;
30 
31 import com.android.tv.ApplicationSingletons;
32 import com.android.tv.InputSessionManager;
33 import com.android.tv.InputSessionManager.OnTvViewChannelChangeListener;
34 import com.android.tv.MainActivity;
35 import com.android.tv.TvApplication;
36 import com.android.tv.common.WeakHandler;
37 import com.android.tv.data.Channel;
38 import com.android.tv.data.ChannelDataManager;
39 import com.android.tv.dvr.DvrDataManager.ScheduledRecordingListener;
40 import com.android.tv.dvr.DvrScheduleManager;
41 import com.android.tv.dvr.data.ScheduledRecording;
42 import com.android.tv.dvr.ui.DvrUiHelper;
43 
44 import java.util.ArrayList;
45 import java.util.HashMap;
46 import java.util.List;
47 import java.util.Map;
48 import java.util.Set;
49 import java.util.concurrent.TimeUnit;
50 
51 /**
52  * Checking the runtime conflict of DVR recording.
53  * <p>
54  * This class runs only while the {@link MainActivity} is resumed and holds the upcoming conflicts.
55  */
56 @TargetApi(Build.VERSION_CODES.N)
57 @MainThread
58 public class ConflictChecker {
59     private static final String TAG = "ConflictChecker";
60     private static final boolean DEBUG = false;
61 
62     private static final int MSG_CHECK_CONFLICT = 1;
63 
64     private static final long CHECK_RETRY_PERIOD_MS = TimeUnit.SECONDS.toMillis(30);
65 
66     /**
67      * To show watch conflict dialog, the start time of the earliest conflicting schedule should be
68      * less than or equal to this time.
69      */
70     private static final long MAX_WATCH_CONFLICT_CHECK_TIME_MS = TimeUnit.MINUTES.toMillis(5);
71     /**
72      * To show watch conflict dialog, the start time of the earliest conflicting schedule should be
73      * greater than or equal to this time.
74      */
75     private static final long MIN_WATCH_CONFLICT_CHECK_TIME_MS = TimeUnit.SECONDS.toMillis(30);
76 
77     private final MainActivity mMainActivity;
78     private final ChannelDataManager mChannelDataManager;
79     private final DvrScheduleManager mScheduleManager;
80     private final InputSessionManager mSessionManager;
81     private final ConflictCheckerHandler mHandler = new ConflictCheckerHandler(this);
82 
83     private final List<ScheduledRecording> mUpcomingConflicts = new ArrayList<>();
84     private final Set<OnUpcomingConflictChangeListener> mOnUpcomingConflictChangeListeners =
85             new ArraySet<>();
86     private final Map<Long, List<ScheduledRecording>> mCheckedConflictsMap = new HashMap<>();
87 
88     private final ScheduledRecordingListener mScheduledRecordingListener =
89             new ScheduledRecordingListener() {
90         @Override
91         public void onScheduledRecordingAdded(ScheduledRecording... scheduledRecordings) {
92             if (DEBUG) Log.d(TAG, "onScheduledRecordingAdded: " + scheduledRecordings);
93             mHandler.sendEmptyMessage(MSG_CHECK_CONFLICT);
94         }
95 
96         @Override
97         public void onScheduledRecordingRemoved(ScheduledRecording... scheduledRecordings) {
98             if (DEBUG) Log.d(TAG, "onScheduledRecordingRemoved: " + scheduledRecordings);
99             mHandler.sendEmptyMessage(MSG_CHECK_CONFLICT);
100         }
101 
102         @Override
103         public void onScheduledRecordingStatusChanged(ScheduledRecording... scheduledRecordings) {
104             if (DEBUG) Log.d(TAG, "onScheduledRecordingStatusChanged: " + scheduledRecordings);
105             mHandler.sendEmptyMessage(MSG_CHECK_CONFLICT);
106         }
107     };
108 
109     private final OnTvViewChannelChangeListener mOnTvViewChannelChangeListener =
110             new OnTvViewChannelChangeListener() {
111                 @Override
112                 public void onTvViewChannelChange(@Nullable Uri channelUri) {
113                     mHandler.sendEmptyMessage(MSG_CHECK_CONFLICT);
114                 }
115             };
116 
117     private boolean mStarted;
118 
ConflictChecker(MainActivity mainActivity)119     public ConflictChecker(MainActivity mainActivity) {
120         mMainActivity = mainActivity;
121         ApplicationSingletons appSingletons = TvApplication.getSingletons(mainActivity);
122         mChannelDataManager = appSingletons.getChannelDataManager();
123         mScheduleManager = appSingletons.getDvrScheduleManager();
124         mSessionManager = appSingletons.getInputSessionManager();
125     }
126 
127     /**
128      * Starts checking the conflict.
129      */
start()130     public void start() {
131         if (mStarted) {
132             return;
133         }
134         mStarted = true;
135         mHandler.sendEmptyMessage(MSG_CHECK_CONFLICT);
136         mScheduleManager.addScheduledRecordingListener(mScheduledRecordingListener);
137         mSessionManager.addOnTvViewChannelChangeListener(mOnTvViewChannelChangeListener);
138     }
139 
140     /**
141      * Stops checking the conflict.
142      */
stop()143     public void stop() {
144         if (!mStarted) {
145             return;
146         }
147         mStarted = false;
148         mSessionManager.removeOnTvViewChannelChangeListener(mOnTvViewChannelChangeListener);
149         mScheduleManager.removeScheduledRecordingListener(mScheduledRecordingListener);
150         mHandler.removeCallbacksAndMessages(null);
151     }
152 
153     /**
154      * Returns the upcoming conflicts.
155      */
getUpcomingConflicts()156     public List<ScheduledRecording> getUpcomingConflicts() {
157         return new ArrayList<>(mUpcomingConflicts);
158     }
159 
160     /**
161      * Adds a {@link OnUpcomingConflictChangeListener}.
162      */
addOnUpcomingConflictChangeListener(OnUpcomingConflictChangeListener listener)163     public void addOnUpcomingConflictChangeListener(OnUpcomingConflictChangeListener listener) {
164         mOnUpcomingConflictChangeListeners.add(listener);
165     }
166 
167     /**
168      * Removes the {@link OnUpcomingConflictChangeListener}.
169      */
removeOnUpcomingConflictChangeListener(OnUpcomingConflictChangeListener listener)170     public void removeOnUpcomingConflictChangeListener(OnUpcomingConflictChangeListener listener) {
171         mOnUpcomingConflictChangeListeners.remove(listener);
172     }
173 
notifyUpcomingConflictChanged()174     private void notifyUpcomingConflictChanged() {
175         for (OnUpcomingConflictChangeListener l : mOnUpcomingConflictChangeListeners) {
176             l.onUpcomingConflictChange();
177         }
178     }
179 
180     /**
181      * Remembers the user's decision to record while watching the channel.
182      */
setCheckedConflictsForChannel(long mChannelId, List<ScheduledRecording> conflicts)183     public void setCheckedConflictsForChannel(long mChannelId, List<ScheduledRecording> conflicts) {
184         mCheckedConflictsMap.put(mChannelId, new ArrayList<>(conflicts));
185     }
186 
onCheckConflict()187     void onCheckConflict() {
188         // Checks the conflicting schedules and setup the next re-check time.
189         // If there are upcoming conflicts soon, it opens the conflict dialog.
190         if (DEBUG) Log.d(TAG, "Handling MSG_CHECK_CONFLICT");
191         mHandler.removeMessages(MSG_CHECK_CONFLICT);
192         mUpcomingConflicts.clear();
193         if (!mScheduleManager.isInitialized()
194                 || !mChannelDataManager.isDbLoadFinished()) {
195             mHandler.sendEmptyMessageDelayed(MSG_CHECK_CONFLICT, CHECK_RETRY_PERIOD_MS);
196             notifyUpcomingConflictChanged();
197             return;
198         }
199         if (mSessionManager.getCurrentTvViewChannelUri() == null) {
200             // As MainActivity is not using a tuner, no need to check the conflict.
201             notifyUpcomingConflictChanged();
202             return;
203         }
204         Uri channelUri = mSessionManager.getCurrentTvViewChannelUri();
205         if (TvContract.isChannelUriForPassthroughInput(channelUri)) {
206             notifyUpcomingConflictChanged();
207             return;
208         }
209         long channelId = ContentUris.parseId(channelUri);
210         Channel channel = mChannelDataManager.getChannel(channelId);
211         // The conflicts caused by watching the channel.
212         List<ScheduledRecording> conflicts = mScheduleManager
213                 .getConflictingSchedulesForWatching(channel.getId());
214         long earliestToCheck = Long.MAX_VALUE;
215         long currentTimeMs = System.currentTimeMillis();
216         for (ScheduledRecording schedule : conflicts) {
217             long startTimeMs = schedule.getStartTimeMs();
218             if (startTimeMs < currentTimeMs + MIN_WATCH_CONFLICT_CHECK_TIME_MS) {
219                 // The start time of the upcoming conflict remains less than the minimum
220                 // check time.
221                 continue;
222             }
223             if (startTimeMs > currentTimeMs + MAX_WATCH_CONFLICT_CHECK_TIME_MS) {
224                 // The start time of the upcoming conflict remains greater than the
225                 // maximum check time. Setup the next re-check time.
226                 long nextCheckTimeMs = startTimeMs - MAX_WATCH_CONFLICT_CHECK_TIME_MS;
227                 if (earliestToCheck > nextCheckTimeMs) {
228                     earliestToCheck = nextCheckTimeMs;
229                 }
230             } else {
231                 // Found upcoming conflicts which will start soon.
232                 mUpcomingConflicts.add(schedule);
233                 // The schedule will be removed from the "upcoming conflict" when the
234                 // recording is almost started.
235                 long nextCheckTimeMs = startTimeMs - MIN_WATCH_CONFLICT_CHECK_TIME_MS;
236                 if (earliestToCheck > nextCheckTimeMs) {
237                     earliestToCheck = nextCheckTimeMs;
238                 }
239             }
240         }
241         if (earliestToCheck != Long.MAX_VALUE) {
242             mHandler.sendEmptyMessageDelayed(MSG_CHECK_CONFLICT,
243                     earliestToCheck - currentTimeMs);
244         }
245         if (DEBUG) Log.d(TAG, "upcoming conflicts: " + mUpcomingConflicts);
246         notifyUpcomingConflictChanged();
247         if (!mUpcomingConflicts.isEmpty()
248                 && !DvrUiHelper.isChannelWatchConflictDialogShown(mMainActivity)) {
249             // Don't show the conflict dialog if the user already knows.
250             List<ScheduledRecording> checkedConflicts = mCheckedConflictsMap.get(
251                     channel.getId());
252             if (checkedConflicts == null
253                     || !checkedConflicts.containsAll(mUpcomingConflicts)) {
254                 DvrUiHelper.showChannelWatchConflictDialog(mMainActivity, channel);
255             }
256         }
257     }
258 
259     private static class ConflictCheckerHandler extends WeakHandler<ConflictChecker> {
ConflictCheckerHandler(ConflictChecker conflictChecker)260         ConflictCheckerHandler(ConflictChecker conflictChecker) {
261             super(conflictChecker);
262         }
263 
264         @Override
handleMessage(Message msg, @NonNull ConflictChecker conflictChecker)265         protected void handleMessage(Message msg, @NonNull ConflictChecker conflictChecker) {
266             switch (msg.what) {
267                 case MSG_CHECK_CONFLICT:
268                     conflictChecker.onCheckConflict();
269                     break;
270             }
271         }
272     }
273 
274     /**
275      * A listener for the change of upcoming conflicts.
276      */
277     public interface OnUpcomingConflictChangeListener {
onUpcomingConflictChange()278         void onUpcomingConflictChange();
279     }
280 }
281