解决部分华为,乐视手机听筒模式听不到声音问题_audiomanager.setspeakerphoneon部分手机不生效-程序员宅基地

技术标签: android  解决听筒模式听不到声音  华为手机听筒模式没有声音  安卓开发中的小技巧  切换听筒模式  

最近项目中出现一个问题,部分华为手机,乐视手机切换到听筒模式后,听不到声音的现象,搜索了好多答案,发现都是说设置模式不对,试了好多也没有解决掉。最后发现应该是重新播放的方式不对,在一部分手机上重新播放不兼容。

这是开始的代码,切换到听筒模式时调用:

VoicePlayer.setAudioManagerMode(AudioManager.MODE_IN_CALL);
((ChatActivity)context).setSpeakerphoneOn(false);
VoicePlayer.player.seekTo(0);//重新播放

public void setSpeakerphoneOn(boolean on) {
    try {
        AudioManager audioManager = (AudioManager) AndroidUtils.appCtx().getSystemService(Context.AUDIO_SERVICE);
        //播放音频流类型
        setVolumeControlStream(AudioManager.STREAM_VOICE_CALL);
        //获得当前类
        Class audioSystemClass = Class.forName("android.media.AudioSystem");
        //得到这个方法
        Method setForceUse = audioSystemClass.getMethod("setForceUse", int.class, int.class);

        if (on) {
            audioManager.setMicrophoneMute(false);
            audioManager.setSpeakerphoneOn(true);
            audioManager.setMode(AudioManager.MODE_NORMAL);
        } else {
		audioManager.setSpeakerphoneOn(false);
 		audioManager.setMode(AudioManager.MODE_NORMAL);
 		setForceUse.invoke(null, 0, 0);
 		audioManager.setMode(AudioManager.MODE_IN_COMMUNICATION);
} } catch (Exception e) { e.printStackTrace(); }}
这是后来改进的代码 ,切换到听筒模式时调用

VoicePlayer.rePlay(context,mAudioManager,false);

public class VoicePlayer {

   public static interface OnPlayListener extends OnCompletionListener {
      public String getPublicId();

      public void onCancel();

      public void onDownloadBegin();

      public void onDownloadEnd();

      public void onError();
      
      public void onPlayBegin(MediaPlayer player);
      
      public void onReplay();
      
      public void setRecMessageItem(RecMessageItem item);
   }
   /**
    * 管理其他音乐播放器
    * **/
   public interface OtherPlayerManagerI{
      /**
       * 暂停其他播放器
       * **/
      public void pauseOtherPlayer();
      /**
       * 恢复其他播放器
       * **/
      public void replyOtherPlayer();
      /**
       * 释放资源
       * **/
      public void releaseResource();
   }
   /**
    * 播放语音,管理音频焦点
    * **/
   public static  class  OtherPlayerManagerImpl implements OtherPlayerManagerI{
         public static Context otcontext=null;
          public static AudioManager    am=null;
         public  static OtherPlayerManagerImpl playerManagerImpl=null;
         //单例返回
         public  static OtherPlayerManagerImpl getInstance(Context context){       
            OtherPlayerManagerImpl.otcontext = context;
            if(playerManagerImpl==null){
               playerManagerImpl=new OtherPlayerManagerImpl();
            }
            if(am==null){
               am = (AudioManager)otcontext.getSystemService(Context.AUDIO_SERVICE);
            }         
            return playerManagerImpl;
         }
      @Override 
      public void pauseOtherPlayer() {
         int result = am.requestAudioFocus(null,AudioManager.STREAM_MUSIC,AudioManager.AUDIOFOCUS_GAIN_TRANSIENT);
      }
      @Override
      public void replyOtherPlayer() {         
         am.abandonAudioFocus(null);    
      }
      @Override
      public void releaseResource() {
         context=null;
         am=null;
         playerManagerImpl=null;
      }  
   }
    public static Context context;
   public static MediaPlayer player;
   public static RecMessageItem curPlayMessage;
   public static OnPlayListener curOnPlayListener;
   private static Set<String> downloading = Collections.synchronizedSet(new HashSet<String>());

   public static int audioManagerMode = AudioManager.MODE_NORMAL;
   
   public static void setAudioManagerMode(int mode){
      audioManagerMode = mode;
   }
   
   public static int getAudioManagerMode(){
      return audioManagerMode;
   }
   
