Android模仿微信錄音聘惦、發(fā)送語音效果實現(xiàn)

在項目開發(fā)中,有個需求:實現(xiàn)模仿微信錄音儒恋,發(fā)送語音的功能善绎。長按按鈕錄音,彈框顯示語音時間诫尽,以及上滑取消發(fā)送禀酱。我重寫了一個發(fā)送語音的控件,以實現(xiàn)該功能牧嫉。
首先添加權限:

<uses-permission android:name="android.permission.RECORD_AUDIO" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />

AudioRecorderButton剂跟,自己實現(xiàn)的自定義錄音控件

public class AudioRecorderButton extends Button {
    private static final int STATE_NORMAL = 1;// 默認的狀態(tài)
    private static final int STATE_RECORDING = 2;// 正在錄音
    private static final int STATE_WANT_TO_CANCEL = 3;// 希望取消

    private int mCurrentState = STATE_NORMAL; // 當前的狀態(tài)
    private boolean isRecording = false;// 已經(jīng)開始錄音

    private static final int DISTANCE_Y_CANCEL = 50;

    private DialogManager mDialogManager;
    private AudioManager mAudioManager;

    // 是否觸發(fā)longClick
    private boolean mReady;
    android.media.AudioManager audioManager;

    private static final int MSG_AUDIO_PREPARED = 0x110;
    private static final int MSG_VOICE_CHANGED = 0x111;
    private static final int MSG_DIALOG_DIMISS = 0x112;
    private static final int MSG_TIME_OUT = 0x113;
    private static final int UPDATE_TIME = 0x114;

    private boolean mThreadFlag = false;
    private int time = 0;
    private float mTime;

