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.content.Context; 20 import android.content.Intent; 21 import android.media.AudioAttributes; 22 import android.media.session.MediaSession; 23 import android.os.Handler; 24 import android.os.Looper; 25 import android.os.Message; 26 import android.telecom.Log; 27 import android.view.KeyEvent; 28 29 import com.android.internal.annotations.VisibleForTesting; 30 31 /** 32 * Static class to handle listening to the headset media buttons. 33 */ 34 public class HeadsetMediaButton extends CallsManagerListenerBase { 35 36 // Types of media button presses 37 @VisibleForTesting 38 public static final int SHORT_PRESS = 1; 39 @VisibleForTesting 40 public static final int LONG_PRESS = 2; 41 42 private static final AudioAttributes AUDIO_ATTRIBUTES = new AudioAttributes.Builder() 43 .setContentType(AudioAttributes.CONTENT_TYPE_SPEECH) 44 .setUsage(AudioAttributes.USAGE_VOICE_COMMUNICATION).build(); 45 46 private static final int MSG_MEDIA_SESSION_INITIALIZE = 0; 47 private static final int MSG_MEDIA_SESSION_SET_ACTIVE = 1; 48 49 private final MediaSession.Callback mSessionCallback = new MediaSession.Callback() { 50 @Override 51 public boolean onMediaButtonEvent(Intent intent) { 52 try { 53 Log.startSession("HMB.oMBE"); 54 KeyEvent event = (KeyEvent) intent.getParcelableExtra(Intent.EXTRA_KEY_EVENT); 55 Log.v(this, "SessionCallback.onMediaButton()... event = %s.", event); 56 if ((event != null) && ((event.getKeyCode() == KeyEvent.KEYCODE_HEADSETHOOK) || 57 (event.getKeyCode() == KeyEvent.KEYCODE_MEDIA_PLAY_PAUSE))) { 58 synchronized (mLock) { 59 Log.v(this, "SessionCallback: HEADSETHOOK/MEDIA_PLAY_PAUSE"); 60 boolean consumed = handleCallMediaButton(event); 61 Log.v(this, "==> handleCallMediaButton(): consumed = %b.", consumed); 62 return consumed; 63 } 64 } 65 return true; 66 } finally { 67 Log.endSession(); 68 } 69 } 70 }; 71 72 private final Handler mMediaSessionHandler = new Handler(Looper.getMainLooper()) { 73 @Override 74 public void handleMessage(Message msg) { 75 switch (msg.what) { 76 case MSG_MEDIA_SESSION_INITIALIZE: { 77 MediaSession session = new MediaSession( 78 mContext, 79 HeadsetMediaButton.class.getSimpleName()); 80 session.setCallback(mSessionCallback); 81 session.setFlags(MediaSession.FLAG_EXCLUSIVE_GLOBAL_PRIORITY 82 | MediaSession.FLAG_HANDLES_MEDIA_BUTTONS); 83 session.setPlaybackToLocal(AUDIO_ATTRIBUTES); 84 mSession = session; 85 break; 86 } 87 case MSG_MEDIA_SESSION_SET_ACTIVE: { 88 if (mSession != null) { 89 boolean activate = msg.arg1 != 0; 90 if (activate != mSession.isActive()) { 91 mSession.setActive(activate); 92 } 93 } 94 break; 95 } 96 default: 97 break; 98 } 99 } 100 }; 101 102 private final Context mContext; 103 private final CallsManager mCallsManager; 104 private final TelecomSystem.SyncRoot mLock; 105 private MediaSession mSession; 106 private KeyEvent mLastHookEvent; 107 108 public HeadsetMediaButton( 109 Context context, 110 CallsManager callsManager, 111 TelecomSystem.SyncRoot lock) { 112 mContext = context; 113 mCallsManager = callsManager; 114 mLock = lock; 115 116 // Create a MediaSession but don't enable it yet. This is a 117 // replacement for MediaButtonReceiver 118 mMediaSessionHandler.obtainMessage(MSG_MEDIA_SESSION_INITIALIZE).sendToTarget(); 119 } 120 121 /** 122 * Handles the wired headset button while in-call. 123 * 124 * @return true if we consumed the event. 125 */ 126 private boolean handleCallMediaButton(KeyEvent event) { 127 Log.d(this, "handleCallMediaButton()...%s %s", event.getAction(), event.getRepeatCount()); 128 129 // Save ACTION_DOWN Event temporarily. 130 if (event.getAction() == KeyEvent.ACTION_DOWN) { 131 mLastHookEvent = event; 132 } 133 134 if (event.isLongPress()) { 135 return mCallsManager.onMediaButton(LONG_PRESS); 136 } else if (event.getAction() == KeyEvent.ACTION_UP) { 137 // We should not judge SHORT_PRESS by ACTION_UP event repeatCount, because it always 138 // return 0. 139 // Actually ACTION_DOWN event repeatCount only increases when LONG_PRESS performed. 140 if (mLastHookEvent != null && mLastHookEvent.getRepeatCount() == 0) { 141 return mCallsManager.onMediaButton(SHORT_PRESS); 142 } 143 } 144 145 if (event.getAction() != KeyEvent.ACTION_DOWN) { 146 mLastHookEvent = null; 147 } 148 149 return true; 150 } 151 152 /** ${inheritDoc} */ 153 @Override 154 public void onCallAdded(Call call) { 155 if (call.isExternalCall()) { 156 return; 157 } 158 mMediaSessionHandler.obtainMessage(MSG_MEDIA_SESSION_SET_ACTIVE, 1, 0).sendToTarget(); 159 } 160 161 /** ${inheritDoc} */ 162 @Override 163 public void onCallRemoved(Call call) { 164 if (call.isExternalCall()) { 165 return; 166 } 167 if (!mCallsManager.hasAnyCalls()) { 168 mMediaSessionHandler.obtainMessage(MSG_MEDIA_SESSION_SET_ACTIVE, 0, 0).sendToTarget(); 169 } 170 } 171 172 /** ${inheritDoc} */ 173 @Override 174 public void onExternalCallChanged(Call call, boolean isExternalCall) { 175 if (isExternalCall) { 176 onCallRemoved(call); 177 } else { 178 onCallAdded(call); 179 } 180 } 181 } 182