Android仿微信拍摄短视频

近期做项目需要添加上传短视频功能,功能设置为类似于微信,点击开始拍摄,设置最长拍摄时间,经过研究最终实现了这个功能,下面就和大家分享一下,希望对你有帮助。

1.视频录制自定义控件:

/**
 * 视频播放控件
 */
public class MovieRecorderView extends LinearLayout implements OnErrorListener {
 private SurfaceView mSurfaceView;
 private SurfaceHolder mSurfaceHolder;
 private ProgressBar mProgressBar;
 private MediaRecorder mMediaRecorder;
 private Camera mCamera;
 private Timer mTimer;// 计时器
 private OnRecordFinishListener mOnRecordFinishListener;// 录制完成回调接口
 private int mWidth;// 视频分辨率宽度
 private int mHeight;// 视频分辨率高度
 private boolean isOpenCamera;// 是否一开始就打开摄像头
 private int mRecordMaxTime;// 一次拍摄最长时间
 private int mTimeCount;// 时间计数
 private File mVecordFile = null;// 文件
 public MovieRecorderView(Context context) {
 this(context, null);
 }
 public MovieRecorderView(Context context, AttributeSet attrs) {
 this(context, attrs, 0);
 }
 @SuppressLint("NewApi")
 public MovieRecorderView(Context context, AttributeSet attrs, int defStyle) {
 super(context, attrs, defStyle);
 TypedArray a = context.obtainStyledAttributes(attrs,
 R.styleable.MovieRecorderView, defStyle, 0);
 mWidth = a.getInteger(R.styleable.MovieRecorderView_width, 320);// 默认320
 mHeight = a.getInteger(R.styleable.MovieRecorderView_height, 240);// 默认240
 isOpenCamera = a.getBoolean(
 R.styleable.MovieRecorderView_is_open_camera, true);// 默认打开
 mRecordMaxTime = a.getInteger(
 R.styleable.MovieRecorderView_record_max_time, 10);// 默认为10
 LayoutInflater.from(context)
 .inflate(R.layout.movie_recorder_view, this);
 mSurfaceView = (SurfaceView) findViewById(R.id.surfaceview);
 mProgressBar = (ProgressBar) findViewById(R.id.progressBar);
 mProgressBar.setMax(mRecordMaxTime);// 设置进度条最大量
 mSurfaceHolder = mSurfaceView.getHolder();
 mSurfaceHolder.addCallback(new CustomCallBack());
 mSurfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
 a.recycle();
 }
 /**
 *
 */
 private class CustomCallBack implements Callback {
 @Override
 public void surfaceCreated(SurfaceHolder holder) {
 if (!isOpenCamera)
 return;
 try {
 initCamera();
 } catch (IOException e) {
 e.printStackTrace();
 }
 }
 @Override
 public void surfaceChanged(SurfaceHolder holder, int format, int width,
 int height) {
 }
 @Override
 public void surfaceDestroyed(SurfaceHolder holder) {
 if (!isOpenCamera)
 return;
 freeCameraResource();
 }
 }
 /**
 * 初始化摄像头
 */
 private void initCamera() throws IOException {
 if (mCamera != null) {
 freeCameraResource();
 }
 try {
 mCamera = Camera.open();
 } catch (Exception e) {
 e.printStackTrace();
 freeCameraResource();
 }
 if (mCamera == null)
 return;
 setCameraParams();
 mCamera.setDisplayOrientation(90);
 mCamera.setPreviewDisplay(mSurfaceHolder);
 mCamera.startPreview();
 mCamera.unlock();
 }
 /**
 * 设置摄像头为竖屏
 */
 private void setCameraParams() {
 if (mCamera != null) {
 Parameters params = mCamera.getParameters();
 params.set("orientation", "portrait");
 mCamera.setParameters(params);
 }
 }
 /**
 * 释放摄像头资源
 */
 private void freeCameraResource() {
 if (mCamera != null) {
 mCamera.setPreviewCallback(null);
 mCamera.stopPreview();
 mCamera.lock();
 mCamera.release();
 mCamera = null;
 }
 }
 private void createRecordDir() {
 //录制的视频保存文件夹
 File sampleDir = new File(Environment.getExternalStorageDirectory()
 + File.separator + "ysb/video/");//录制视频的保存地址
 if (!sampleDir.exists()) {
 sampleDir.mkdirs();
 }
 File vecordDir = sampleDir;
 // 创建文件
 try {
 mVecordFile = File.createTempFile("recording", ".mp4", vecordDir);// mp4格式的录制的视频文件
 } catch (IOException e) {
 e.printStackTrace();
 }
 }
 /**
 * 初始化
 * @throws IOException
 */
 @SuppressLint("NewApi")
 private void initRecord() throws IOException {
 mMediaRecorder = new MediaRecorder();
 mMediaRecorder.reset();
 if (mCamera != null)
 mMediaRecorder.setCamera(mCamera);
 mMediaRecorder.setOnErrorListener(this);
 mMediaRecorder.setPreviewDisplay(mSurfaceHolder.getSurface());
 mMediaRecorder.setVideoSource(VideoSource.CAMERA);// 视频源
 mMediaRecorder.setAudioSource(AudioSource.MIC);// 音频源
 mMediaRecorder.setOutputFormat(OutputFormat.MPEG_4);// 视频输出格式
 mMediaRecorder.setAudioEncoder(AudioEncoder.AMR_NB);// 音频格式
 mMediaRecorder.setVideoSize(mWidth, mHeight);// 设置分辨率:
 // mMediaRecorder.setVideoFrameRate(16);// 这个我把它去掉了,感觉没什么用
 mMediaRecorder.setVideoEncodingBitRate(1 * 1024 * 1024 * 100);// 设置帧频率,然后就清晰了
 mMediaRecorder.setOrientationHint(90);// 输出旋转90度,保持竖屏录制
 mMediaRecorder.setVideoEncoder(VideoEncoder.MPEG_4_SP);// 视频录制格式
 // mediaRecorder.setMaxDuration(Constant.MAXVEDIOTIME * 1000);
 mMediaRecorder.setOutputFile(mVecordFile.getAbsolutePath());
 mMediaRecorder.prepare();
 try {
 mMediaRecorder.start();
 } catch (IllegalStateException e) {
 e.printStackTrace();
 } catch (RuntimeException e) {
 e.printStackTrace();
 } catch (Exception e) {
 e.printStackTrace();
 }
 }
 /**
 * 开始录制视频
 * @param fileName
 * 视频储存位置
 * @param onRecordFinishListener
 * 达到指定时间之后回调接口
 */
 public void record(final OnRecordFinishListener onRecordFinishListener) {
 this.mOnRecordFinishListener = onRecordFinishListener;
 createRecordDir();
 try {
 if (!isOpenCamera)// 如果未打开摄像头,则打开
 initCamera();
 initRecord();
 mTimeCount = 0;// 时间计数器重新赋值
 mTimer = new Timer();
 mTimer.schedule(new TimerTask() {
 @Override
 public void run() {
  mTimeCount++;
  mProgressBar.setProgress(mTimeCount);// 设置进度条
  if (mTimeCount == mRecordMaxTime) {// 达到指定时间,停止拍摄
  stop();
  if (mOnRecordFinishListener != null)  mOnRecordFinishListener.onRecordFinish();
  }
 }
 }, 0, 1000);
 } catch (IOException e) {
 e.printStackTrace();
 }
 }
 /**
 * 停止拍摄
 */
 public void stop() {
 stopRecord();
 releaseRecord();
 freeCameraResource();
 }
 /**
 * 停止录制
 */
 public void stopRecord() {
 mProgressBar.setProgress(0);
 if (mTimer != null)
 mTimer.cancel();
 if (mMediaRecorder != null) {
 // 设置后不会崩
 mMediaRecorder.setOnErrorListener(null);
 mMediaRecorder.setPreviewDisplay(null);
 try {
 mMediaRecorder.stop();
 } catch (IllegalStateException e) {
 e.printStackTrace();
 } catch (RuntimeException e) {
 e.printStackTrace();
 } catch (Exception e) {
 e.printStackTrace();
 }
 }
 }
 /**
 * 释放资源
 */
 private void releaseRecord() {
 if (mMediaRecorder != null) {
 mMediaRecorder.setOnErrorListener(null);
 try {
 mMediaRecorder.release();
 } catch (IllegalStateException e) {
 e.printStackTrace();
 } catch (Exception e) {
 e.printStackTrace();
 }
 }
 mMediaRecorder = null;
 }
 public int getTimeCount() {
 return mTimeCount;
 }
 //返回录制的视频文件
 public File getmVecordFile() {
 return mVecordFile;
 }
 /**
 * 录制完成回调接口
 */
 public interface OnRecordFinishListener {
 public void onRecordFinish();
 }
 @Override
 public void onError(MediaRecorder mr, int what, int extra) {
 try {
 if (mr != null)
 mr.reset();
 } catch (IllegalStateException e) {
 e.printStackTrace();
 } catch (Exception e) {
 e.printStackTrace();
 }
 }
}