    /*
    * 獲取音量大小的線程
    */
    private Runnable mGetVoiceLevelRunnable = new Runnable() {
        public void run() {
            while (isRecording) {
                try {
                    Thread.sleep(100);
                    mTime += 0.1f;
                    time++;
                    if (isWantToCancel) {
                    } else {
                        if (time % 10 == 0) {
                            mHandler.sendEmptyMessage(UPDATE_TIME);
                        }
                    }
                    mHandler.sendEmptyMessage(MSG_VOICE_CHANGED);
                    if (mTime >= 10.0f) {//如果時間超過10秒,自動結(jié)束錄音
                        while (!mThreadFlag) {//記錄已經(jīng)結(jié)束了錄音酣藻,不需要再次結(jié)束曹洽,以免出現(xiàn)問題
                            mDialogManager.dimissDialog();
                            mAudioManager.release();
                            if (audioFinishRecorderListener != null) {
                                //發(fā)消息給主線程,告訴他reset();
                                mHandler.sendEmptyMessage(MSG_TIME_OUT);
                                audioFinishRecorderListener.onFinish(mTime, mAudioManager.getCurrentFilePath());
                            }
                            mThreadFlag = !mThreadFlag;
                        }
                        isRecording = false;
                    }
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    };

    private Handler mHandler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            switch (msg.what) {
                case MSG_AUDIO_PREPARED:
                    // 顯示對話框在開始錄音以后
                    mDialogManager.showRecordingDialog();
                    isRecording = true;
                    // 開啟一個線程
                    new Thread(mGetVoiceLevelRunnable).start();
                    break;
               case MSG_VOICE_CHANGED:
                    mDialogManager.updateVoiceLevel(mAudioManager.getVoiceLevel(7));
                   break;
                case MSG_DIALOG_DIMISS:
                    mDialogManager.dimissDialog();
                    break;
                case MSG_TIME_OUT://錄音超時
                    reset();
                    break;
                case UPDATE_TIME://更新時間
                    if (time % 10 == 0) {
                        mDialogManager.updateTime(time / 10);
                    }
                    break;
            }
        }
    };

    /**
     * 以下2個方法是構(gòu)造方法
     */
    public AudioRecorderButton(Context context, AttributeSet attrs) {
        super(context, attrs);
        mDialogManager = new DialogManager(context);
        audioManager = (android.media.AudioManager) context.getSystemService(Context.AUDIO_SERVICE);
        String dir = Environment.getExternalStorageDirectory().getPath() + "/" + LibraryContantsUtil.APP_NAME + "/audio";
        mAudioManager = AudioManager.getInstance(dir);
        mAudioManager.setOnAudioStateListener(new AudioManager.AudioStateListener() {
            public void wellPrepared() {
                if (!isOverDue) {
                    mHandler.sendEmptyMessage(MSG_AUDIO_PREPARED);//開啟線程
                }
            }
        });
        // 由于這個類是button所以在構(gòu)造方法中添加監(jiān)聽事件
        setOnLongClickListener(new OnLongClickListener() {
            public boolean onLongClick(View v) {
                if (!isOverDue) {
                    mReady = true;
                    mAudioManager.prepareAudio();
                }
                return true;
            }
        });
    }

    public AudioRecorderButton(Context context) {
        this(context, null);
    }

    /**
     * 錄音完成后的回調(diào)
     */
    public interface AudioFinishRecorderListener {
        void onFinish(float seconds, String filePath);
    }

    private AudioFinishRecorderListener audioFinishRecorderListener;

    public void setAudioFinishRecorderListener(AudioFinishRecorderListener listener) {
        audioFinishRecorderListener = listener;
    }

    android.media.AudioManager.OnAudioFocusChangeListener afChangeListener = new android.media.AudioManager.OnAudioFocusChangeListener() {
        public void onAudioFocusChange(int focusChange) {
            if (focusChange == AUDIOFOCUS_LOSS_TRANSIENT) {
                // Pause playback
            } else if (focusChange == android.media.AudioManager.AUDIOFOCUS_GAIN) {
                // Resume playback
                LogUtil.e("===", "+++");
            } else if (focusChange == android.media.AudioManager.AUDIOFOCUS_LOSS) {
                audioManager.abandonAudioFocus(afChangeListener);
                // Stop playback
            }
        }
    };

    public void myRequestAudioFocus() {
        audioManager.requestAudioFocus(afChangeListener, android.media.AudioManager.STREAM_MUSIC, android.media.AudioManager.AUDIOFOCUS_GAIN_TRANSIENT);
    }

    /**
     * 屏幕的觸摸事件
     */
    @Override
    public boolean onTouchEvent(MotionEvent event) {
//        int x = (int) event.getX();// 獲得x軸坐標
        float last = 0;// 獲得y軸坐標
        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:
                   mThreadFlag = false;
                   changeState(STATE_RECORDING);
                   myRequestAudioFocus();
                break;
            case MotionEvent.ACTION_MOVE:
                    if (isRecording) {
                        // 如果想要取消辽剧,根據(jù)x,y的坐標看是否需要取消
                        if (event.getY() < 0 && Math.abs(event.getY()) > 120) {
                            changeState(STATE_WANT_TO_CANCEL);
                        } else {
                            changeState(STATE_RECORDING);
                        }
                    }
                break;
            case MotionEvent.ACTION_UP:
                    if (!mReady) {
                        reset();
                        return super.onTouchEvent(event);
                    }
                    if (!isRecording || mTime <= 1.0f) {//小于1秒
                        mDialogManager.tooShort();
                        mAudioManager.cancel();
                        mHandler.sendEmptyMessage(MSG_DIALOG_DIMISS);//顯示對話框
                    } else if (mCurrentState == STATE_RECORDING) { // 正在錄音的時候送淆,結(jié)束
                        mDialogManager.dimissDialog();
                        mAudioManager.release();
                        if (audioFinishRecorderListener != null) {
                            audioFinishRecorderListener.onFinish(mTime, mAudioManager.getCurrentFilePath());
                        }
                    } else if (mCurrentState == STATE_WANT_TO_CANCEL) { // 想要取消
                        mDialogManager.dimissDialog();
                        mAudioManager.cancel();
                    }
                    reset();
                    audioManager.abandonAudioFocus(afChangeListener);
                }
                break;
        return super.onTouchEvent(event);
    }

    /**
     * 恢復狀態(tài)及標志位
     */
    private void reset() {
        isRecording = false;
        mTime = 0;
        time = 1;
        mReady = false;
        changeState(STATE_NORMAL);
    }