   public static void clear() {
      player = null;
      curPlayMessage = null;
   }
   public static void rePlay(Activity act, AudioManager audioManager, boolean isSpeakerphoneOn) {
      player.stop();
      if (isSpeakerphoneOn) {
         setAudioManagerMode(AudioManager.MODE_NORMAL);
         act.setVolumeControlStream(AudioManager.STREAM_MUSIC);
         audioManager.setMode(AudioManager.MODE_NORMAL);// 切换到外放模式, 继续播放
         audioManager.setSpeakerphoneOn(isSpeakerphoneOn);
      } else {
         setAudioManagerMode(AudioManager.MODE_IN_CALL);
         act.setVolumeControlStream(AudioManager.STREAM_VOICE_CALL);
         audioManager.setMode(AudioManager.MODE_IN_CALL);//切换到听筒模式
         if (Build.VERSION.SDK_INT >= 11) {
            audioManager.setMode(AudioManager.MODE_IN_COMMUNICATION);//
         }
         audioManager.setSpeakerphoneOn(isSpeakerphoneOn);
      }
      player.release();
      player = null;
      doPlay(curPlayMessage, curOnPlayListener);
   }
   private static boolean doPlay(final RecMessageItem item, final OnPlayListener onPlayListener) {
      // 停掉前一个播放器
      if (player != null) {
         RecMessageItem cpmsg = curPlayMessage;
         try {
            player.release();
            curOnPlayListener.onCancel();
            SystemClock.sleep(300);
         } catch (Exception e) {
            // ignore...
         } finally {
            player = null;
            curOnPlayListener = null;
         }
         if (cpmsg != null && cpmsg.equals(item)) {
            return true;
         }
      }
      
      curPlayMessage = item;
      curOnPlayListener = onPlayListener;
      
      curOnPlayListener.setRecMessageItem(item);
      
      File encryptedFile = new File(MessageUtils.genMsgFileName(item.msgId));

      // 播放的文件已经缓存在本地,则直接播放
      if (encryptedFile.exists()) {
         FileInputStream voiceIS = null;
         File voiceFile = null;
         try {
            // decrypt it first...
            FileOutputStream voiceOS = AndroidUtils.appCtx().openFileOutput("voice", Context.MODE_PRIVATE);
            MessageUtils.deCrypt(new FileInputStream(encryptedFile), voiceOS);
            voiceFile = AndroidUtils.appCtx().getFileStreamPath("voice");
            voiceIS = new FileInputStream(voiceFile);

            // then play it
            player = new MediaPlayer();
            player.setDataSource(voiceIS.getFD());
            player.prepare();
            player.setOnCompletionListener(curOnPlayListener);
            player.start();
            curOnPlayListener.onPlayBegin(player);
            Log.e("AudioManager", "" + VoicePlayer.audioManagerMode);
            AudioManager audioManager = (AudioManager) AndroidUtils.appCtx().getSystemService(Context.AUDIO_SERVICE);
            if(VoicePlayer.audioManagerMode == AudioManager.MODE_NORMAL){
               audioManager.setMode(AudioManager.MODE_NORMAL);// 切换到外放模式, 继续播放
               audioManager.setSpeakerphoneOn(true);
            }else if (VoicePlayer.audioManagerMode == AudioManager.MODE_IN_CALL) {
               audioManager.setMode(AudioManager.MODE_IN_CALL);// //切换到听筒模式
               audioManager.setSpeakerphoneOn(false);
            }
         } catch (Exception e) {
            LogUtil.i("VoicePlayer", e.getMessage(), e);
            encryptedFile.delete();
            curOnPlayListener.onError();
            return false;
         } finally {
            if (voiceIS != null) {
               try {
                  voiceIS.close();
               } catch (Exception e) {
               }
            }
            if (voiceFile != null)
               voiceFile.delete();
         }

         return true;
      } else if (downloading.contains(item.msgId)) {
         return true;
      }

      return false;
   }

   private static void downloadAndPlay(final RecMessageItem item, final OnPlayListener onPlayListener) {
      if (!downloading.add(item.msgId)) {
         return;
      }

      onPlayListener.onDownloadBegin();

      new AsyncTask<String, Integer, Boolean>() {

         @Override
         protected Boolean doInBackground(String... params) {

            try {
               MessageUtils.getFileFromServer(onPlayListener.getPublicId(),item.groupId, item.msgId);
            } catch (Exception e) {
               e.printStackTrace();
               return false;

            }
            return true;
         };

         protected void onPostExecute(Boolean result) {
            onPlayListener.onDownloadEnd();
            downloading.remove(item.msgId);
            if (!result) {
               AndroidUtils.toastShort("无法下载当前音频");
               onPlayListener.onError();
               return;
            }

            if (curPlayMessage == null || curPlayMessage.msgId.equals(item.msgId)) {
               doPlay(item, onPlayListener);
            }

         }

      }.execute("");

   }

   public static void onDestroy() {
      // 停掉前一个播放器
      if (player != null) {
         player.release();
         if (curOnPlayListener != null)
            curOnPlayListener.onCancel();
         player = null;
         curPlayMessage = null;
      }
   }

   public static void play(final RecMessageItem item, final OnPlayListener onPlayListener,Context context) {
      VoicePlayer.context=context;
      if (item == null)
         return;
      if (doPlay(item, onPlayListener)) {
         return;
      }

      // 文件还未下载,则直接下载之后再播放
      downloadAndPlay(item, onPlayListener);

   }

}
希望我的经历对你有所帮助,欢迎大家评论交流!

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/ludi604902795/article/details/73741921

智能推荐

while循环&CPU占用率高问题深入分析与解决方案_main函数使用while(1)循环cpu占用99-程序员宅基地

