Android 仿微信语音聊天(1)

  • “/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;

}

}

MyAudioManager:

package com.zms.wechatrecorder;

import java.io.File;

import java.io.IOException;

import java.util.UUID;

import android.media.MediaRecorder;

public class MyAudioManager {

private MediaRecorder mediaRecorder;

private String dir;

private String currentFilePath;

private static MyAudioManager audioInstance; // 单例

public boolean isPrepared = false;

private MyAudioManager(String dir) {

this.dir = dir;

}

public interface AudioStateChangeListener {

void wellPrepared();

}

public AudioStateChangeListener audioStateChangeListener;

public void setOnAudioStateChangeListener(AudioStateChangeListener listener) {

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

最后

今天关于面试的分享就到这里,还是那句话,有些东西你不仅要懂,而且要能够很好地表达出来,能够让面试官认可你的理解,例如Handler机制,这个是面试必问之题。有些晦涩的点,或许它只活在面试当中,实际工作当中你压根不会用到它,但是你要知道它是什么东西。

最后在这里小编分享一份自己收录整理上述技术体系图相关的几十套腾讯、头条、阿里、美团等公司2021年的面试题,把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分。

还有 高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。

【算法合集】

【延伸Android必备知识点】

【Android部分高级架构视频学习资源】

**Android精讲视频领取学习后更加是如虎添翼!**进军BATJ大厂等(备战)!现在都说互联网寒冬,其实无非就是你上错了车,且穿的少(技能),要是你上对车,自身技术能力够强,公司换掉的代价大,怎么可能会被裁掉,都是淘汰末端的业务Curd而已!现如今市场上初级程序员泛滥,这套教程针对Android开发工程师1-6年的人员、正处于瓶颈期,想要年后突破自己涨薪的,进阶Android中高级、架构师对你更是如鱼得水,赶快领取吧!

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

,还是那句话,有些东西你不仅要懂,而且要能够很好地表达出来,能够让面试官认可你的理解,例如Handler机制,这个是面试必问之题。有些晦涩的点,或许它只活在面试当中,实际工作当中你压根不会用到它,但是你要知道它是什么东西。

最后在这里小编分享一份自己收录整理上述技术体系图相关的几十套腾讯、头条、阿里、美团等公司2021年的面试题,把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分。

还有 高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。

[外链图片转存中…(img-Ux0y0aHv-1712515299823)]

【算法合集】

[外链图片转存中…(img-FkKO0rl4-1712515299824)]

【延伸Android必备知识点】

[外链图片转存中…(img-Mv7xAwPI-1712515299824)]

【Android部分高级架构视频学习资源】

**Android精讲视频领取学习后更加是如虎添翼!**进军BATJ大厂等(备战)!现在都说互联网寒冬,其实无非就是你上错了车,且穿的少(技能),要是你上对车,自身技术能力够强,公司换掉的代价大,怎么可能会被裁掉,都是淘汰末端的业务Curd而已!现如今市场上初级程序员泛滥,这套教程针对Android开发工程师1-6年的人员、正处于瓶颈期,想要年后突破自己涨薪的,进阶Android中高级、架构师对你更是如鱼得水,赶快领取吧!

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值