    private boolean wantToCancle(int x, int y) {
        if (y < 0) {
            return true;
        } else {
            return false;
        }
    }
    boolean isWantToCancel = false;
    /**
     * 改變
     */
    private void changeState(int state) {
        if (mCurrentState != state) {
            mCurrentState = state;
            switch (state) {
                case STATE_NORMAL:
                    setBackgroundResource(R.drawable.shape_voice_chat_circle);
                    setText("按住說話");
                    break;
                case STATE_RECORDING:
                    if (isRecording) {
                        mDialogManager.recording();
                    }
                    setBackgroundResource(R.drawable.shape_voice_chat_circle);
                    setText("松開結(jié)束");
                    isWantToCancel = false;
                    break;
                case STATE_WANT_TO_CANCEL:
                    setBackgroundResource(R.drawable.shape_voice_chat_circle);
                    setText("松開手指,取消發(fā)送");
                    mDialogManager.wantToCancel();
                    isWantToCancel = true;
                    break;
            }
        }
    }
}

下面是DialogManager抖仅,錄音時控制Dialog的顯示隱藏坊夫,以及需要Dialog中需要向用戶展示的內(nèi)容。

public class DialogManager {
    private AlertDialog.Builder builder;
    private ImageView mIcon;
    //    private ImageView mVoice;
    private TextView mLable;

    private Context context;

    private AlertDialog dialog;//用于取消AlertDialog.Builder

    /**
     * 構(gòu)造方法 傳入上下文
     */
    public DialogManager(Context context) {
        this.context = context;
    }

    // 顯示錄音的對話框
    public void showRecordingDialog() {
        builder = new AlertDialog.Builder(context, R.style.voice_chat_dialog);
        LayoutInflater inflater = LayoutInflater.from(context);
        View view = inflater.inflate(R.layout.dialog_recorder, null);

        mIcon = (ImageView) view.findViewById(R.id.id_recorder_dialog_icon);
        mLable = (TextView) view.findViewById(R.id.id_recorder_dialog_label);
//        GlideUtil.loadSquarePicture(R.drawable.gif_record, mIcon);
        builder.setView(view);
        builder.create();
        dialog = builder.show();
        dialog.setCanceledOnTouchOutside(false);//設置點擊外部不消失
    }

    public void recording() {
        if (dialog != null && dialog.isShowing()) { //顯示狀態(tài)
            mIcon.setVisibility(View.VISIBLE);
            mLable.setVisibility(View.VISIBLE);
//            GlideUtil.loadSquarePicture(R.drawable.gif_record, mIcon);
            mIcon.setImageResource(R.mipmap.ic_record);
            mLable.setText("手指上滑撤卢,取消發(fā)送");
        }
    }

    // 顯示想取消的對話框
    public void wantToCancel() {
        if (dialog != null && dialog.isShowing()) { //顯示狀態(tài)
            mIcon.setVisibility(View.VISIBLE);
            mLable.setVisibility(View.VISIBLE);
            mIcon.setImageResource(R.mipmap.ic_release_to_cancel);
            mLable.setText("松開手指,取消發(fā)送");
        }
    }

    public void updateTime(int time) {
        if (dialog != null && dialog.isShowing()) { //顯示狀態(tài)
            mIcon.setVisibility(View.VISIBLE);
            mLable.setVisibility(View.VISIBLE);
            mLable.setText(time + "s");
        }
    }

    // 顯示時間過短的對話框
    public void tooShort() {
        if (dialog != null && dialog.isShowing()) { //顯示狀態(tài)
            mIcon.setVisibility(View.VISIBLE);
            mLable.setVisibility(View.VISIBLE);
            mLable.setText("錄音時間過短");
        }
    }

    // 顯示取消的對話框
    public void dimissDialog() {
        if (dialog != null && dialog.isShowing()) { //顯示狀態(tài)
            dialog.dismiss();
            dialog = null;
        }
    }

    // 顯示更新音量級別的對話框
    public void updateVoiceLevel(int level) {
        if (dialog != null && dialog.isShowing()) { //顯示狀態(tài)
            mIcon.setVisibility(View.VISIBLE);
            mLable.setVisibility(View.VISIBLE);
//          在這里沒有做操作
//可以再這里根據(jù)音量大小梧兼,設置圖片放吩,實現(xiàn)效果;
        }
    }
}

下面是AudioManager羽杰,錄音類

public class AudioManager {
    private MediaRecorder mMediaRecorder;
    private String mDir;
    private String mCurrentFilePath;