2.视频录制界面文件movie_recorder_view.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
 xmlns:tools="http://schemas.android.com/tools"
 android:layout_width="match_parent"
 android:layout_height="match_parent"
android:background="@android:color/background_dark"
 android:orientation="vertical">
 <SurfaceView
 android:id="@+id/surfaceview"
 android:layout_width="fill_parent"
 android:layout_height="0dp"
 android:layout_weight="1"
 />
 <ProgressBar
 android:id="@+id/progressBar"
 style="?android:attr/progressBarStyleHorizontal"
 android:layout_width="match_parent"
 android:layout_height="2dp"
 />
</LinearLayout>

做好这些准备工作,下面我们就可以开始设计我们的视频录制功能了。PS:以上代码取至网上,在此向大牛致敬。

3.拍摄主界面,拍摄界面有两部分组成,上面是视频拍摄控件显示,下面是用户点击拍摄按钮,配置文件:activity_main.xml。

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
 xmlns:tools="http://schemas.android.com/tools"
 android:layout_width="match_parent"
 android:layout_height="match_parent"
 android:background="@android:color/white"
 android:orientation="vertical">
<com.example.wechatvideorecorddemo.MovieRecorderView
 android:id="@+id/movieRecorderView"
 android:layout_width="match_parent"
 android:layout_height="0dp"
 android:layout_weight="1"
 android:layout_margin="3dp" />
 <Button
 android:id="@+id/shoot_button"
 android:layout_width="wrap_content"
 android:layout_height="wrap_content"
 android:layout_gravity="center" android:background="@drawable/bg_movie_add_shoot"
 android:text="按住拍"
 android:textColor="#20b6ff"/>
</LinearLayout>

4.有了主界面的视图,下面我们就开始书写我们的Activity文件MainActivity.java:

public class MainActivity extends Activity {
 private MovieRecorderView mRecorderView;//视频录制控件
 private Button mShootBtn;//视频开始录制按钮
 private boolean isFinish = true;
 private boolean success = false;//防止录制完成后出现多次跳转事件
 @Override
 protected void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.activity_main);
 mRecorderView = (MovieRecorderView) findViewById(R.id.movieRecorderView);
 mShootBtn = (Button) findViewById(R.id.shoot_button);
 //用户长按事件监听
 mShootBtn.setOnTouchListener(new OnTouchListener() {
 @Override
 public boolean onTouch(View v, MotionEvent event) {
 if (event.getAction() == MotionEvent.ACTION_DOWN) {//用户按下拍摄按钮
  mShootBtn.setBackgroundResource(R.drawable.bg_movie_add_shoot_select);
  mRecorderView.record(new OnRecordFinishListener() {
  @Override
  public void onRecordFinish() {
  if(!success&&mRecorderView.getTimeCount()<10){//判断用户按下时间是否大于10秒
  success = true;
  handler.sendEmptyMessage(1);
  }
  }
  });
 } else if (event.getAction() == MotionEvent.ACTION_UP) {//用户抬起拍摄按钮
  mShootBtn.setBackgroundResource(R.drawable.bg_movie_add_shoot);
  if (mRecorderView.getTimeCount() > 3){//判断用户按下时间是否大于3秒
  if(!success){
  success = true;
  handler.sendEmptyMessage(1);
  }
  } else {
  success = false;
  if (mRecorderView.getmVecordFile() != null)
 mRecorderView.getmVecordFile().delete();//删除录制的过短视频
  mRecorderView.stop();//停止录制
  Toast.makeText(MainActivity.this, "视频录制时间太短", Toast.LENGTH_SHORT).show();
  }
 }
 return true;
 }
 });
 }
 @Override
 public void onResume() {
 super.onResume();
 isFinish = true;
 if (mRecorderView.getmVecordFile() != null)
 mRecorderView.getmVecordFile().delete();//视频使用后删除
 }
 @Override
 public void onSaveInstanceState(Bundle outState) {
 super.onSaveInstanceState(outState);
 isFinish = false;
 success = false;
 mRecorderView.stop();//停止录制
 }
 @Override
 public void onPause() {
 super.onPause();
 }
 @Override
 public void onDestroy() {
 super.onDestroy();
 }
 private Handler handler = new Handler() {
 @Override
 public void handleMessage(Message msg) {
 if(success){
 finishActivity();
 }
 }
 };
 //视频录制结束后,跳转的函数
 private void finishActivity() {
 if (isFinish) {
 mRecorderView.stop();
 Intent intent = new Intent(this, SuccessActivity.class);
 Bundle bundle = new Bundle();
 bundle.putString("text", mRecorderView.getmVecordFile().toString());
 intent.putExtras(bundle);
 startActivity(intent);
 }
 success = false;
 }
 /**
 * 录制完成回调
 */
 public interface OnShootCompletionListener {
 public void OnShootSuccess(String path, int second);
 public void OnShootFailure();
 }
}

