Android - Получение звука для воспроизведения через динамик
В настоящее время у меня есть код, который считывает запись с микрофона устройств с использованием класса AudioRecord, а затем воспроизводит его с помощью класса AudioTrack.
Моя проблема в том, что когда я его воспроизвожу, она воспроизводится через громкоговоритель.
Я хочу, чтобы он воспроизводился через ушную часть устройства.
Вот мой код:
public class LoopProg extends Activity {
boolean isRecording; //currently not used
AudioManager am;
int count = 0;
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
am = (AudioManager) getSystemService(Context.AUDIO_SERVICE);
am.setMicrophoneMute(true);
while(count <= 1000000){
Record record = new Record();
record.run();
count ++;
Log.d("COUNT", "Count is : " + count);
}
}
public class Record extends Thread{
static final int bufferSize = 200000;
final short[] buffer = new short[bufferSize];
short[] readBuffer = new short[bufferSize];
public void run() {
isRecording = true;
android.os.Process.setThreadPriority(android.os.Process.THREAD_PRIORITY_URGENT_AUDIO);
int buffersize = AudioRecord.getMinBufferSize(11025, AudioFormat.CHANNEL_CONFIGURATION_MONO, AudioFormat.ENCODING_PCM_16BIT);
AudioRecord arec = new AudioRecord(MediaRecorder.AudioSource.MIC, 11025, AudioFormat.CHANNEL_CONFIGURATION_MONO, AudioFormat.ENCODING_PCM_16BIT, buffersize);
AudioTrack atrack = new AudioTrack(AudioManager.STREAM_MUSIC, 11025, AudioFormat.CHANNEL_CONFIGURATION_MONO, AudioFormat.ENCODING_PCM_16BIT, buffersize, AudioTrack.MODE_STREAM);
am.setRouting(AudioManager.MODE_NORMAL,1, AudioManager.STREAM_MUSIC);
int ok = am.getRouting(AudioManager.ROUTE_EARPIECE);
Log.d("ROUTING", "getRouting = " + ok);
setVolumeControlStream(AudioManager.STREAM_VOICE_CALL);
//am.setSpeakerphoneOn(true);
Log.d("SPEAKERPHONE", "Is speakerphone on? : " + am.isSpeakerphoneOn());
am.setSpeakerphoneOn(false);
Log.d("SPEAKERPHONE", "Is speakerphone on? : " + am.isSpeakerphoneOn());
atrack.setPlaybackRate(11025);
byte[] buffer = new byte[buffersize];
arec.startRecording();
atrack.play();
while(isRecording) {
arec.read(buffer, 0, buffersize);
atrack.write(buffer, 0, buffer.length);
}
arec.stop();
atrack.stop();
isRecording = false;
}
}
}
Как вы можете видеть, был ли код, который я пробовал использовать с классом AudioManager и его методами, включая устаревший метод setRouting и ничего не работает, метод setSpeakerphoneOn, похоже, не имеет никакого эффекта, и метод маршрутизации.
Есть ли у кого-нибудь какие-либо идеи о том, как заставить его играть через динамик вместо звонящего телефона?
Ответы
Ответ 1
Просто ему пришлось работать над 2.2. Мне все еще нужна установка In_Call, которая мне не нравится, но сейчас я общаюсь с ней. Я смог отбросить данные маршрутизации вызовов, которые теперь устарели. Я нашел, что вам определенно нужно разрешение Modify_Audio_Settings
, без него нет ошибки, но метод setSpeakerPhone
просто ничего не делает. Вот макет кода, который я использовал.
private AudioManager m_amAudioManager;
m_amAudioManager = (AudioManager)getSystemService(Context.AUDIO_SERVICE);
m_amAudioManager.setMode(AudioManager.MODE_IN_CALL);
m_amAudioManager.setSpeakerphoneOn(false);
Ответ 2
В этом недавнем вопросе была некоторая взаимосвязанная дискуссия:
Android - можно ли отключить звук в настоящее время в аудио-приложениях?
На основе исходного кода AudioManager
кажется, что вы должны находиться в режиме "вызова" до того, как метод setSpeakerphoneOn
имеет какой-либо эффект.
Однако недавно я увидел приложение, которое могло бы легко переключаться между динамиком и громкой связью, пока показывая текущий поток как поток "media", поэтому я буду заинтересован в дальнейших ответах.
Ответ 3
Пропущенные некоторые ответы здесь довольно долго. Я использую Android 2.2. "AudioManager.setSpeakerphoneOn(ложь);" работает.
audioManager.setSpeakerphoneOn(false);
...
mediaPlayer.setDataSource(..);
mediaPlayer.setAudioStreamType(AudioManager.STREAM_VOICE_CALL);
mediaPlayer.prepare();
Ответ 4
Пожалуйста, используйте этот код, хорошо работает:
//PLAY ON EARPIECE
mPlayer.setAudioStreamType(AudioManager.STREAM_VOICE_CALL);
audioManager.setMode(AudioManager.MODE_IN_CALL);
audioManager.setSpeakerphoneOn(false);
//PLAY ON SPEAKER
mPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
audioManager.setMode(AudioManager.MODE_IN_CALL);
audioManager.setSpeakerphoneOn(true);
Ответ 5
Кажется, что я работал над 1.6.
Итак, я сказал, что опубликую здесь, как я это сделал.
Чтобы заставить его работать в 1.6 I:
Использовал класс AudioManager
для установки setSpeakerphoneOn(false)
, затем я использовал Voice_Call_Stream
и добавил регулятор громкости в Voice_Call_Stream
.
Метод setSpeakerphoneOn(false)
используется в onCreate()
для этой активности, и, как представляется, он направляется на гарнитуру, я затем использовал кнопку и использовал метод setSpeakerphoneOn(true)
, и звук передается в динамик.
Метод работает только тогда, когда он используется в onCreate()
для меня, и я не тестировал его широко, но на данный момент он позволяет мне переключаться между гарнитурой и динамиком на устройстве 1.6.
Ответ 6
public MediaPlayer m_mpSpeakerPlayer;
private AudioManager m_amAudioManager;
m_amAudioManager = (AudioManager)getSystemService(Context.AUDIO_SERVICE);
// 從Receiver Earpiece發音
m_amAudioManager.setSpeakerphoneOn(false);
m_amAudioManager.setRouting(AudioManager.MODE_NORMAL, AudioManager.ROUTE_EARPIECE, AudioManager.ROUTE_ALL);
Log.i(TAG, String.valueOf(m_amAudioManager.getRouting(AudioManager.ROUTE_EARPIECE)));
setVolumeControlStream(AudioManager.STREAM_VOICE_CALL);
// 把聲音設定成從Earpiece出來
// 重點在這行,要把它設定為正在通話中
m_amAudioManager.setMode(AudioManager.MODE_IN_CALL);
// 開始放音樂
m_mpSpeakerPlayer.reset();
m_mpSpeakerPlayer.setDataSource("sdcard/receiver.mp3");
m_mpSpeakerPlayer.prepare();
m_mpSpeakerPlayer.start();
//最後再把它設定從Speaker放音,達成!
m_amAudioManager.setMode(AudioManager.MODE_NORMAL);
Ответ 7
Если слуховой аппарат подключен к телефону с помощью bluetooth (как я предполагаю), попробовали ли вы позвонить AudioManager.setBluetoothScoOn(true)
?
Я прошел через ссылку на Android, и это единственное, что я мог найти, что вы не упомянули о попытке.