文章浏览阅读3.8k次,点赞9次,收藏28次。直接上一个工作中碰到的问题,另外一个系统开启多线程调用我这边的接口,然后我这边会开启多线程批量查询第三方接口并且返回给调用方。使用的是两三年前别人遗留下来的方法,放到线上后发现确实是可以正常取到结果,但是一旦调用,CPU占用就直接100%(部署环境是win server服务器)。因此查看了下相关的老代码并使用JProfiler查看发现是在某个while循环的时候有问题。具体项目代码就不贴了,类似于下面这段代码。​​​​​​while(flag) {//your code;}这里的flag._main函数使用while(1)循环cpu占用99

【无标题】jetbrains idea shift f6不生效_idea shift +f6快捷键不生效-程序员宅基地

文章浏览阅读347次。idea shift f6 快捷键无效_idea shift +f6快捷键不生效

node.js学习笔记之Node中的核心模块_node模块中有很多核心模块,以下不属于核心模块,使用时需下载的是-程序员宅基地

文章浏览阅读135次。Ecmacript 中没有DOM 和 BOM核心模块Node为JavaScript提供了很多服务器级别,这些API绝大多数都被包装到了一个具名和核心模块中了,例如文件操作的 fs 核心模块 ,http服务构建的http 模块 path 路径操作模块 os 操作系统信息模块// 用来获取机器信息的var os = require('os')// 用来操作路径的var path = require('path')// 获取当前机器的 CPU 信息console.log(os.cpus._node模块中有很多核心模块,以下不属于核心模块,使用时需下载的是

数学建模【SPSS 下载-安装、方差分析与回归分析的SPSS实现(软件概述、方差分析、回归分析)】_化工数学模型数据回归软件-程序员宅基地

文章浏览阅读10w+次,点赞435次,收藏3.4k次。SPSS 22 下载安装过程7.6 方差分析与回归分析的SPSS实现7.6.1 SPSS软件概述1 SPSS版本与安装2 SPSS界面3 SPSS特点4 SPSS数据7.6.2 SPSS与方差分析1 单因素方差分析2 双因素方差分析7.6.3 SPSS与回归分析SPSS回归分析过程牙膏价格问题的回归分析_化工数学模型数据回归软件

利用hutool实现邮件发送功能_hutool发送邮件-程序员宅基地

文章浏览阅读7.5k次。如何利用hutool工具包实现邮件发送功能呢?1、首先引入hutool依赖<dependency> <groupId>cn.hutool</groupId> <artifactId>hutool-all</artifactId> <version>5.7.19</version></dependency>2、编写邮件发送工具类package com.pc.c..._hutool发送邮件

docker安装elasticsearch,elasticsearch-head,kibana,ik分词器_docker安装kibana连接elasticsearch并且elasticsearch有密码-程序员宅基地

文章浏览阅读867次,点赞2次,收藏2次。docker安装elasticsearch,elasticsearch-head,kibana,ik分词器安装方式基本有两种,一种是pull的方式,一种是Dockerfile的方式,由于pull的方式pull下来后还需配置许多东西且不便于复用,个人比较喜欢使用Dockerfile的方式所有docker支持的镜像基本都在https://hub.docker.com/docker的官网上能找到合..._docker安装kibana连接elasticsearch并且elasticsearch有密码

随便推点

Python 攻克移动开发失败!_beeware-程序员宅基地

文章浏览阅读1.3w次,点赞57次,收藏92次。整理 | 郑丽媛出品 | CSDN(ID:CSDNnews)近年来,随着机器学习的兴起,有一门编程语言逐渐变得火热——Python。得益于其针对机器学习提供了大量开源框架和第三方模块,内置..._beeware

Swift4.0_Timer 的基本使用_swift timer 暂停-程序员宅基地

文章浏览阅读7.9k次。//// ViewController.swift// Day_10_Timer//// Created by dongqiangfei on 2018/10/15.// Copyright 2018年 飞飞. All rights reserved.//import UIKitclass ViewController: UIViewController { ..._swift timer 暂停

元素三大等待-程序员宅基地

文章浏览阅读986次,点赞2次,收藏2次。1.硬性等待让当前线程暂停执行,应用场景:代码执行速度太快了,但是UI元素没有立马加载出来,造成两者不同步,这时候就可以让代码等待一下,再去执行找元素的动作线程休眠,强制等待 Thread.sleep(long mills)package com.example.demo;import org.junit.jupiter.api.Test;import org.openqa.selenium.By;import org.openqa.selenium.firefox.Firefox.._元素三大等待

Java软件工程师职位分析_java岗位分析-程序员宅基地

文章浏览阅读3k次,点赞4次,收藏14次。Java软件工程师职位分析_java岗位分析

Java:Unreachable code的解决方法_java unreachable code-程序员宅基地

文章浏览阅读2k次。Java:Unreachable code的解决方法_java unreachable code

标签data-*自定义属性值和根据data属性值查找对应标签_如何根据data-*属性获取对应的标签对象-程序员宅基地

文章浏览阅读1w次。1、html中设置标签data-*的值 标题 11111 222222、点击获取当前标签的data-url的值$('dd').on('click', function() { var urlVal = $(this).data('ur_如何根据data-*属性获取对应的标签对象

推荐文章

热门文章

相关标签