import android.content.Context;
import android.os.Environment;
import android.os.Handler;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.widget.Button;
public class AudioRecordButton extends Button {
private static final int STATE_NORMAL = 1;
private static final int STATE_RECORDING = 2;
private static final int STATE_WANT_CANCEL = 3;
private static final int DISTANCE_CANCEL_Y = 50;
private int currentState = STATE_NORMAL;
private boolean isRecording = false;
private AudioRecordDialog dialogManager;
private MyAudioManager audioManager;
private float mTime;
// 是否触发LongClick
private boolean isReady = false;
public AudioRecordButton(Context context) {
this(context, null);
}
public AudioRecordButton(Context context, AttributeSet attrs) {
super(context, attrs);
dialogManager = new AudioRecordDialog(getContext());
String dir = Environment.getExternalStorageDirectory()
- “/zms_chat_audios”;
audioManager = MyAudioManager.getInstance(dir);
audioManager
.setOnAudioStateChangeListener(new MyOnAudioStateChangeListener());
setOnLongClickListener(new OnLongClickListener() {
@Override
public boolean onLongClick(View v) {
isReady = true;
audioManager.prepareAudio();
return false;
}
});
}
class MyOnAudioStateChangeListener implements AudioStateChangeListener {
@Override
public void wellPrepared() {
mHanlder.sendEmptyMessage(MSG_AUDIO_PREPARED);
}
}
/**
-
录音完成后的回调
*/
public interface AudioRecordFinishListener {
void onFinish(float second, String filePath);
}
private AudioRecordFinishListener audioRecordFinishListener;
public void setAudioRecordFinishListener(AudioRecordFinishListener listener) {
audioRecordFinishListener = listener;
}
private Runnable getVolumeRunnable = new Runnable() {
@Override
public void run() {
while (isRecording) {
try {
Thread.sleep(100);
mTime += 0.1f;
mHanlder.sendEmptyMessage(MSG_VOLUME_CHAMGED);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}
};
private static final int MSG_AUDIO_PREPARED = 0x110;
private static final int MSG_VOLUME_CHAMGED = 0x111;
private static final int MSG_DIALOG_DISMISS = 0x112;
private Handler mHanlder = new Handler() {
public void handleMessage(android.os.Message msg) {
switch (msg.what) {
case MSG_AUDIO_PREPARED:
dialogManager.showDialog();
isRecording = true;
// 音量
new Thread(getVolumeRunnable).start();
break;
case MSG_VOLUME_CHAMGED:
dialogManager.updateVolumeLevel(audioManager.getVoiceLevel(7));
break;
case MSG_DIALOG_DISMISS:
dialogManager.dismissDialog();
break;
default:
break;
}
};
};
@Override
public boolean onTouchEvent(MotionEvent event) {
int action = event.getAction();
int x = (int) event.getX();
int y = (int) event.getY();
switch (action) {
case MotionEvent.ACTION_DOWN:
changeState(STATE_RECORDING);
break;
case MotionEvent.ACTION_MOVE:
// 已经开始录音
if (isRecording) {
// 根据X,Y的坐标判断是否想要取消
if (wantCancel(x, y)) {
changeState(STATE_WANT_CANCEL);
dialogManager.stateWantCancel();
} else {
changeState(STATE_RECORDING);
dialogManager.stateRecording();
}
}
break;
case MotionEvent.ACTION_UP:
// 没有触发longClick
if (!isReady) {
resetState();
return super.onTouchEvent(event);
}
// prepare未完成就up,录音时间过短
if (!isRecording || mTime < 0.6f) {
dialogManager.stateLengthShort();
audioManager.cancel();
mHanlder.sendEmptyMessageDelayed(MSG_DIALOG_DISMISS, 1300);
} else if (currentState == STATE_RECORDING) { // 正常录制结束
dialogManager.dismissDialog();
audioManager.release();
// callbackToActivity
if (audioRecordFinishListener != null) {
audioRecordFinishListener.onFinish(mTime,
audioManager.getCurrentPath());
}
} else if (currentState == STATE_WANT_CANCEL) {
dialogManager.dismissDialog();
audioManager.cancel();
}
resetState();
break;
default:
break;
}
return super.onTouchEvent(event);
}
/**
- 恢复标志位
*/
private void resetState() {
isRecording = false;
isReady = false;
changeState(STATE_NORMAL);
mTime = 0;
}
private boolean wantCancel(int x, int y) {
if (x < 0 || x > getWidth()) {
return true;
}
// 零点在左下角?
if (y < -DISTANCE_CANCEL_Y || y > getHeight() + DISTANCE_CANCEL_Y) {
return true;
}
return false;
}
private void changeState(int state) {
if (currentState != state) {
currentState = state;
switch (state) {
case STATE_NORMAL:
setBackgroundResource(R.drawable.btn_recorder_normal);
setText(R.string.btn_recorder_normal);
break;
case STATE_RECORDING:
setBackgroundResource(R.drawable.btn_recorder_normal);
setText(R.string.btn_recorder_recording);
if (isRecording) {
dialogManager.stateRecording();
}
break;
case STATE_WANT_CANCEL:
setBackgroundResource(R.drawable.btn_recorder_normal);
setText(R.string.btn_recorder_want_cancel);
dialogManager.stateWantCancel();
break;
default:
break;
}
}
}
}
自定义Dialog
package com.zms.wechatrecorder.view;
import com.zms.wechatrecorder.R;
import android.app.Dialog;
import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.widget.ImageView;
import android.widget.TextView;
public class AudioRecordDialog {
private Dialog dialog;
private ImageView imageRecord, imageVolume;
private TextView textHint;
private Context context;
public AudioRecordDialog(Context context) {
this.context = context;
}
public void showDialog() {
dialog = new Dialog(context, R.style.Theme_RecorderDialog);
LayoutInflater inflater = LayoutInflater.from(context);
View view = inflater.inflate(R.layout.dialog, null);
dialog.setContentView(view);
imageRecord = (ImageView) dialog.findViewById(R.id.imageRecord);
imageVolume = (ImageView) dialog.findViewById(R.id.imageVolume);
textHint = (TextView) dialog.findViewById(R.id.textHint);
dialog.show();
}
public void stateRecording() {
if (dialog != null && dialog.isShowing()) {
imageRecord.setVisibility(View.VISIBLE);
imageVolume.setVisibility(View.VISIBLE);
textHint.setVisibility(View.VISIBLE);
imageRecord.setImageResource(R.drawable.icon_dialog_recording);
textHint.setText(“手指上滑,取消发送”);
}
}
public void stateWantCancel() {
if (dialog != null && dialog.isShowing()) {
imageRecord.setVisibility(View.VISIBLE);
imageRecord.setImageResource(R.drawable.icon_dialog_cancel);
imageVolume.setVisibility(View.GONE);
textHint.setVisibility(View.VISIBLE);
textHint.setText(“松开手指,取消发送”);
}
}
public void stateLengthShort() {
if (dialog != null && dialog.isShowing()) {
imageRecord.setVisibility(View.VISIBLE);
imageRecord.setImageResource(R.drawable.icon_dialog_length_short);
imageVolume.setVisibility(View.GONE);
textHint.setVisibility(View.VISIBLE);
textHint.setText(“录音时间过短”);
}
}
public void dismissDialog() {
if (dialog != null && dialog.isShowing()) {
dialog.dismiss();
dialog = null;
}
}
/**
-
更新音量
-
@param level
*/
public void updateVolumeLevel(int level) {
if (dialog != null && dialog.isShowing()) {
// imageRecord.setVisibility(View.VISIBLE);
// imageVolume.setVisibility(View.VISIBLE);
// textHint.setVisibility(View.VISIBLE);
int volumeResId = context.getResources().getIdentifier(
“icon_volume_” + level, “drawable”,
context.getPackageName());
imageVolume.setImageResource(volumeResId);
}
}
}
VoiceListAdapter:
package com.zms.wechatrecorder;
import java.util.List;
import com.zms.wechatrecorder.MainActivity.Recorder;
import android.content.Context;
import android.util.DisplayMetrics;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.view.WindowManager;
import android.widget.ArrayAdapter;
import android.widget.TextView;
public class VoiceListAdapter extends ArrayAdapter {
private List mDatas;
private Context context;
private int minItemWidth;
private int maxItemWidth;
private LayoutInflater inflater;
public VoiceListAdapter(Context context, List datas) {
super(context, -1, datas);
this.context = context;
mDatas = datas;
WindowManager wm = (WindowManager) context
.getSystemService(Context.WINDOW_SERVICE);
DisplayMetrics outMetrics = new DisplayMetrics();
wm.getDefaultDisplay().getMetrics(outMetrics);
maxItemWidth = (int) (outMetrics.widthPixels * 0.8);
maxItemWidth = (int) (outMetrics.widthPixels * 0.2);
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
ViewHolder holder = null;
if (convertView == null) {
inflater = LayoutInflater.from(getContext());
convertView = inflater.inflate(R.layout.list_item_voice, parent,
false);
holder = new ViewHolder();
holder.seconds = (TextView) convertView
.findViewById(R.id.textLength);
holder.length = convertView.findViewById(R.id.voiceAnim);
convertView.setTag(holder);
} else {
holder = (ViewHolder) convertView.getTag();
}
holder.seconds.setText(Math.round(getItem(position).audioLength) + “”");
// ViewGroup.LayoutParams params = holder.length.getLayoutParams();
// params.width = (int) (minItemWidth + maxItemWidth / 60f
// * getItem(position).audioLength);
// holder.length.setLayoutParams(params);
return convertView;
}
private class ViewHolder {
TextView seconds;
View length;
}
}
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)
总结
**其实上面说了这么多,钱是永远赚不完的,在这个知识付费的时代,知识技能提升才是是根本!我作为一名8年的高级工程师,知识技能已经学习的差不多。**在看这篇文章的可能有刚刚入门,刚刚开始工作,或者大佬级人物。
像刚刚开始学Android开发小白想要快速提升自己,最快捷的方式,就是有人可以带着你一起分析,这样学习起来最为高效,所以这里分享一套高手学习的源码和框架视频等精品Android架构师教程,保证你学了以后保证薪资上升一个台阶。
这么重要的事情说三遍啦!点赞+点赞+点赞!
【Android高级架构师系统学习资料】高级架构师进阶必备——设计思想解读开源框架
第一章、热修复设计
第二章、插件化框架设计
第三章、组件化框架设计
第四章、图片加载框架
第五章、网络访问框架设计
第六章、RXJava 响应式编程框架设计
第七章、IOC 架构设计
第八章、Android 架构组件 Jetpack
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!
基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!**
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)
总结
**其实上面说了这么多,钱是永远赚不完的,在这个知识付费的时代,知识技能提升才是是根本!我作为一名8年的高级工程师,知识技能已经学习的差不多。**在看这篇文章的可能有刚刚入门,刚刚开始工作,或者大佬级人物。
像刚刚开始学Android开发小白想要快速提升自己,最快捷的方式,就是有人可以带着你一起分析,这样学习起来最为高效,所以这里分享一套高手学习的源码和框架视频等精品Android架构师教程,保证你学了以后保证薪资上升一个台阶。
这么重要的事情说三遍啦!点赞+点赞+点赞!
[外链图片转存中…(img-BPCOyuNZ-1712151434573)]
【Android高级架构师系统学习资料】高级架构师进阶必备——设计思想解读开源框架
第一章、热修复设计
第二章、插件化框架设计
第三章、组件化框架设计
第四章、图片加载框架
第五章、网络访问框架设计
第六章、RXJava 响应式编程框架设计
第七章、IOC 架构设计
第八章、Android 架构组件 Jetpack
[外链图片转存中…(img-E7Vc7b9B-1712151434573)]