溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶(hù)服務(wù)條款》

Android如何實(shí)現(xiàn)客戶(hù)端語(yǔ)音動(dòng)彈界面

發(fā)布時(shí)間:2021-04-16 12:52:59 來(lái)源:億速云 閱讀:152 作者:小新 欄目:移動(dòng)開(kāi)發(fā)

這篇文章主要介紹Android如何實(shí)現(xiàn)客戶(hù)端語(yǔ)音動(dòng)彈界面,文中介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們一定要看完!

具體內(nèi)容如下:

Android如何實(shí)現(xiàn)客戶(hù)端語(yǔ)音動(dòng)彈界面

首先,整體界面分三部分,最上層自定義ActionBar相信不需要我講大家就能看出來(lái)了。

中間部分是文字動(dòng)彈部分,主體就是一個(gè)設(shè)置了Padding(margin)的EditText,在EditText下面是一個(gè)剩余輸入字?jǐn)?shù)的描述。其實(shí)在“您還可以輸入XX字”的左邊還有一個(gè)用于顯示錄音圖標(biāo)的ImageView.

最下層是本文主要講解的錄音自定義控件的實(shí)現(xiàn)。

下面一整塊整體都是自定義控件的區(qū)域,我將其命名為RecordButton,是一個(gè)繼承自RelateiveLayout的ViewGroup。

在其中包括了左中右三個(gè)ImageView:試聽(tīng)與刪除,中間的錄音按鈕。

在錄音按鈕的上下各有一個(gè)用于提示的TextView。

整體布局的載入可以通過(guò)調(diào)用 View.inflater(cxt, R.layout.xxx, null); 就行了。

同前一篇講的一樣,作為控件界面控制邏輯,我們主要看一下onTouchEvent方法:當(dāng)手指按下的時(shí)候,初始化錄音器。手指在屏幕上移動(dòng)的時(shí)候如果滑到按鈕之上的時(shí)候,event.getY會(huì)返回一個(gè)負(fù)值(因?yàn)榛隹丶寺铮?。這里我寫(xiě)的是-50主要是為了多一點(diǎn)緩沖,防止誤操作。

先來(lái)看代碼

@Override
 public boolean onTouchEvent(MotionEvent event) {
  if (mAudioFile == null) {
   return false;
  }
  if (!mTouchInPlayButton) {
   return false;
  }
  switch (event.getAction()) {
  case MotionEvent.ACTION_DOWN:
   initBorderLine();
   break;
  case MotionEvent.ACTION_MOVE:
   if (event.getY() < 0) {
    viewToInit();
    break;
   }
   if (event.getX() > mRightButtonX) {
    mIsCancel = true;
    scaleView(mImgDelete, 1.5f);
   } else if (event.getX() < mLeftButtonX) {
    scaleView(mImgListen, 1.5f);
   } else {
    mIsCancel = false;
    viewToInit();
   }
   break;
  case MotionEvent.ACTION_UP:
   if (mIsCancel || event.getY() < -50) {
    cancelRecord();
   } else if (event.getX() < mLeftButtonX) {// 試聽(tīng)
    playRecord();
    finishRecord();
   } else if (event.getX() > mRightButtonX) {// 刪除
    cancelRecord();
   } else {
    finishRecord();
   }
   viewToInit();
   bottomFlag.setVisibility(View.VISIBLE);
   topFlag.setVisibility(View.GONE);
   mIsCancel = false;
   mTouchInPlayButton = false;
   break;
  }
  return true;
 }

其中錄音相關(guān)的工具類(lèi)還是和之前的一樣,這就是把功能與視圖分開(kāi)的好處,隨時(shí)用隨時(shí)復(fù)制粘貼過(guò)來(lái)就用了。

/**
 * {@link #RecordButton}需要的工具類(lèi)
 * 
 * @author kymjs(http://www.kymjs.com/)
 */
