Android自定義View實(shí)現(xiàn)微信語音界面
前言
因?yàn)樽罱捻?xiàng)目需要使用錄音功能,開始的想法是Button+OnTouchListener+Dialog實(shí)現(xiàn),在大部分手機(jī)中都沒問題,只有MI8會偶爾無法觸發(fā)MotionEvent.ACTION_UP,導(dǎo)致程序異常。所以就自己寫了個(gè)自定義View來實(shí)現(xiàn),主要也是通過監(jiān)聽
OnTouchListener+Dialog來實(shí)現(xiàn)。這里只實(shí)現(xiàn)了自定義View,并不涉及錄音和播放。效果圖如下:
代碼
代碼并不復(fù)雜,配合注釋應(yīng)該很容易理解。
/** * Author : BlackHao * Time : 2019/4/18 14:03 * Description : 自定義錄音按鈕布局界面 */ public class PressedView extends View implements View.OnTouchListener { private int normalRes; private String normalText = ""; private int pressedRes; private String pressedText = ""; // private Paint paint; private Rect rect; //當(dāng)前是否是按下狀態(tài) private boolean isPressed = false; // private PressCallback callback; //按下的位置y坐標(biāo) private int pressedY = 0; //當(dāng)前是否是outSize private boolean isOutSize = false; //字體dp大小 private static int TEXT_SIZE = 20; //對話框相關(guān) private Dialog soundVolumeDialog = null; //音量圖片 private ImageView soundVolumeImg = null; //對話框背景 private RelativeLayout soundVolumeLayout = null; public PressedView(Context context) { super(context); init(); } public PressedView(Context context, @Nullable AttributeSet attrs) { super(context, attrs); init(); } public PressedView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); init(); } private void init() { // paint = new Paint(); paint.setAntiAlias(true); paint.setTextSize(DensityUtil.dip2px(getContext(), TEXT_SIZE)); paint.setColor(Color.WHITE); rect = new Rect(); // normalRes = R.drawable.blue_btn_bk; normalText = "按住 說話"; pressedRes = R.drawable.red_btn_bk; pressedText = "松開 結(jié)束"; // setOnTouchListener(this); // initSoundVolumeDlg(); } @Override protected void onDraw(Canvas canvas) { super.onDraw(canvas); rect.set(0, 0, getWidth(), getHeight()); if (!isPressed) { setBackgroundResource(normalRes); drawTextOnRect(canvas, rect, normalText); } else { setBackgroundResource(pressedRes); drawTextOnRect(canvas, rect, pressedText); } } @Override public boolean onTouch(View v, MotionEvent event) { switch (event.getAction()) { case MotionEvent.ACTION_DOWN: pressedY = (int) event.getRawY(); isOutSize = false; if (!isPressed) { isPressed = true; postInvalidate(); if (callback != null) { //回調(diào) callback.onStartRecord(); //按下,彈出對話框 soundVolumeImg.setImageResource(R.mipmap.sound_volume_01); soundVolumeImg.setVisibility(View.VISIBLE); soundVolumeLayout.setBackgroundResource(R.mipmap.sound_volume_default_bk); soundVolumeDialog.show(); } } break; case MotionEvent.ACTION_UP: if (isPressed) { isPressed = false; postInvalidate(); if (callback != null) { int upY = (int) event.getRawY(); if (pressedY - upY < getHeight()) { //錄音結(jié)束 if (soundVolumeDialog.isShowing()) { soundVolumeDialog.dismiss(); } callback.onStopRecord(); } else { //錄音取消 if (soundVolumeDialog.isShowing()) { soundVolumeDialog.dismiss(); } callback.onCancelRecord(); } } } break; case MotionEvent.ACTION_MOVE: if (isPressed && callback != null) { int upY = (int) event.getRawY(); if (pressedY - upY < getHeight()) { if (isOutSize) { isOutSize = false; soundVolumeLayout.setBackgroundResource(R.mipmap.sound_volume_default_bk); } } else { if (!isOutSize) { isOutSize = true; soundVolumeLayout.setBackgroundResource(R.mipmap.sound_volume_cancel_bk); } } } break; } return true; } public void setCallback(PressCallback callback) { this.callback = callback; } public interface PressCallback { //開始錄音 void onStartRecord(); //停止錄音 void onStopRecord(); //取消錄音 void onCancelRecord(); } /** * 在指定矩形中間drawText * * @param canvas 畫布 * @param targetRect 指定矩形 * @param text 需要繪制的Text */ private void drawTextOnRect(Canvas canvas, Rect targetRect, String text) { Paint.FontMetricsInt fontMetrics = paint.getFontMetricsInt(); // 獲取baseLine int baseline = targetRect.top + (targetRect.bottom - targetRect.top - fontMetrics.bottom + fontMetrics.top) / 2 - fontMetrics.top; // 下面這行是實(shí)現(xiàn)水平居中,drawText對應(yīng)改為傳入targetRect.centerX() paint.setTextAlign(Paint.Align.CENTER); canvas.drawText(text, targetRect.centerX(), baseline, paint); } /** * 初始化音量信息對話框 */ private void initSoundVolumeDlg() { soundVolumeDialog = new Dialog(getContext(), R.style.SoundVolumeStyle); soundVolumeDialog.requestWindowFeature(Window.FEATURE_NO_TITLE); soundVolumeDialog.getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN); soundVolumeDialog.setContentView(R.layout.tt_sound_volume_dialog); soundVolumeDialog.setCanceledOnTouchOutside(true); soundVolumeImg = (ImageView) soundVolumeDialog.findViewById(R.id.sound_volume_img); soundVolumeLayout = (RelativeLayout) soundVolumeDialog.findViewById(R.id.sound_volume_bk); } /** * 根據(jù)分貝值設(shè)置錄音時(shí)的音量動畫 */ public void setVolume(int voiceValue) { if (voiceValue < 200.0) { soundVolumeImg.setImageResource(R.mipmap.sound_volume_01); } else if (voiceValue > 200.0 && voiceValue < 600) { soundVolumeImg.setImageResource(R.mipmap.sound_volume_02); } else if (voiceValue > 600.0 && voiceValue < 1200) { soundVolumeImg.setImageResource(R.mipmap.sound_volume_03); } else if (voiceValue > 1200.0 && voiceValue < 2400) { soundVolumeImg.setImageResource(R.mipmap.sound_volume_04); } else if (voiceValue > 2400.0 && voiceValue < 10000) { soundVolumeImg.setImageResource(R.mipmap.sound_volume_05); } else if (voiceValue > 10000.0 && voiceValue < 28000.0) { soundVolumeImg.setImageResource(R.mipmap.sound_volume_06); } else if (voiceValue > 28000.0) { soundVolumeImg.setImageResource(R.mipmap.sound_volume_07); } } }
結(jié)語
源碼github地址:仿微信語音界面
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android 多圖上傳后將圖片進(jìn)行九宮格展示的實(shí)例代碼
這篇文章主要介紹了Android 多圖上傳后將圖片進(jìn)行九宮格展示,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-11-11Kotlin類型安全構(gòu)建器的一次運(yùn)用記錄
這篇文章主要給大家介紹了關(guān)于Kotlin類型安全構(gòu)建器的一次運(yùn)用,文中通過示例代碼介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用Kotlin具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2020-06-06android實(shí)現(xiàn)音樂播放器進(jìn)度條效果
這篇文章主要為大家詳細(xì)介紹了android實(shí)現(xiàn)音樂播放器進(jìn)度條效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-04-04Android四大組件之Service服務(wù)詳細(xì)講解
Android的服務(wù)是開發(fā)Android應(yīng)用程序的重要組成部分。不同于活動Activity,服務(wù)是在后臺運(yùn)行,服務(wù)沒有接口,生命周期也與活動Activity非常不同。通過使用服務(wù)我們可以實(shí)現(xiàn)一些后臺操作,比如想從遠(yuǎn)程服務(wù)器加載一個(gè)網(wǎng)頁等,下面來看看詳細(xì)內(nèi)容,需要的朋友可以參考下2022-07-07Android開發(fā)之MediaPlayer多媒體(音頻,視頻)播放工具類
這篇文章主要介紹了Android開發(fā)之MediaPlayer多媒體(音頻,視頻)播放工具類,涉及Android針對音頻文件的讀取、播放、暫停、繼續(xù)等操作實(shí)現(xiàn)技巧,需要的朋友可以參考下2017-12-12