到这里我们仿微信的短视频拍摄就已经大功告成,那么下面我们检验一下,我们录制的效果如何,下面我以Android提供的视频播放控件(VideoView)为大家介绍一下如何播放录制的短视频。

5.播放视频的配置文件activity_success.xml:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
 xmlns:tools="http://schemas.android.com/tools"
 android:layout_width="match_parent"
 android:layout_height="match_parent"
 android:background="@android:color/white"
 android:orientation="vertical">
 <TextView
 android:id="@+id/text"
 android:layout_width="wrap_content"
 android:layout_height="wrap_content"
 android:layout_gravity="center"
 android:text="@string/app_name" />
 <LinearLayout
 android:layout_width="match_parent"
 android:layout_height="wrap_content"
 android:orientation="horizontal"
 >
 <Button
 android:id="@+id/button1"
 android:layout_width="match_parent"
 android:layout_height="wrap_content"
 android:layout_weight="1"
 android:gravity="center"
 android:padding="5dp"
 android:text="播放"
 />
 <Button
 android:id="@+id/button2"
 android:layout_width="match_parent"
 android:layout_height="wrap_content"
 android:layout_weight="1"
 android:gravity="center"
 android:padding="5dp"
 android:text="暂停"
 />
 <Button
 android:id="@+id/button3"
 android:layout_width="match_parent"
 android:layout_height="wrap_content"
 android:layout_weight="1"
 android:gravity="center"
 android:padding="5dp"
 android:text="重播"
 />
 <Button
 android:id="@+id/button4"
 android:layout_width="match_parent"
 android:layout_height="wrap_content"
 android:layout_weight="1"
 android:gravity="center"
 android:padding="5dp"
 android:text="视频长度"
 />
 </LinearLayout>
 <VideoView
 android:id="@+id/videoView1"
 android:layout_width="wrap_content"
 android:layout_height="500dp" />
