• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright 2014, 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.server.telecom;
18 
19 import android.media.Ringtone;
20 import android.net.Uri;
21 import android.os.Handler;
22 import android.os.HandlerThread;
23 import android.os.Message;
24 
25 import com.android.internal.annotations.VisibleForTesting;
26 import com.android.internal.os.SomeArgs;
27 import com.android.internal.util.Preconditions;
28 
29 /**
30  * Plays the default ringtone. Uses {@link Ringtone} in a separate thread so that this class can be
31  * used from the main thread.
32  */
33 @VisibleForTesting
34 public class AsyncRingtonePlayer {
35     // Message codes used with the ringtone thread.
36     private static final int EVENT_PLAY = 1;
37     private static final int EVENT_STOP = 2;
38     private static final int EVENT_REPEAT = 3;
39 
40     // The interval in which to restart the ringer.
41     private static final int RESTART_RINGER_MILLIS = 3000;
42 
43     /** Handler running on the ringtone thread. */
44     private Handler mHandler;
45 
46     /** The current ringtone. Only used by the ringtone thread. */
47     private Ringtone mRingtone;
48 
49     /** Plays the ringtone. */
play(RingtoneFactory factory, Call incomingCall)50     public void play(RingtoneFactory factory, Call incomingCall) {
51         Log.d(this, "Posting play.");
52         SomeArgs args = SomeArgs.obtain();
53         args.arg1 = factory;
54         args.arg2 = incomingCall;
55         postMessage(EVENT_PLAY, true /* shouldCreateHandler */, args);
56     }
57 
58     /** Stops playing the ringtone. */
stop()59     public void stop() {
60         Log.d(this, "Posting stop.");
61         postMessage(EVENT_STOP, false /* shouldCreateHandler */, null);
62     }
63 
64     /**
65      * Posts a message to the ringtone-thread handler. Creates the handler if specified by the
66      * parameter shouldCreateHandler.
67      *
68      * @param messageCode The message to post.
69      * @param shouldCreateHandler True when a handler should be created to handle this message.
70      */
postMessage(int messageCode, boolean shouldCreateHandler, SomeArgs args)71     private void postMessage(int messageCode, boolean shouldCreateHandler, SomeArgs args) {
72         synchronized(this) {
73             if (mHandler == null && shouldCreateHandler) {
74                 mHandler = getNewHandler();
75             }
76 
77             if (mHandler == null) {
78                 Log.d(this, "Message %d skipped because there is no handler.", messageCode);
79             } else {
80                 mHandler.obtainMessage(messageCode, args).sendToTarget();
81             }
82         }
83     }
84 
85     /**
86      * Creates a new ringtone Handler running in its own thread.
87      */
getNewHandler()88     private Handler getNewHandler() {
89         Preconditions.checkState(mHandler == null);
90 
91         HandlerThread thread = new HandlerThread("ringtone-player");
92         thread.start();
93 
94         return new Handler(thread.getLooper()) {
95             @Override
96             public void handleMessage(Message msg) {
97                 switch(msg.what) {
98                     case EVENT_PLAY:
99                         handlePlay((SomeArgs) msg.obj);
100                         break;
101                     case EVENT_REPEAT:
102                         handleRepeat();
103                         break;
104                     case EVENT_STOP:
105                         handleStop();
106                         break;
107                 }
108             }
109         };
110     }
111 
112     /**
113      * Starts the actual playback of the ringtone. Executes on ringtone-thread.
114      */
115     private void handlePlay(SomeArgs args) {
116         RingtoneFactory factory = (RingtoneFactory) args.arg1;
117         Call incomingCall = (Call) args.arg2;
118         args.recycle();
119         // don't bother with any of this if there is an EVENT_STOP waiting.
120         if (mHandler.hasMessages(EVENT_STOP)) {
121             return;
122         }
123 
124         // If the Ringtone Uri is EMPTY, then the "None" Ringtone has been selected. Do not play
125         // anything.
126         if(Uri.EMPTY.equals(incomingCall.getRingtone())) {
127             mRingtone = null;
128             return;
129         }
130 
131         ThreadUtil.checkNotOnMainThread();
132         Log.i(this, "Play ringtone.");
133 
134         if (mRingtone == null) {
135             mRingtone = factory.getRingtone(incomingCall);
136             if (mRingtone == null) {
137                 Uri ringtoneUri = incomingCall.getRingtone();
138                 String ringtoneUriString = (ringtoneUri == null) ? "null" :
139                         ringtoneUri.toSafeString();
140                 Log.event(null, Log.Events.ERROR_LOG, "Failed to get ringtone from factory. " +
141                         "Skipping ringing. Uri was: " + ringtoneUriString);
142                 return;
143             }
144         }
145 
146         handleRepeat();
147     }
148 
149     private void handleRepeat() {
150         if (mRingtone == null) {
151             return;
152         }
153 
154         if (mRingtone.isPlaying()) {
155             Log.d(this, "Ringtone already playing.");
156         } else {
157             mRingtone.play();
158             Log.i(this, "Repeat ringtone.");
159         }
160 
161         // Repost event to restart ringer in {@link RESTART_RINGER_MILLIS}.
162         synchronized(this) {
163             if (!mHandler.hasMessages(EVENT_REPEAT)) {
164                 mHandler.sendEmptyMessageDelayed(EVENT_REPEAT, RESTART_RINGER_MILLIS);
165             }
166         }
167     }
168 
169     /**
170      * Stops the playback of the ringtone. Executes on the ringtone-thread.
171      */
172     private void handleStop() {
173         ThreadUtil.checkNotOnMainThread();
174         Log.i(this, "Stop ringtone.");
175 
176         if (mRingtone != null) {
177             Log.d(this, "Ringtone.stop() invoked.");
178             mRingtone.stop();
179             mRingtone = null;
180         }
181 
182         synchronized(this) {
183             // At the time that STOP is handled, there should be no need for repeat messages in the
184             // queue.
185             mHandler.removeMessages(EVENT_REPEAT);
186 
187             if (mHandler.hasMessages(EVENT_PLAY)) {
188                 Log.v(this, "Keeping alive ringtone thread for subsequent play request.");
189             } else {
190                 mHandler.removeMessages(EVENT_STOP);
191                 mHandler.getLooper().quitSafely();
192                 mHandler = null;
193                 Log.v(this, "Handler cleared.");
194             }
195         }
196     }
197 }
198