    private static AudioManager mInstance;

    private boolean isPrepare;

    private AudioManager(String dir) {
        mDir = dir;
    }

    public static AudioManager getInstance(String dir) {
        if (mInstance == null) {
            synchronized (AudioManager.class) {
                if (mInstance == null) {
                    mInstance = new AudioManager(dir);
                }
            }
        }
        return mInstance;
    }

    /**
     * 使用接口 用于回調(diào)
     */
    public interface AudioStateListener {
        void wellPrepared();
    }

    public AudioStateListener mAudioStateListener;

    /**
     * 回調(diào)方法
     */
    public void setOnAudioStateListener(AudioStateListener listener) {
        mAudioStateListener = listener;
    }

    // 去準備
    public void prepareAudio() {
        try {
            isPrepare = false;
            File dir = new File(mDir);
            if (!dir.exists()) {
                dir.mkdirs();
            } else {
                if (!dir.isDirectory()) {
                    dir.delete();
                    dir.mkdirs();
                }
            }
            String fileName = generateFileName();
            File file = new File(dir, fileName);
            mCurrentFilePath = file.getAbsolutePath();
            mMediaRecorder = new MediaRecorder();
            // 設置輸出文件
            mMediaRecorder.setOutputFile(file.getAbsolutePath());
            // 設置MediaRecorder的音頻源為麥克風
            mMediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
            // 設置音頻格式
            mMediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.RAW_AMR);
            // 設置音頻編碼
            mMediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
            // 準備錄音
            mMediaRecorder.prepare();
            // 開始
            mMediaRecorder.start();
            // 準備結(jié)束
            isPrepare = true;
            if (mAudioStateListener != null) {
                mAudioStateListener.wellPrepared();
            }
        } catch (IllegalStateException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    /**
     * 隨機生成文件的名稱
     */
    private String generateFileName() {
        return UUID.randomUUID().toString() + ".amr";
    }

    public int getVoiceLevel(int maxlevel) {
        if (isPrepare) {
            try {
                return maxlevel * mMediaRecorder.getMaxAmplitude() / 32768 + 1;
            } catch (Exception e) {
            }
        }
        return 1;
    }

    /**
     * 釋放資源
     */
    public void release() {
        if (mMediaRecorder != null) {
            mMediaRecorder.reset();
            mMediaRecorder = null;
        }
    }

    /**
     * 取消錄音
     */
    public void cancel() {
        release();
        if (mCurrentFilePath != null) {
            File file = new File(mCurrentFilePath);
            file.delete();
            mCurrentFilePath = null;
        }
    }

    public String getCurrentFilePath() {
        return mCurrentFilePath;
    }
}

下面是MediaManager渡紫,播放,暫停錄音

public class MediaManager {
    private static MediaPlayer mMediaPlayer;
    private static boolean isPause;
    private String currentFilePath;
    private AudioManager.AudioStateListener onAudioStateListener;
    static Context context;

    public MediaManager(Context context) {
        this.context = context;
    }

    /**
     * 播放音樂
     *
     * @param filePath
     * @param onCompletionListener
     */
    public static void playSound(String filePath, MediaPlayer.OnCompletionListener onCompletionListener) {
        if (mMediaPlayer == null) {
            mMediaPlayer = new MediaPlayer();
//            mMediaPlayer = MediaPlayer.create(context, Uri.fromFile(new File(filePath)));
            //設置一個error監(jiān)聽器
            mMediaPlayer.setOnErrorListener(new MediaPlayer.OnErrorListener() {
                public boolean onError(MediaPlayer arg0, int arg1, int arg2) {
                    mMediaPlayer.reset();
                    return false;
                }
            });
        } else {
            mMediaPlayer.reset();
        }
        try {
            mMediaPlayer.setAudioStreamType(android.media.AudioManager.STREAM_MUSIC);
            mMediaPlayer.setOnCompletionListener(onCompletionListener);
            mMediaPlayer.setDataSource(filePath);
            mMediaPlayer.prepare();
            mMediaPlayer.start();
        } catch (Exception e) {

        }
    }