public class RecordButtonUtil {
 public static final String AUDOI_DIR = Environment
   .getExternalStorageDirectory().getAbsolutePath() + "/oschina/audio"; // 錄音音頻保存根路徑
 private String mAudioPath; // 要播放的聲音的路徑
 private boolean mIsRecording;// 是否正在錄音
 private boolean mIsPlaying;// 是否正在播放
 private OnPlayListener listener;
 // 初始化 錄音器
 private void initRecorder() {
  mRecorder = new MediaRecorder();
  mRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
  mRecorder.setOutputFormat(MediaRecorder.OutputFormat.AMR_NB);
  mRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
  mRecorder.setOutputFile(mAudioPath);
  mIsRecording = true;
 }
 /** 開(kāi)始錄音,并保存到文件中 */
 public void recordAudio() {
  initRecorder();
  try {
   mRecorder.prepare();
  } catch (IOException e) {
   e.printStackTrace();
  }
  mRecorder.start();
 }
 /** 獲取音量值,只是針對(duì)錄音音量 */
 public int getVolumn() {
  int volumn = 0;
  // 錄音
  if (mRecorder != null && mIsRecording) {
   volumn = mRecorder.getMaxAmplitude();
   if (volumn != 0)
    volumn = (int) (10 * Math.log(volumn) / Math.log(10)) / 7;
  }
  return volumn;
 }
 /** 停止錄音 */
 public void stopRecord() {
  if (mRecorder != null) {
   mRecorder.stop();
   mRecorder.release();
   mRecorder = null;
   mIsRecording = false;
  }
 }
 public void startPlay(String audioPath) {
  if (!mIsPlaying) {
   if (!StringUtils.isEmpty(audioPath)) {
    mPlayer = new MediaPlayer();
    try {
     mPlayer.setDataSource(audioPath);
     mPlayer.prepare();
     mPlayer.start();
     if (listener != null) {
      listener.starPlay();
     }
     mIsPlaying = true;
     mPlayer.setOnCompletionListener(new MediaPlayer.OnCompletionListener() {
      @Override
      public void onCompletion(MediaPlayer mp) {
       if (listener != null) {
        listener.stopPlay();
       }
       mp.release();
       mPlayer = null;
       mIsPlaying = false;
      }
     });
    } catch (Exception e) {
     e.printStackTrace();
    }
   } else {
    AppContext.showToastShort(R.string.record_sound_notfound);
   }
  } // end playing
 }
 public interface OnPlayListener {
  /** 播放聲音結(jié)束時(shí)調(diào)用 */
  void stopPlay();
  /** 播放聲音開(kāi)始時(shí)調(diào)用 */
  void starPlay();
 }
}

如果細(xì)心,你會(huì)發(fā)現(xiàn)左右兩個(gè)圓形按鈕,會(huì)隨著手指移動(dòng)到上面的時(shí)候放大,這其實(shí)也是一個(gè)通過(guò)監(jiān)聽(tīng)ontouch事件,對(duì)兩個(gè)圓形按鈕設(shè)置動(dòng)畫(huà)產(chǎn)生的效果,和諧帶人就是下面這句了。(注,setScaleX和setScaleY方法只有在A(yíng)PI10,也就是3.0以上的版本才能調(diào)用):

if (event.getX() > mRightButtonX) {
 mIsCancel = true;scaleView(mImgDelete, 1.5f);
} else if (event.getX() < mLeftButtonX) {
 scaleView(mImgListen, 1.5f);
} else {
 mIsCancel = false;viewToInit();
}
private void scaleView(View view, float scaleXY) {
 if (android.os.Build.VERSION.SDK_INT > 10) {
  view.setScaleX(scaleXY);
  view.setScaleY(scaleXY);
 }
}

以上是“Android如何實(shí)現(xiàn)客戶(hù)端語(yǔ)音動(dòng)彈界面”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對(duì)大家有幫助,更多相關(guān)知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀(guān)點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI