解决Android SurfaceView绘制触摸轨迹闪烁问题的方法

 更新时间:2020年08月20日 14:35:11   投稿:lijiao  
这篇文章主要为大家详细介绍了解决Android SurfaceView绘制触摸轨迹闪烁问题的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文分享了解决SurfaceView触摸轨迹闪烁问题的方法,供大家参考,具体内容如下

第一种解决SurfaceView触摸轨迹闪烁问题的方法:

由于SurfaceView使用双缓存机制,两张画布轮流显示到屏幕上。那么,要存储触摸轨迹并避免两张画布内容不一致造成的闪烁问题,完全可以利用保存绘制过程并不断重新绘制的方法解决闪烁,而且这样还顺带解决了多次试验中偶尔出现的因为moveTo()函数不能读取到参数执行默认设置(参数设为上次的触摸点)而出现的断线连接闪烁问题,详细代码如下:

package com.tobacco.touchdraw;
import java.util.ArrayList;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Path;
import android.view.MotionEvent;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.View;
import android.view.SurfaceHolder.Callback;
import android.view.View.OnTouchListener;
public class LSurfaceView extends SurfaceView implements Callback,OnTouchListener,Runnable{
 private SurfaceHolder sfh;
 private Canvas canvas;
 private Paint paint;
 private Path path;
 private ArrayList<Path> paths;
 private boolean flag;
 public LSurfaceView(Context context) {
 super(context);
 sfh=this.getHolder();
 sfh.addCallback(this);
 paint=new Paint();
 paint.setColor(Color.RED);
 paint.setAntiAlias(true);
 paint.setStrokeWidth(4);
 paint.setStyle(Paint.Style.STROKE);
 paint.setStrokeCap(Paint.Cap.ROUND);
 paths=new ArrayList<Path>();
 path=new Path();
 }
 public void myDraw(MotionEvent e){
 int action=e.getAction();
 switch(action){
 case MotionEvent.ACTION_DOWN:
 path.moveTo(e.getX(),e.getY());
 break;
 case MotionEvent.ACTION_MOVE:
 path.lineTo(e.getX(),e.getY());
 break;
 case MotionEvent.ACTION_UP:
 //path.close();
 Path path1=new Path(path);
 paths.add(path1);
 path.reset();
 break;
 }
 
 
 }
 @Override
 public void surfaceChanged(SurfaceHolder arg0, int arg1, int arg2, int arg3) {
 
 
 }
 @Override
 public void surfaceCreated(SurfaceHolder holder) {
 flag=true;
 setOnTouchListener(this);
 new Thread(this).start();
 
 }
 @Override
 public void surfaceDestroyed(SurfaceHolder holder) {
 flag=false;
 }
 @Override
 public boolean onTouch(View v, MotionEvent event) {
 myDraw(event);
 return true;
 }
 @Override
 public void run() {
 while(flag){
 long start=System.currentTimeMillis();
 canvas=sfh.lockCanvas();
 if(canvas!=null){
 canvas.drawColor(Color.BLACK);
 for(int i=0;i<paths.size();i++)
  canvas.drawPath(paths.get(i),paint);
 canvas.drawPath(path,paint);
 sfh.unlockCanvasAndPost(canvas);
 }
 long end=System.currentTimeMillis();
 try{
 if(end-start<30){
  Thread.sleep(30-(end-start));
 }
 }
 catch(Exception e){
 }
 }
 
 }
}

这里还要注意的是:ArrayList保存的是对象的引用,所以要在每次添加时都新建一个对象实体。

第二种解决SurfaceView触摸轨迹闪烁问题的方法:

处理触屏轨迹的绘制时,用到了SurfaceView,建立Path对象,在点击时开始设置Path对象,滑动过程中记录触摸点,离开后重新设置Path对象,因不能阻塞主线程,所以新建了一个子线程来不断刷新屏幕,也就是将path不断绘制。但是,接着就出现了一个问题:屏幕中每条轨迹线的终点都会有一小段直线段不断闪烁。猜测可能是lockCanvas()获取的对象区域不一样,就试着使用了lockCanvas(Rect re),但是,运行后发现还是没有解决问题;接着想到可能是因为每次lockCanvas()后获取的对象不同,就在主线程中添加了一个Canvas对象,每次都在Canvas对象中修改画面,然后提交显示,但是,程序运行后效果丝毫没有改变!程序代码如下:

package com.tobacco.touchdraw;
import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Path;
import android.view.MotionEvent;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.View;
import android.view.SurfaceHolder.Callback;
import android.view.View.OnTouchListener;
public class MySurfaceView extends SurfaceView implements Callback,OnTouchListener,Runnable{
 private SurfaceHolder sfh;
 private Canvas canvas;
 private Paint paint;
 private float lastX,lastY;
 private Path path;
 private boolean flag;
 public MySurfaceView(Context context) {
 super(context);
 sfh=this.getHolder();
 sfh.addCallback(this);
 paint=new Paint();
 paint.setColor(Color.RED);
 paint.setAntiAlias(true);
 paint.setStrokeWidth(5);
 paint.setStyle(Paint.Style.STROKE);
 paint.setStrokeCap(Paint.Cap.ROUND);
 path=new Path();
 
 }
 public void myDraw(MotionEvent e){
 int action=e.getAction();
 switch(action){
 case MotionEvent.ACTION_DOWN:
 path.moveTo(e.getX(),e.getY());
 lastX=e.getX();
 lastY=e.getY();
 break;
 case MotionEvent.ACTION_MOVE:
 path.quadTo(lastX,lastY,e.getX(),e.getY());
 lastX=e.getX();
 lastY=e.getY();
 break;
 case MotionEvent.ACTION_UP:
 //path.quadTo(lastX,lastY,e.getX(),e.getY());
 path.reset();
 break;
 }
 
 
 }
 @Override
 public void surfaceChanged(SurfaceHolder arg0, int arg1, int arg2, int arg3) {
 
 
 }
 @Override
 public void surfaceCreated(SurfaceHolder holder) {
 flag=true;
 setOnTouchListener(this);
 new Thread(this).start();
 
 }
 @Override
 public void surfaceDestroyed(SurfaceHolder holder) {
 flag=false;
 }
 @Override
 public boolean onTouch(View v, MotionEvent event) {
 myDraw(event);
 return true;
 }
 @Override
 public void run() {
 while(flag){
 long start=System.currentTimeMillis();
 canvas=sfh.lockCanvas();
 if(canvas!=null){
 canvas.drawPath(path,paint);
 sfh.unlockCanvasAndPost(canvas);
 }
 long end=System.currentTimeMillis();
 try{
 if(end-start<100){
  Thread.sleep(100-(end-start));
 }
 }
 catch(Exception e){
 }
 }
 
 }
}

以上就是本文的全部内容,希望能够帮助大家轻松解决SurfaceView触摸轨迹闪烁问题。

相关文章

  • RxJava+Retrofit+OkHttp实现文件上传

    RxJava+Retrofit+OkHttp实现文件上传

    本篇文章主要介绍了RxJava+Retrofit+OkHttp实现文件上传,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-11-11
  • android实现动态显隐进度条

    android实现动态显隐进度条

    这篇文章主要为大家详细介绍了android实现动态显隐进度条,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-07-07
  • 更新android SDK 失败的解决方法

    更新android SDK 失败的解决方法

    更新android SDK 失败的解决方法,需要的朋友可以参考一下
    2013-03-03
  • Android Studio3安装图文教程

    Android Studio3安装图文教程

    这篇文章主要为大家详细介绍了Android Studio3安装图文教程,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-07-07
  • Cocos2d-x 3.0多线程异步加载资源实例

    Cocos2d-x 3.0多线程异步加载资源实例

    手机游戏,即便是休闲类的小游戏,往往也涉及大量纹理资源、音视频资源、文件读写以及网络通信,处理的稍有不甚就会出现画面卡顿,交互不畅的情况
    2014-04-04
  • Android使用GridView实现表格分割线效果

    Android使用GridView实现表格分割线效果

    这篇文章主要为大家详细介绍了Android使用GridView实现表格分割线效果,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-07-07
  • Android自定义TextView实现文字倾斜效果

    Android自定义TextView实现文字倾斜效果

    有时候Android自带的控件无法满足我们的某些要求,这时就需要我们自定义控件来实现这些功能。比如在实际开发应用中,我们有时需要将TextView的文字倾斜一定的角度,就需要自定义TextView。下面这篇文章就给大家介绍了利用Android TextView如何实现文字倾斜效果。
    2016-11-11
  • Android 实时监测(监听)网络连接状态变化

    Android 实时监测(监听)网络连接状态变化

    这篇文章主要介绍了Android 实时监测(监听)网络连接状态变化的相关知识,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2017-06-06
  • Android动画系列之帧动画和补间动画的示例代码

    Android动画系列之帧动画和补间动画的示例代码

    Android 提供三种动画:帧动画、补间动画和属性动画,本篇文章介绍帧动画以及补间动画的使用,属性动画的使用将在后面的文章中分享,那就来复习一下这两种动画的使用吧
    2020-09-09
  • 更新至Android Studio4.1后发现as打不开的解决方法(原因分析)

    更新至Android Studio4.1后发现as打不开的解决方法(原因分析)

    这篇文章主要介绍了更新至Android Studio4.1后发现as打不开的解决方案,本文给大家分享问题所在原因给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-10-10

最新评论