1 /* 2 * Copyright (C) 2009 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.scoaudiotest; 18 19 import android.app.Activity; 20 import android.bluetooth.BluetoothAdapter; 21 import android.bluetooth.BluetoothDevice; 22 import android.bluetooth.BluetoothHeadset; 23 import android.bluetooth.BluetoothProfile; 24 import android.content.BroadcastReceiver; 25 import android.content.Context; 26 import android.content.Intent; 27 import android.content.IntentFilter; 28 import android.content.res.AssetFileDescriptor; 29 import android.media.AudioManager; 30 import android.media.MediaPlayer; 31 import android.media.MediaRecorder; 32 import android.os.Bundle; 33 import android.os.Environment; 34 import android.os.Handler; 35 import android.speech.tts.TextToSpeech; 36 import android.speech.tts.TextToSpeech.OnUtteranceCompletedListener; 37 import android.util.Log; 38 import android.view.KeyEvent; 39 import android.view.View; 40 import android.view.View.OnClickListener; 41 import android.widget.ArrayAdapter; 42 import android.widget.CheckBox; 43 import android.widget.CompoundButton; 44 import android.widget.CompoundButton.OnCheckedChangeListener; 45 import android.widget.EditText; 46 import android.widget.ImageButton; 47 import android.widget.ImageView; 48 import android.widget.Spinner; 49 import android.widget.TextView; 50 import android.widget.ToggleButton; 51 52 import java.io.File; 53 import java.util.HashMap; 54 import java.util.List; 55 import java.util.Locale; 56 57 public class ScoAudioTest extends Activity { 58 59 final static String TAG = "ScoAudioTest"; 60 61 AudioManager mAudioManager; 62 AudioManager mAudioManager2; 63 boolean mForceScoOn; 64 ToggleButton mScoButton; 65 ToggleButton mVoiceDialerButton; 66 boolean mVoiceDialerOn; 67 String mLastRecordedFile; 68 SimpleMediaController mMediaControllers[] = new SimpleMediaController[2]; 69 private TextToSpeech mTts; 70 private HashMap<String, String> mTtsParams; 71 private int mOriginalVoiceVolume; 72 EditText mSpeakText; 73 boolean mTtsInited; 74 private Handler mHandler; 75 private static final String UTTERANCE = "utterance"; 76 private static Intent sVoiceCommandIntent; 77 private File mSampleFile; 78 ToggleButton mTtsToFileButton; 79 private boolean mTtsToFile; 80 private int mCurrentMode; 81 Spinner mModeSpinner; 82 private BluetoothHeadset mBluetoothHeadset; 83 private BluetoothDevice mBluetoothHeadsetDevice; 84 TextView mScoStateTxt; 85 TextView mVdStateTxt; 86 87 private final BroadcastReceiver mReceiver = new ScoBroadcastReceiver(); 88 89 public ScoAudioTest() { 90 Log.e(TAG, "contructor"); 91 } 92 93 /** Called when the activity is first created. */ 94 @Override 95 public void onCreate(Bundle icicle) { 96 super.onCreate(icicle); 97 98 setContentView(R.layout.scoaudiotest); 99 100 mScoStateTxt = findViewById(R.id.scoStateTxt); 101 mVdStateTxt = findViewById(R.id.vdStateTxt); 102 103 IntentFilter intentFilter = 104 new IntentFilter(BluetoothHeadset.ACTION_AUDIO_STATE_CHANGED); 105 intentFilter.addAction(AudioManager.ACTION_SCO_AUDIO_STATE_CHANGED); 106 intentFilter.addAction(AudioManager.ACTION_SCO_AUDIO_STATE_UPDATED); 107 registerReceiver(mReceiver, intentFilter); 108 109 mAudioManager = (AudioManager) getSystemService(Context.AUDIO_SERVICE); 110 mAudioManager2 = (AudioManager) getApplicationContext().getSystemService(Context.AUDIO_SERVICE); 111 mHandler = new Handler(); 112 113 mMediaControllers[0] = new SimplePlayerController(this, R.id.playPause1, R.id.stop1, 114 R.raw.sine440_mo_16b_16k, AudioManager.STREAM_BLUETOOTH_SCO); 115 TextView name = findViewById(R.id.playPause1Text); 116 name.setText("VOICE_CALL stream"); 117 118 mScoButton = (ToggleButton)findViewById(R.id.ForceScoButton); 119 mScoButton.setOnCheckedChangeListener(mForceScoChanged); 120 mForceScoOn = false; 121 mScoButton.setChecked(mForceScoOn); 122 123 mVoiceDialerButton = (ToggleButton)findViewById(R.id.VoiceDialerButton); 124 mVoiceDialerButton.setOnCheckedChangeListener(mVoiceDialerChanged); 125 mVoiceDialerOn = false; 126 mVoiceDialerButton.setChecked(mVoiceDialerOn); 127 128 129 mMediaControllers[1] = new SimpleRecordController(this, R.id.recStop1, 0, "Sco_record_"); 130 mTtsInited = false; 131 mTts = new TextToSpeech(this, new TtsInitListener()); 132 mTtsParams = new HashMap<String, String>(); 133 mTtsParams.put(TextToSpeech.Engine.KEY_PARAM_STREAM, 134 String.valueOf(AudioManager.STREAM_BLUETOOTH_SCO)); 135 mTtsParams.put(TextToSpeech.Engine.KEY_PARAM_UTTERANCE_ID, 136 UTTERANCE); 137 138 mSpeakText = findViewById(R.id.speakTextEdit); 139 mSpeakText.setOnKeyListener(mSpeakKeyListener); 140 mSpeakText.setText("sco audio test sentence"); 141 mTtsToFileButton = (ToggleButton)findViewById(R.id.TtsToFileButton); 142 mTtsToFileButton.setOnCheckedChangeListener(mTtsToFileChanged); 143 mTtsToFile = true; 144 mTtsToFileButton.setChecked(mTtsToFile); 145 146 mModeSpinner = findViewById(R.id.modeSpinner); 147 ArrayAdapter<String> adapter = new ArrayAdapter<String>(this, 148 android.R.layout.simple_spinner_item, mModeStrings); 149 adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item); 150 mModeSpinner.setAdapter(adapter); 151 mModeSpinner.setOnItemSelectedListener(mModeChanged); 152 mCurrentMode = mAudioManager.getMode(); 153 mModeSpinner.setSelection(mCurrentMode); 154 155 mBluetoothHeadsetDevice = null; 156 BluetoothAdapter btAdapter = BluetoothAdapter.getDefaultAdapter(); 157 if (btAdapter != null) { 158 btAdapter.getProfileProxy(this, mBluetoothProfileServiceListener, 159 BluetoothProfile.HEADSET); 160 } 161 162 sVoiceCommandIntent = new Intent(Intent.ACTION_VOICE_COMMAND); 163 sVoiceCommandIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); 164 } 165 166 @Override 167 public void onDestroy() { 168 super.onDestroy(); 169 mTts.shutdown(); 170 unregisterReceiver(mReceiver); 171 if (mBluetoothHeadset != null) { 172 BluetoothAdapter btAdapter = BluetoothAdapter.getDefaultAdapter(); 173 if (btAdapter != null) { 174 btAdapter.closeProfileProxy(BluetoothProfile.HEADSET, mBluetoothHeadset); 175 } 176 } 177 } 178 179 @Override 180 protected void onPause() { 181 super.onPause(); 182 // mForceScoOn = false; 183 // mScoButton.setChecked(mForceScoOn); 184 mMediaControllers[0].stop(); 185 mMediaControllers[1].stop(); 186 mAudioManager.setStreamVolume(AudioManager.STREAM_BLUETOOTH_SCO, 187 mOriginalVoiceVolume, 0); 188 } 189 190 @Override 191 protected void onResume() { 192 super.onResume(); 193 mLastRecordedFile = ""; 194 mMediaControllers[0].mFileName = ""; 195 mOriginalVoiceVolume = mAudioManager.getStreamVolume( 196 AudioManager.STREAM_BLUETOOTH_SCO); 197 setVolumeControlStream(AudioManager.STREAM_BLUETOOTH_SCO); 198 mCurrentMode = mAudioManager.getMode(); 199 mModeSpinner.setSelection(mCurrentMode); 200 } 201 202 private OnCheckedChangeListener mForceScoChanged 203 = new OnCheckedChangeListener(){ 204 @Override 205 public void onCheckedChanged(CompoundButton buttonView, 206 boolean isChecked) { 207 if (mForceScoOn != isChecked) { 208 mForceScoOn = isChecked; 209 AudioManager mngr = mAudioManager; 210 boolean useVirtualCall = false; 211 CheckBox box = findViewById(R.id.useSecondAudioManager); 212 if (box.isChecked()) { 213 Log.i(TAG, "Using 2nd audio manager"); 214 mngr = mAudioManager2; 215 } 216 box = findViewById(R.id.useVirtualCallCheckBox); 217 useVirtualCall = box.isChecked(); 218 219 if (mForceScoOn) { 220 if (useVirtualCall) { 221 Log.e(TAG, "startBluetoothScoVirtualCall() IN"); 222 mngr.startBluetoothScoVirtualCall(); 223 Log.e(TAG, "startBluetoothScoVirtualCall() OUT"); 224 } else { 225 Log.e(TAG, "startBluetoothSco() IN"); 226 mngr.startBluetoothSco(); 227 Log.e(TAG, "startBluetoothSco() OUT"); 228 } 229 } else { 230 Log.e(TAG, "stopBluetoothSco() IN"); 231 mngr.stopBluetoothSco(); 232 Log.e(TAG, "stopBluetoothSco() OUT"); 233 } 234 } 235 } 236 }; 237 238 private OnCheckedChangeListener mVoiceDialerChanged 239 = new OnCheckedChangeListener(){ 240 @Override 241 public void onCheckedChanged(CompoundButton buttonView, 242 boolean isChecked) { 243 if (mVoiceDialerOn != isChecked) { 244 mVoiceDialerOn = isChecked; 245 if (mBluetoothHeadset != null && mBluetoothHeadsetDevice != null) { 246 if (mVoiceDialerOn) { 247 mBluetoothHeadset.startVoiceRecognition(mBluetoothHeadsetDevice); 248 } else { 249 mBluetoothHeadset.stopVoiceRecognition(mBluetoothHeadsetDevice); 250 } 251 } 252 } 253 } 254 }; 255 256 private OnCheckedChangeListener mTtsToFileChanged 257 = new OnCheckedChangeListener(){ 258 @Override 259 public void onCheckedChanged(CompoundButton buttonView, 260 boolean isChecked) { 261 mTtsToFile = isChecked; 262 } 263 }; 264 265 private class SimpleMediaController implements OnClickListener { 266 int mPlayPauseButtonId; 267 int mStopButtonId; 268 Context mContext; 269 ImageView mPlayPauseButton; 270 int mPlayImageResource; 271 int mPauseImageResource; 272 String mFileNameBase; 273 String mFileName; 274 int mFileResId; 275 276 SimpleMediaController(Context context, int playPausebuttonId, int stopButtonId, String fileName) { 277 mContext = context; 278 mPlayPauseButtonId = playPausebuttonId; 279 mStopButtonId = stopButtonId; 280 mFileNameBase = fileName; 281 mPlayPauseButton = findViewById(playPausebuttonId); 282 ImageButton stop = findViewById(stopButtonId); 283 284 mPlayPauseButton.setOnClickListener(this); 285 mPlayPauseButton.requestFocus(); 286 if (stop != null) { 287 stop.setOnClickListener(this); 288 } 289 } 290 291 SimpleMediaController(Context context, int playPausebuttonId, int stopButtonId, int fileResId) { 292 mContext = context; 293 mPlayPauseButtonId = playPausebuttonId; 294 mStopButtonId = stopButtonId; 295 mFileNameBase = ""; 296 mFileResId = fileResId; 297 mPlayPauseButton = findViewById(playPausebuttonId); 298 ImageButton stop = findViewById(stopButtonId); 299 300 mPlayPauseButton.setOnClickListener(this); 301 mPlayPauseButton.requestFocus(); 302 if (stop != null) { 303 stop.setOnClickListener(this); 304 } 305 } 306 307 @Override 308 public void onClick(View v) { 309 if (v.getId() == mPlayPauseButtonId) { 310 playOrPause(); 311 } else if (v.getId() == mStopButtonId) { 312 stop(); 313 } 314 } 315 316 public void playOrPause() { 317 } 318 319 public void stop() { 320 } 321 322 public boolean isPlaying() { 323 return false; 324 } 325 326 public void updatePlayPauseButton() { 327 mPlayPauseButton.setImageResource(isPlaying() ? mPauseImageResource : mPlayImageResource); 328 } 329 } 330 331 private class SimplePlayerController extends SimpleMediaController { 332 private MediaPlayer mMediaPlayer; 333 private int mStreamType; 334 SimplePlayerController(Context context, int playPausebuttonId, int stopButtonId, String fileName, int stream) { 335 super(context, playPausebuttonId, stopButtonId, fileName); 336 337 mPlayImageResource = android.R.drawable.ic_media_play; 338 mPauseImageResource = android.R.drawable.ic_media_pause; 339 mStreamType = stream; 340 mFileName = Environment.getExternalStorageDirectory().toString() + "/music/" + 341 mFileNameBase + "_" + ".wav"; 342 } 343 344 SimplePlayerController(Context context, int playPausebuttonId, int stopButtonId, int fileResId, int stream) { 345 super(context, playPausebuttonId, stopButtonId, fileResId); 346 347 mPlayImageResource = android.R.drawable.ic_media_play; 348 mPauseImageResource = android.R.drawable.ic_media_pause; 349 mStreamType = stream; 350 mFileName = ""; 351 } 352 353 @Override 354 public void playOrPause() { 355 Log.e(TAG, "playOrPause playing: "+((mMediaPlayer == null)?false:!mMediaPlayer.isPlaying())+ 356 " mMediaPlayer: "+mMediaPlayer+ 357 " mFileName: "+mFileName+ 358 " mLastRecordedFile: "+mLastRecordedFile); 359 if (mMediaPlayer == null || !mMediaPlayer.isPlaying()){ 360 if (mMediaPlayer == null) { 361 if (mFileName != mLastRecordedFile) { 362 mFileName = mLastRecordedFile; 363 Log.e(TAG, "new recorded file: "+mFileName); 364 } 365 try { 366 mMediaPlayer = new MediaPlayer(); 367 if (mFileName.equals("")) { 368 Log.e(TAG, "Playing from resource"); 369 AssetFileDescriptor afd = mContext.getResources().openRawResourceFd(mFileResId); 370 mMediaPlayer.setDataSource(afd.getFileDescriptor(), afd.getStartOffset(), afd.getLength()); 371 afd.close(); 372 } else { 373 Log.e(TAG, "Playing file: "+mFileName); 374 mMediaPlayer.setDataSource(mFileName); 375 } 376 mMediaPlayer.setAudioStreamType(mStreamType); 377 mMediaPlayer.prepare(); 378 mMediaPlayer.setLooping(true); 379 } catch (Exception ex) { 380 Log.e(TAG, "mMediaPlayercreate failed:", ex); 381 mMediaPlayer.release(); 382 mMediaPlayer = null; 383 } 384 385 if (mMediaPlayer != null) { 386 mMediaPlayer.setOnCompletionListener(new MediaPlayer.OnCompletionListener() { 387 @Override 388 public void onCompletion(MediaPlayer mp) { 389 updatePlayPauseButton(); 390 } 391 }); 392 } 393 } 394 if (mMediaPlayer != null) { 395 mMediaPlayer.start(); 396 } 397 } else { 398 mMediaPlayer.pause(); 399 } 400 updatePlayPauseButton(); 401 } 402 @Override 403 public void stop() { 404 if (mMediaPlayer != null) { 405 mMediaPlayer.stop(); 406 mMediaPlayer.release(); 407 mMediaPlayer = null; 408 } 409 updatePlayPauseButton(); 410 } 411 412 @Override 413 public boolean isPlaying() { 414 if (mMediaPlayer != null) { 415 return mMediaPlayer.isPlaying(); 416 } else { 417 return false; 418 } 419 } 420 } 421 422 private class SimpleRecordController extends SimpleMediaController { 423 private MediaRecorder mMediaRecorder; 424 private int mFileCount = 0; 425 private int mState = 0; 426 SimpleRecordController(Context context, int playPausebuttonId, int stopButtonId, String fileName) { 427 super(context, playPausebuttonId, stopButtonId, fileName); 428 Log.e(TAG, "SimpleRecordController cstor"); 429 mPlayImageResource = R.drawable.record; 430 mPauseImageResource = R.drawable.stop; 431 } 432 433 @Override 434 public void playOrPause() { 435 if (mState == 0) { 436 setup(); 437 try { 438 mMediaRecorder.start(); 439 mState = 1; 440 } catch (Exception e) { 441 Log.e(TAG, "Could start MediaRecorder: ", e); 442 mMediaRecorder.release(); 443 mMediaRecorder = null; 444 mState = 0; 445 } 446 } else { 447 try { 448 mMediaRecorder.stop(); 449 mMediaRecorder.reset(); 450 } catch (Exception e) { 451 Log.e(TAG, "Could not stop MediaRecorder: ", e); 452 mMediaRecorder.release(); 453 mMediaRecorder = null; 454 } finally { 455 mState = 0; 456 } 457 } 458 updatePlayPauseButton(); 459 } 460 461 public void setup() { 462 Log.e(TAG, "SimpleRecordController setup()"); 463 if (mMediaRecorder == null) { 464 mMediaRecorder = new MediaRecorder(); 465 } 466 mMediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC); 467 mMediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP); 468 mMediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB); 469 mFileName = Environment.getExternalStorageDirectory().toString() + "/music/" + 470 mFileNameBase + "_" + ++mFileCount + ".amr"; 471 mLastRecordedFile = mFileName; 472 Log.e(TAG, "recording to file: "+mLastRecordedFile); 473 mMediaRecorder.setOutputFile(mFileName); 474 try { 475 mMediaRecorder.prepare(); 476 } 477 catch (Exception e) { 478 Log.e(TAG, "Could not prepare MediaRecorder: ", e); 479 mMediaRecorder.release(); 480 mMediaRecorder = null; 481 } 482 } 483 484 @Override 485 public void stop() { 486 if (mMediaRecorder != null) { 487 try { 488 mMediaRecorder.stop(); 489 } catch (Exception e) { 490 Log.e(TAG, "Could not stop MediaRecorder: ", e); 491 } finally { 492 mMediaRecorder.release(); 493 mMediaRecorder = null; 494 } 495 } 496 updatePlayPauseButton(); 497 } 498 499 @Override 500 public boolean isPlaying() { 501 if (mState == 1) { 502 return true; 503 } else { 504 return false; 505 } 506 } 507 } 508 509 class TtsInitListener implements TextToSpeech.OnInitListener { 510 @Override 511 public void onInit(int status) { 512 // status can be either TextToSpeech.SUCCESS or TextToSpeech.ERROR. 513 Log.e(TAG, "onInit for tts"); 514 if (status != TextToSpeech.SUCCESS) { 515 // Initialization failed. 516 Log.e(TAG, "Could not initialize TextToSpeech."); 517 return; 518 } 519 520 if (mTts == null) { 521 Log.e(TAG, "null tts"); 522 return; 523 } 524 525 int result = mTts.setLanguage(Locale.US); 526 if (result == TextToSpeech.LANG_MISSING_DATA || 527 result == TextToSpeech.LANG_NOT_SUPPORTED) { 528 // Lanuage data is missing or the language is not supported. 529 Log.e(TAG, "Language is not available."); 530 return; 531 } 532 mTts.setOnUtteranceCompletedListener(new MyUtteranceCompletedListener(UTTERANCE)); 533 mTtsInited = true; 534 } 535 } 536 537 class MyUtteranceCompletedListener implements OnUtteranceCompletedListener { 538 private final String mExpectedUtterance; 539 540 public MyUtteranceCompletedListener(String expectedUtteranceId) { 541 mExpectedUtterance = expectedUtteranceId; 542 } 543 544 @Override 545 public void onUtteranceCompleted(String utteranceId) { 546 Log.e(TAG, "onUtteranceCompleted " + utteranceId); 547 if (mTtsToFile) { 548 if (mSampleFile != null && mSampleFile.exists()) { 549 MediaPlayer mediaPlayer = new MediaPlayer(); 550 try { 551 mediaPlayer.setDataSource(mSampleFile.getPath()); 552 mediaPlayer.setAudioStreamType(AudioManager.STREAM_BLUETOOTH_SCO); 553 mediaPlayer.prepare(); 554 } catch (Exception ex) { 555 Log.e(TAG, "mMediaPlayercreate failed:", ex); 556 mediaPlayer.release(); 557 mediaPlayer = null; 558 } 559 560 if (mediaPlayer != null) { 561 mediaPlayer.setOnCompletionListener(new MediaPlayer.OnCompletionListener() { 562 @Override 563 public void onCompletion(MediaPlayer mp) { 564 mp.release(); 565 if (mSampleFile != null && mSampleFile.exists()) { 566 mSampleFile.delete(); 567 mSampleFile = null; 568 } 569 mAudioManager.setStreamVolume(AudioManager.STREAM_BLUETOOTH_SCO, 570 mOriginalVoiceVolume, 0); 571 // Debug.stopMethodTracing(); 572 } 573 }); 574 mediaPlayer.start(); 575 } 576 } else { 577 Log.e(TAG, "synthesizeToFile did not create file"); 578 } 579 } else { 580 mAudioManager.setStreamVolume(AudioManager.STREAM_BLUETOOTH_SCO, 581 mOriginalVoiceVolume, 0); 582 // Debug.stopMethodTracing(); 583 } 584 585 Log.e(TAG, "end speak, volume: "+mOriginalVoiceVolume); 586 } 587 } 588 589 590 private View.OnKeyListener mSpeakKeyListener 591 = new View.OnKeyListener() { 592 @Override 593 public boolean onKey(View v, int keyCode, KeyEvent event) { 594 if (event.getAction() == KeyEvent.ACTION_DOWN) { 595 switch (keyCode) { 596 case KeyEvent.KEYCODE_DPAD_CENTER: 597 case KeyEvent.KEYCODE_ENTER: 598 if (!mTtsInited) { 599 Log.e(TAG, "Tts not inited "); 600 return false; 601 } 602 mOriginalVoiceVolume = mAudioManager.getStreamVolume( 603 AudioManager.STREAM_BLUETOOTH_SCO); 604 Log.e(TAG, "start speak, volume: "+mOriginalVoiceVolume); 605 mAudioManager.setStreamVolume(AudioManager.STREAM_BLUETOOTH_SCO, 606 mOriginalVoiceVolume/2, 0); 607 608 // we now have SCO connection and TTS, so we can start. 609 mHandler.post(new Runnable() { 610 @Override 611 public void run() { 612 // Debug.startMethodTracing("tts"); 613 614 if (mTtsToFile) { 615 if (mSampleFile != null && mSampleFile.exists()) { 616 mSampleFile.delete(); 617 mSampleFile = null; 618 } 619 mSampleFile = new File(Environment.getExternalStorageDirectory(), "mytts.wav"); 620 mTts.synthesizeToFile(mSpeakText.getText().toString(), mTtsParams, mSampleFile.getPath()); 621 } else { 622 mTts.speak(mSpeakText.getText().toString(), 623 TextToSpeech.QUEUE_FLUSH, 624 mTtsParams); 625 } 626 } 627 }); 628 return true; 629 } 630 } 631 return false; 632 } 633 }; 634 635 private static final String[] mModeStrings = { 636 "NORMAL", "RINGTONE", "IN_CALL", "IN_COMMUNICATION" 637 }; 638 639 private Spinner.OnItemSelectedListener mModeChanged 640 = new Spinner.OnItemSelectedListener() { 641 @Override 642 public void onItemSelected(android.widget.AdapterView av, View v, 643 int position, long id) { 644 if (mCurrentMode != position) { 645 mCurrentMode = position; 646 mAudioManager.setMode(mCurrentMode); 647 } 648 } 649 650 @Override 651 public void onNothingSelected(android.widget.AdapterView av) { 652 } 653 }; 654 655 private BluetoothProfile.ServiceListener mBluetoothProfileServiceListener = 656 new BluetoothProfile.ServiceListener() { 657 @Override 658 public void onServiceConnected(int profile, BluetoothProfile proxy) { 659 mBluetoothHeadset = (BluetoothHeadset) proxy; 660 List<BluetoothDevice> deviceList = mBluetoothHeadset.getConnectedDevices(); 661 if (deviceList.size() > 0) { 662 mBluetoothHeadsetDevice = deviceList.get(0); 663 } else { 664 mBluetoothHeadsetDevice = null; 665 } 666 } 667 @Override 668 public void onServiceDisconnected(int profile) { 669 if (mBluetoothHeadset != null) { 670 List<BluetoothDevice> devices = mBluetoothHeadset.getConnectedDevices(); 671 if (devices.size() == 0) { 672 mBluetoothHeadsetDevice = null; 673 } 674 mBluetoothHeadset = null; 675 } 676 } 677 }; 678 679 private int mChangedState = -1; 680 private int mUpdatedState = -1; 681 private int mUpdatedPrevState = -1; 682 683 private class ScoBroadcastReceiver extends BroadcastReceiver { 684 @Override 685 public void onReceive(Context context, Intent intent) { 686 String action = intent.getAction(); 687 688 if (action.equals(BluetoothHeadset.ACTION_AUDIO_STATE_CHANGED)) { 689 int state = intent.getIntExtra(BluetoothProfile.EXTRA_STATE, -1); 690 mVdStateTxt.setText(Integer.toString(state)); 691 Log.e(TAG, "BluetoothHeadset.ACTION_AUDIO_STATE_CHANGED: "+state); 692 } else if (action.equals(AudioManager.ACTION_SCO_AUDIO_STATE_CHANGED)) { 693 mChangedState = intent.getIntExtra(AudioManager.EXTRA_SCO_AUDIO_STATE, -1); 694 Log.e(TAG, "ACTION_SCO_AUDIO_STATE_CHANGED: "+mChangedState); 695 mScoStateTxt.setText("changed: "+Integer.toString(mChangedState)+ 696 " updated: "+Integer.toString(mUpdatedState)+ 697 " prev updated: "+Integer.toString(mUpdatedPrevState)); 698 } else if (action.equals(AudioManager.ACTION_SCO_AUDIO_STATE_UPDATED)) { 699 mUpdatedState = intent.getIntExtra(AudioManager.EXTRA_SCO_AUDIO_STATE, -1); 700 mUpdatedPrevState = intent.getIntExtra(AudioManager.EXTRA_SCO_AUDIO_PREVIOUS_STATE, -1); 701 Log.e(TAG, "ACTION_SCO_AUDIO_STATE_UPDATED, state: "+mUpdatedState+" prev state: "+mUpdatedPrevState); 702 mScoStateTxt.setText("changed: "+Integer.toString(mChangedState)+ 703 " updated: "+Integer.toString(mUpdatedState)+ 704 " prev updated: "+Integer.toString(mUpdatedPrevState)); 705 if (mForceScoOn && mUpdatedState == AudioManager.SCO_AUDIO_STATE_DISCONNECTED) { 706 mForceScoOn = false; 707 mScoButton.setChecked(mForceScoOn); 708 mAudioManager.stopBluetoothSco(); 709 } 710 } 711 } 712 } 713 714 } 715