    /**
     * 暫停播放
     */
    public static void pause() {
        if (mMediaPlayer != null && mMediaPlayer.isPlaying()) { 
//正在播放的時候
            mMediaPlayer.pause();
            isPause = true;
        }
    }

    public static boolean isPlaying() {
        if (mMediaPlayer != null && mMediaPlayer.isPlaying()) {
            return true;
        } else {
            return false;
        }
    }

    /**
     * 當前是isPause狀態(tài)
     */
    public static void resume() {
        if (mMediaPlayer != null && isPause) {
            mMediaPlayer.start();
            isPause = false;
        }
    }

    /**
     * 釋放資源
     */
    public static void release() {
        if (mMediaPlayer != null) {
            mMediaPlayer.release();
            mMediaPlayer = null;
        }
    }
}

至此考赛,其整體的代碼已經(jīng)編寫完成了惕澎,在需要錄音的地方調(diào)用:

 audioRecorderButton.setAudioFinishRecorderListener(new AudioRecorderButton.AudioFinishRecorderListener() {
            @Override
            public void onFinish(float seconds, String filePath) {
//seconds 錄音時長,filePath 錄音文件的位置
                //錄音完成
                upLoadVoice(filePath, (int) seconds);
            }
        });
Screenshot_2017-07-25-16-24-38-397_com.zijing.xin.png
最后編輯于
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末颜骤,一起剝皮案震驚了整個濱河市唧喉,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖八孝,帶你破解...
    沈念sama閱讀 212,454評論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件董朝,死亡現(xiàn)場離奇詭異,居然都是意外死亡干跛,警方通過查閱死者的電腦和手機子姜,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,553評論 3 385
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來楼入,“玉大人哥捕,你說我怎么就攤上這事〖涡埽” “怎么了扭弧?”我有些...
    開封第一講書人閱讀 157,921評論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀的道長记舆。 經(jīng)常有香客問我鸽捻,道長,這世上最難降的妖魔是什么泽腮? 我笑而不...
    開封第一講書人閱讀 56,648評論 1 284
  • 正文 為了忘掉前任御蒲,我火速辦了婚禮,結(jié)果婚禮上诊赊,老公的妹妹穿的比我還像新娘厚满。我一直安慰自己,他們只是感情好碧磅,可當我...
    茶點故事閱讀 65,770評論 6 386
  • 文/花漫 我一把揭開白布碘箍。 她就那樣靜靜地躺著,像睡著了一般鲸郊。 火紅的嫁衣襯著肌膚如雪丰榴。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,950評論 1 291
  • 那天秆撮,我揣著相機與錄音四濒,去河邊找鬼。 笑死职辨,一個胖子當著我的面吹牛盗蟆,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播舒裤,決...
    沈念sama閱讀 39,090評論 3 410
  • 文/蒼蘭香墨 我猛地睜開眼喳资,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了腾供?” 一聲冷哼從身側(cè)響起仆邓,我...
    開封第一講書人閱讀 37,817評論 0 268
  • 序言:老撾萬榮一對情侶失蹤鲜滩,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后宏赘,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體绒北,經(jīng)...
    沈念sama閱讀 44,275評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,592評論 2 327
  • 正文 我和宋清朗相戀三年察署,在試婚紗的時候發(fā)現(xiàn)自己被綠了闷游。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,724評論 1 341
  • 序言:一個原本活蹦亂跳的男人離奇死亡贴汪,死狀恐怖脐往,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情扳埂,我是刑警寧澤业簿,帶...
    沈念sama閱讀 34,409評論 4 333
  • 正文 年R本政府宣布,位于F島的核電站阳懂,受9級特大地震影響梅尤,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜岩调,卻給世界環(huán)境...
    茶點故事閱讀 40,052評論 3 316
  • 文/蒙蒙 一巷燥、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧号枕,春花似錦缰揪、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,815評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至赞厕,卻和暖如春艳狐,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背坑傅。 一陣腳步聲響...
    開封第一講書人閱讀 32,043評論 1 266
  • 我被黑心中介騙來泰國打工僵驰, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人唁毒。 一個月前我還...
    沈念sama閱讀 46,503評論 2 361
  • 正文 我出身青樓,卻偏偏與公主長得像星爪,于是被迫代替她去往敵國和親浆西。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 43,627評論 2 350

推薦閱讀更多精彩內(nèi)容