解決Android SurfaceView繪制觸摸軌跡閃爍問題的方法
本文分享了解決SurfaceView觸摸軌跡閃爍問題的方法,供大家參考,具體內(nèi)容如下
第一種解決SurfaceView觸摸軌跡閃爍問題的方法:
由于SurfaceView使用雙緩存機制,兩張畫布輪流顯示到屏幕上。那么,要存儲觸摸軌跡并避免兩張畫布內(nèi)容不一致造成的閃爍問題,完全可以利用保存繪制過程并不斷重新繪制的方法解決閃爍,而且這樣還順帶解決了多次試驗中偶爾出現(xiàn)的因為moveTo()函數(shù)不能讀取到參數(shù)執(zhí)行默認設(shè)置(參數(shù)設(shè)為上次的觸摸點)而出現(xiàn)的斷線連接閃爍問題,詳細代碼如下:
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對象,在點擊時開始設(shè)置Path對象,滑動過程中記錄觸摸點,離開后重新設(shè)置Path對象,因不能阻塞主線程,所以新建了一個子線程來不斷刷新屏幕,也就是將path不斷繪制。但是,接著就出現(xiàn)了一個問題:屏幕中每條軌跡線的終點都會有一小段直線段不斷閃爍。猜測可能是lockCanvas()獲取的對象區(qū)域不一樣,就試著使用了lockCanvas(Rect re),但是,運行后發(fā)現(xiàn)還是沒有解決問題;接著想到可能是因為每次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){
}
}
}
}
以上就是本文的全部內(nèi)容,希望能夠幫助大家輕松解決SurfaceView觸摸軌跡閃爍問題。
相關(guān)文章
RxJava+Retrofit+OkHttp實現(xiàn)文件上傳
本篇文章主要介紹了RxJava+Retrofit+OkHttp實現(xiàn)文件上傳,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-11-11
Android使用GridView實現(xiàn)表格分割線效果
這篇文章主要為大家詳細介紹了Android使用GridView實現(xiàn)表格分割線效果,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2020-07-07
Android自定義TextView實現(xiàn)文字傾斜效果
有時候Android自帶的控件無法滿足我們的某些要求,這時就需要我們自定義控件來實現(xiàn)這些功能。比如在實際開發(fā)應(yīng)用中,我們有時需要將TextView的文字傾斜一定的角度,就需要自定義TextView。下面這篇文章就給大家介紹了利用Android TextView如何實現(xiàn)文字傾斜效果。2016-11-11
Android 實時監(jiān)測(監(jiān)聽)網(wǎng)絡(luò)連接狀態(tài)變化
這篇文章主要介紹了Android 實時監(jiān)測(監(jiān)聽)網(wǎng)絡(luò)連接狀態(tài)變化的相關(guān)知識,非常不錯,具有參考借鑒價值,需要的朋友可以參考下2017-06-06
更新至Android Studio4.1后發(fā)現(xiàn)as打不開的解決方法(原因分析)
這篇文章主要介紹了更新至Android Studio4.1后發(fā)現(xiàn)as打不開的解決方案,本文給大家分享問題所在原因給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-10-10

