android实现录屏小功能
作者:poorSir 发布时间:2022-08-29 17:08:10
本文实例为大家分享了android实现录屏小功能的具体代码,供大家参考,具体内容如下
思路
android实现录屏功能有两种方案,一种是直接使用android自带的MediaProjectionManager实现录屏功能,第二种是是只录语音,用户的操作通过某种方式进行记录保存,最后通过某种协议进行播放。
两种方案各有各的优缺点,前者实现方式简单,但无法只录制特定区域的画面,并且生成的视频文件一般都比较大。后者实现较为繁琐,音频录制android7.0之前没有暂停方法,只能生成多个文件,然后对音频进行合成。用户的操作需要自己进行保存,播放时还原。播放器需要自定义生成。但后者的好处是可扩展性高,支持特定区域录制,并且生成的音频文件比较小。
需求
录制画板,画板要求可以更改颜色粗细,可以擦除。画板底部可以是白板,图片。图片要求是相机拍摄或者本地图片。可以播放录制内容;需要上传,所以文件要小,所有只能选择第二种方式。
github地址
整个项目生成的是一个文件夹,文件夹中包含一个MP3文件,一个cw协议文件(存储用户的操作),图片。整个画板是一个recyclerView,item中包含一个涂鸦画板,图片控件。播放时读取cw协议文件,按照时间一个个绘制,协议内容包含画板各个页的内容是空白画板还是图片,时间点,操作(切换图片/画线)。
音频
//开始录音
if (mMediaRecorder == null) {
mMediaRecorder = new MediaRecorder();
}
mMediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
mMediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.AMR_NB);
mMediaRecorder.setOutputFile(mRecordFilePath);
mMediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);//amr_nb格式头部有6个字节的头信息
try {
mMediaRecorder.prepare();
mMediaRecorder.start();
isRunning = true;
AudioUtil.startAudio();
mHandler.sendEmptyMessageDelayed(MSG_TYPE_COUNT_DOWN, 1000);
} catch (IOException e) {
e.printStackTrace();
}
/**
* 合成amr_nb编码的音频
* @param partsPaths
* @param unitedFilePath
*/
public static void uniteAMRFile(List<String> partsPaths, String unitedFilePath) {
try {
File unitedFile = new File(unitedFilePath);
FileOutputStream fos = new FileOutputStream(unitedFile);
RandomAccessFile ra = null;
for (int i = 0; i < partsPaths.size(); i++) {
ra = new RandomAccessFile(partsPaths.get(i), "rw");
if (i != 0) {
ra.seek(6);
}
byte[] buffer = new byte[1024 * 8];
int len = 0;
while ((len = ra.read(buffer)) != -1) {
fos.write(buffer,0,len);
}
File file = new File(partsPaths.get(i));
if(file.exists()){
file.delete();
}
}
if(ra!=null){
ra.close();
}
fos.close();
} catch (Exception e) {
e.printStackTrace();
}
}
音频播放
mediaPlayer = new MediaPlayer();
mediaPlayer.setDataSource(path);
mediaPlayer.prepare();
mediaPlayer.start();
recyclerView
是否禁止滑动
public class ForbitLayoutManager extends LinearLayoutManager {
private boolean canScrollHorizon = true;
private boolean canScrollVertical = true;
public ForbitLayoutManager(Context context) {
super(context);
}
public ForbitLayoutManager(Context context, int orientation, boolean reverseLayout) {
super(context, orientation, reverseLayout);
}
public ForbitLayoutManager(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
super(context, attrs, defStyleAttr, defStyleRes);
}
public void setCanScrollHorizon(boolean canScrollHorizon) {
this.canScrollHorizon = canScrollHorizon;
}
public void setCanScrollVertical(boolean canScrollVertical) {
this.canScrollVertical = canScrollVertical;
}
@Override
public boolean canScrollHorizontally() {
return canScrollHorizon && super.canScrollHorizontally();
}
@Override
public boolean canScrollVertically() {
return canScrollVertical && super.canScrollVertically();
}
}
滑动时只滑动一页类似viewPage
mPagerSnapHelper = new PagerSnapHelper();
mPagerSnapHelper.attachToRecyclerView(recyclerView);
获得当前是第几页,类似viewPage的pageSelect
public class RecyclerViewPageChangeListenerHelper extends RecyclerView.OnScrollListener {
private SnapHelper snapHelper;
private OnPageChangeListener onPageChangeListener;
private int oldPosition = -1;//防止同一Position多次触发
public RecyclerViewPageChangeListenerHelper(SnapHelper snapHelper, OnPageChangeListener onPageChangeListener) {
this.snapHelper = snapHelper;
this.onPageChangeListener = onPageChangeListener;
}
@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
super.onScrolled(recyclerView, dx, dy);
if (onPageChangeListener != null) {
onPageChangeListener.onScrolled(recyclerView, dx, dy);
}
}
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
super.onScrollStateChanged(recyclerView, newState);
int position = 0;
RecyclerView.LayoutManager layoutManager = recyclerView.getLayoutManager();
//获取当前选中的itemView
View view = snapHelper.findSnapView(layoutManager);
if (view != null) {
//获取itemView的position
position = layoutManager.getPosition(view);
}
if (onPageChangeListener != null) {
onPageChangeListener.onScrollStateChanged(recyclerView, newState);
//newState == RecyclerView.SCROLL_STATE_IDLE 当滚动停止时触发防止在滚动过程中不停触发
if (newState == RecyclerView.SCROLL_STATE_IDLE && oldPosition != position) {
oldPosition = position;
onPageChangeListener.onPageSelected(position);
}
}
}
public interface OnPageChangeListener {
void onScrollStateChanged(RecyclerView recyclerView, int newState);
void onScrolled(RecyclerView recyclerView, int dx, int dy);
void onPageSelected(int position);
}
}
获得当前选择的item(只能获得可视页面item)
View view = forbitLayoutManager.findViewByPosition(position);
//有时会获取到null,是因为页面还没有渲染完成,可以使用
recyclerView.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver
.OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
//会多次调用,执行完逻辑之后取消监听
recyclerView.getViewTreeObserver().removeOnGlobalLayoutListener(this);
}
});
根据时间进行播放
private void convertCWACT(CW cw, int seconds,boolean isSeek) {
List<CWACT> cwacts = cw.getACT();
//如何是播放器跳转,先回到首页,清空所有item中的画板,防止从高时间跳转到低时间出现错误
if(isSeek){
position =0;
forbitLayoutManager.scrollToPosition(position);
forbitLayoutManager.setStackFromEnd(true);
for(int i=0;i<recyclerViewList.size();i++){
View view = recyclerViewList.get(i);
if(view!=null){
SimpleDoodleView doodleView = view.findViewById(R.id.doodleView);
doodleView.clear();
}
}
}
for (CWACT cwact : cwacts) {
int time = cwact.getTime();
if(isSeek?time > seconds:time != seconds){
continue;
}
if ("switch".equals(cwact.getAction())) {//切换页面
position = cwact.getCwSwitch().getIndex();
forbitLayoutManager.scrollToPosition(position);
forbitLayoutManager.setStackFromEnd(true);
} else if ("line".equals(cwact.getAction())) {//划线
if(position>recyclerViewList.size()-1){
continue;
}
View view = recyclerViewList.get(position);
if(view!=null){
SimpleDoodleView doodleView = view.findViewById(R.id.doodleView);
doodleView.setDrawPath(cwact.getLine());
}
} else if ("clear".equals(cwact.getAction())) {//清屏
if(position>recyclerViewList.size()-1){
continue;
}
View view = recyclerViewList.get(position);
if(view!=null){
SimpleDoodleView doodleView = view.findViewById(R.id.doodleView);
doodleView.clear();
}
}
}
}
来源:https://blog.csdn.net/qq_31433525/article/details/101533832


猜你喜欢
- 一、概述在日常开发中,获取一批数据后,可能需要跟据一定规则对这批数据进行排序操作。在JAVA中,动态数组ArrayList经常被用来存储数据
- 1、什么是ThreadLocal变量ThreadLoal 变量,线程局部变量,同一个 ThreadLocal 所包含的对象,在不同的 Thr
- Spring Cache抽象-使用SpEL表达式概述在Spring Cache注解属性中(比如key,condition和unless),S
- 一、介绍pair是将2个数据组合成一组数据,当需要这样的需求时就可以使用pair。当然你也可以自定义一个结构体struct。不过大家都是为了
- 本篇文章尝试从What、Why、How这三个角度来探索Java中的弱引用,帮助大家理解Java中弱引用的定义、基本使用场景和使用方法。由于个
- 在主Activity中:listview=(ListView)findViewById(R.id.listview);getData();/
- 这篇文章主要介绍了Java内存缓存工具Guava LoadingCache使用解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有
- 本文实例为大家分享了Java实现多线程在线聊天的具体代码,供大家参考,具体内容如下上一篇博客通过UDP实现了聊天,但只能单方面发送消息,这次
- 目录请求参数校验:对于url中的参数对于request body请求体中的参数响应数据格式:全局异常处理统一定义异常响应码,便于查看使用sp
- CollapsingToolbarLayout作用是提供了一个可以折叠的Toolbar,它继承至FrameLayout,给它设置layout
- 这篇文章主要介绍了Java String的intern用法解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,
- SSH(Secure Shell)是一套协议标准,可以用来实现两台机器之间的安全登录以及安全的数据传送,其保证数据安全的原理是非对称加密。传
- 图像的旋转需要调用 Graphics2D 类的rota
- 本文实例讲述了Java之JFrame输出Helloworld的方法。分享给大家供大家参考。具体如下:JAVA的GUI程序的基本思路是以JFr
- 说到Java的本地存储,肯定使用IO流进行操作。首先,我们需要一个创建文件的函数createNewFile:public static bo
- 我就废话不多说,大家还是直接看代码吧~/*** * 英文 */String abc1 = "百度科技(123)公司1";
- 这一篇,给大家介绍一下ImageView控件的使用,ImageView主要是用来显示图片,可以对图片进行放大、缩小、旋转的功能。androi
- 本文实例讲述了C#使用smtp发送带附件的邮件实现方法。可直接将string类型结果保存为附件。分享给大家供大家参考。具体分析如下:该方式直
- 问题学习完maven后,用maven创建了一个web项目,然后在这个web项目中创建了一个java文件夹并标记这个目录为源码根目录,当我准备
- 什么是volatile关键字volatile是Java中用于修饰变量的关键字,其可以保证该变量的可见性以及顺序性,但是无法保证原子性。更准确