</LinearLayout>

6.视频播放的控制代码SuccessActivity.java:

public class SuccessActivity extends Activity implements OnClickListener{
 private TextView text;//视频保存的路径
 private Button button1;//播放开关
 private Button button2;//暂停开关
 private Button button3;//重新播放开关
 private Button button4;//视频大小开关
 private VideoView videoView1;//视频播放控件
 private String file;//视频路径
 @Override
 protected void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.activity_success);
 Bundle bundle = getIntent().getExtras();
 file = bundle.getString("text");//获得拍摄的短视频保存地址
 init();
 setValue();
 }
 //初始化
 private void init() {
 text = (TextView) findViewById(R.id.text);
 button1 = (Button) findViewById(R.id.button1);
 button2 = (Button) findViewById(R.id.button2);
 button3 = (Button) findViewById(R.id.button3);
 button4 = (Button) findViewById(R.id.button4);
 videoView1 = (VideoView) findViewById(R.id.videoView1);
 }
 //设置
 private void setValue() {
 text.setText(file);
 button1.setOnClickListener(this);
 button2.setOnClickListener(this);
 button3.setOnClickListener(this);
 button4.setOnClickListener(this);
 videoView1.setVideoPath(file);
 }
 @Override
 public void onClick(View v) {
 switch (v.getId()) {
 case R.id.button1:
 videoView1.start();
 break;
 case R.id.button2:
 videoView1.pause();
 break;
 case R.id.button3:
 videoView1.resume();
 videoView1.start();
 break;
 case R.id.button4:
 Toast.makeText(this, "视频长度:"+(videoView1.getDuration()/1024)+"M", Toast.LENGTH_SHORT).show();
 break;
 default:
 break;
 }
 }
}

7.添加权限:

<!-- 视频录制的权限star -->
<!-- 摄像头 -->
<uses-permission android:name="android.permission.CAMERA" />
<!-- 音频即声音 -->
<uses-permission android:name="android.permission.RECORD_AUDIO" />
<!-- sd卡写入权限 -->
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<!-- 硬件支持 -->
<uses-feature android:name="android.hardware.camera" />
<uses-feature android:name="android.hardware.camera.autofocus" />
<!-- 视频录制的权限end -->

功能界面截图:

好了,到这里关于拍摄短视频的知识就和大家分享完毕,具体的实现很简单,相信大家看到这里已经已经学会了,当然如果你还有什么疑问,可以留言讨论。最后给大家分享一个demo的下载地址,方便大家下载学习,下载地址:http://xiazai.jb51.net/201612/yuanma/WeChatVideoRecordDemo_jb51.rar

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持我们!

(0)

相关推荐

  • Android开发仿扫一扫实现拍摄框内的照片功能

    就是仿照现在扫一扫的形式,周围是半透明的遮挡,然后中间是全透明的,拍摄后只截取框内的内容 查了很多博客,实现起来真的太复杂了,本人比较怕麻烦所以在很多地方偷懒了 先上效果图: 第一步:设置照相机预览以及拍照 这是所有步骤的前提,没有预览,用户怎么知道自己拍的什么呢.预览用的是SurfaceView 这篇博文写得已经十分详细了,打开照相机,然后拍照,而且十分简洁!不想别的博客一下就几百行代码不知所云.这篇代码可以复制下去当相机模版使用. 这里遇到一个问题,就是预览的效果是左转90度的,拍出来也是左

  • Android实现相机拍摄、选择、图片裁剪功能

    最近的一些学习心得: 功能实现:点击圆形头像之后可以实现相册上传或者开启相机,然后把得到的图片经过剪裁,把剪裁过的图片设置为头像的背景图 步骤:第一步:自定义一个类,继承ImageView,重写draw方法,实现外观为圆形 第二步:在xml文件中引用该控件 第三步:实现圆形头像的点击事件,点击后显示对话框界面,询问你是打开相册还是相机(自动省略显示对话框的代码) 第四步:根据用户选择情况,打开相册或者相机 第五步:将拍摄的图片或者相册选中的图片进行剪裁,将结果保存在指定内存区域 第六步:更新头像

  • Android实现返回拍摄的图片功能实例

    本文实例讲述了Android实现返回拍摄的图片功能.分享给大家供大家参考.具体如下: 第一步: try { Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); startActivityForResult(intent, 0); } catch (ActivityNotFoundException e) { // Do nothing for now } 第二步: @Override protected void onActi

  • Android 仿微信图像拍摄和选择界面功能(代码分享)

    插件运行后的画面如下: 下面这张图对图像进行筛选,根据照片产生的源头分(QQ和微信和相机) 点击某文件夹后,可以查看该文件夹下包含的所有的图片 图片选择界面 选中后就跳到已经选择界面的窗口,并且可以对该吃图片上传进行简要的描述 首先我想说明的是这个插件默认是不进行图片筛选的,打开app后会有几十个文件夹,但是个人认为开发中常用的图片基本都来自于QQ中拍摄的照片,微信中拍摄的照片,以及相机直接拍摄的照片,因此我对这个插件进行过滤以及文件夹名称的更改,具体做法,主要是对AlbumHelper类bui

  • Android 调用系统相机拍摄获取照片的两种方法实现实例

    Android 调用系统相机拍摄获取照片的两种方法实现实例 在我们Android开发中经常需要做这个一个功能,调用系统相机拍照,然后获取拍摄的照片.下面是我总结的两种方法获取拍摄之后的照片,一种是通过Bundle来获取压缩过的照片,一种是通过SD卡获取的原图. 下面是演示代码: 布局文件: <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http:

  • 什么是Android静默拍摄 Android静默拍摄app制作方法

    引言: 在做用户的头像时,忽然想到前段时间(可能是很久以前了),支付宝传出偷偷拍摄用户的生活照,真实头像,被喷的很厉害.然而作为Android开发者的我第一反应竟然是握草,他是怎么实现的.在我印象中,iOS对权限的控制是很严格的,偷偷调起摄像头这种行为应该是很困难的.然而Android4.2之前可以说开发者几乎拥有了系统权限,能力之强简直可怕.而现在Android已经到了7.0,虽然大多说用户还是在4.4到6.0的.我想我也来做一个静默拍摄的app. 正文: 所谓静默拍摄就是在用户毫无感知的情况

  • Android仿微信拍摄短视频

    近期做项目需要添加上传短视频功能,功能设置为类似于微信,点击开始拍摄,设置最长拍摄时间,经过研究最终实现了这个功能,下面就和大家分享一下,希望对你有帮助. 1.视频录制自定义控件: /** * 视频播放控件 */ public class MovieRecorderView extends LinearLayout implements OnErrorListener { private SurfaceView mSurfaceView; private SurfaceHolder mSurfa

  • Android仿微信录制小视频

    本文实例为大家分享了Android仿微信录制小视频的具体代码,供大家参考,具体内容如下 先上张图片看看效果 简单叙述下 首先通过Camera类调用系统相机 通过surfaceview绘制出来 通过MediaRecorder来录制视频 闪光灯 和 切换摄像头 需要重新配置Camera的参数 Camera预览界面画面拉升的原因是因为Surfaceview的大小与设定的比例不一致的 **本次版本更新了 切换前置摄像头录制视频问题 Android部分手机录制视频适配IOS手机问题 (OPPO手机部分不适

  • Android仿微信长按录制视频并播放功能

    本文实例为大家分享了Android仿微信长按录制视频并播放功能的具体代码,供大家参考,具体内容如下 一.点击按钮进行录制 首先要获取摄像拍照的权限和读取权限 <uses-permission android:name="android.permission.CAMERA" /> <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> <use

  • 微信小程序仿抖音短视频切换效果的实例代码

    一直以为抖音短视频切换假如用小程序做的话应该是比较简单的,直接用swiper实现就好,但在实际写的过程中才发现没那么简单,要控制的逻辑还是挺多的. 还是先看效果 体验路径 自定义组件系列>>仿抖音短视频切换 代码逻辑 直接调用自定义的swiper组件就好 调用代码 js const videoList = [] Page({ data: { videoList, activeId:2, isPlaying:true }, onLoad() { var that = this wx.getSys

  • Android仿微信键盘切换效果

    Android 仿微信的键盘切换(录音,表情,文字,其他),IM通讯,类似朋友圈只要涉及到文字等相关的app都会要涉及到键盘的处理,今天就给大家分享一下Android 仿微信的键盘切换. 效果图如下: Android 仿微信的键盘切换,实现了录音.表情.其他和软键盘显示之间的切换,其中解决了很多博客介绍的键盘切换时,软键盘显示切换到表情(其他)时,出现屏幕晃动的情况,以及点击和滑动键盘显示区域外时,软键盘隐藏的功能等,废话不多说直接上代码,以供大家参考: xml: <?xml version=&qu

  • Android仿微信语音聊天功能

    本文实例讲述了Android仿微信语音聊天功能代码.分享给大家供大家参考.具体如下: 项目效果如下: 具体代码如下: AudioManager.java package com.xuliugen.weichat; import java.io.File; import java.io.IOException; import java.util.UUID; import android.media.MediaRecorder; public class AudioManager { private

  • Android仿微信语音消息的录制和播放功能

    一.简述 效果: 实现功能: 长按Button时改变Button显示文字,弹出Dialog(动态更新音量),动态生成录音文件,开始录音: 监听手指动作,规定区域.录音状态下手指划出规定区域取消录音,删除生成的录音文件: 监听手指动作.当手指抬起时,判断是否开始录音,录音时长是否过短,符合条件则提示录音时长过短:正常结束时通过回调返回该次录音的文件路径和时长. 4.点击录音列表的item时,播放动画,播放对应的音频文件. 主要用到4个核心类: 自定义录音按钮(AudioRecordButton):

  • Android仿微信语音对讲录音功能

    自微信出现以来取得了很好的成绩,语音对讲的实现更加方便了人与人之间的交流.今天来实践一下微信的语音对讲的录音实现,这个也比较容易实现.在此,我将该按钮封装成为一个控件,并通过策略模式的方式实现录音和界面的解耦合,以方便我们在实际情况中对录音方法的不同需求(例如想要实现wav格式的编码时我们也就不能再使用MediaRecorder,而只能使用AudioRecord进行处理). 效果图: 实现思路: 1.在微信中我们可以看到实现语音对讲的是通过点按按钮来完成的,因此在这里我选择重新自己的控件使其继承

  • Android仿微信底部菜单栏效果

    前言 在市面上,大多数的APP都需要通过底部菜单栏来将程序的功能进行分类整理,通常都是分为3-5个大模块,从而正确有效地引导用户去使用我们的APP.实现底部菜单栏的方法也有很多种. 1.仿微信底部菜单栏(ViewPager+ImagerView+TextView) ......(其他方式后续会补充) 效果预览 首先来个开胃菜,看看实现效果: 先贴出项目所需的资源文件,这些可随个人自由更改颜色和文字 colors.xml <color name="bg_line_light_gray&quo

随机推荐