前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Android触摸事件实现笔触画布

Android触摸事件实现笔触画布

作者头像
张风捷特烈
发布2018-12-06 15:30:22
9070
发布2018-12-06 15:30:22
举报

任何View都有触摸事件,经常在自定义控件时重写setOnTouchListener 本篇通过手绘图片来讲述这个知识点

本篇分为三个等级:一览图:

直线

曲线

笔触

LEVEL1.png

LEVEL2.png

LEVER3.png


LEVEL1:基础实现

在Activity中通过一个全屏的Bitmap创建的Canvas绘制 为ImageView添加触摸事件监听。

1.成员变量
代码语言:javascript
复制
ImageView mIdIvShow;
float downX = 0;
float downY = 0;
float upX = 0;
float upY = 0;
private Canvas mCanvas;
private Paint mPaint;
2.创建画布
代码语言:javascript
复制
//获取屏幕尺寸
Point point = new Point();
getWindowManager().getDefaultDisplay().getSize(point);

//创建一个和屏幕一样大的Bitmap
Bitmap bitmap = Bitmap.createBitmap(point.x, point.y, Bitmap.Config.ARGB_8888);
//创建Canvas对象
mCanvas = new Canvas(bitmap);
mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mPaint.setStrokeWidth(10);
mPaint.setColor(Color.RED);
//将bitmap用ImageView展示
mIdIvShow.setImageBitmap(bitmap);
3.监听事件
代码语言:javascript
复制
 mIdIvShow.setOnTouchListener((v, event) -> {
            switch (event.getAction()) {
                case MotionEvent.ACTION_DOWN:
                    downX = event.getX();
                    downY = event.getY();
                    L.d("按下:(" + downX + "," + downY + ")" + L.l());
                    break;
                case MotionEvent.ACTION_CANCEL:
                    break;
                case MotionEvent.ACTION_MOVE:
                    break;
                case MotionEvent.ACTION_UP:
                    upX = event.getX();
                    upY = event.getY();
                    L.d("抬起:(" + upX + "," + upY + ")" + L.l());
                    mCanvas.drawLine(downX, downY, upX, upY, mPaint);
                    mIdIvShow.invalidate();//更新视图
                    break;
            }
            return true;
        });
    }

升级版:LEVER2

LEVEL2.png

代码语言:javascript
复制
mIdIvShow.setOnTouchListener((v, event) -> {
    switch (event.getAction()) {
        case MotionEvent.ACTION_DOWN:
            downX = event.getX();
            downY = event.getY();
            break;
        case MotionEvent.ACTION_CANCEL:
            break;
        case MotionEvent.ACTION_MOVE:
            upX = event.getX();
            upY = event.getY();
            mCanvas.drawLine(downX, downY, upX, upY, mPaint);
            mIdIvShow.invalidate();
            //更新点位
            downY = upY;
            downX = upX;
            break;
        case MotionEvent.ACTION_UP:
            //抬起点Y>1100,清除笔迹
            if (upY > 1100) {
                Paint paint = new Paint();
                paint.setColor(Color.WHITE);
                mCanvas.drawRect(0, 0, mPoint.x, mPoint.y, paint);
            }
            break;
    }
    return true;
});

再升级版:LEVER3

笔触根据绘制的速度动态改变画笔粗细

LEVER3.png

代码语言:javascript
复制
float movingX = 0;
float movingY = 0;
private long lastTimestamp = 0L;//最后一次的时间戳
代码语言:javascript
复制
mIdIvShow.setOnTouchListener((view, event) -> {
    switch (event.getAction()) {
        case MotionEvent.ACTION_DOWN:
            lastTimestamp = System.currentTimeMillis();
            downX = event.getX();
            downY = event.getY();
            break;
        case MotionEvent.ACTION_CANCEL:
            break;
        case MotionEvent.ACTION_MOVE:
            movingX = event.getX();
            movingY = event.getY();
            long curTimestamp = System.currentTimeMillis();
            //计算时间差
            long detaT = curTimestamp - lastTimestamp;
            //计算距离差
            float detaS = Logic.disPos2d(movingX, movingY, downX, downY);
            //由于速度是 px/ms
            double v = detaS / detaT;
            //速度转化为画笔宽度的等式
            float width = 14/(float)v;
            L.d(width + L.l());
            //限制极值情况
            if ((width > 0) && width < 30) {
                mPaint.setStrokeWidth(width);
            }
            mCanvas.drawLine(downX, downY, movingX, movingY, mPaint);
            mIdIvShow.invalidate();
            downX = movingX;
            downY = movingY;
            lastTimestamp = curTimestamp;//更新时间
            movePos.add(new PointF(event.getX(), event.getY()));
            break;
    }
    return true;
});
拓展
1.其中可以改变求宽度的等式实现不同的笔触:如
代码语言:javascript
复制
float width = (float) Math.log10(v) * 40;

LEVER3+.png

2.在图片上绘画
代码语言:javascript
复制
//图片原型
Bitmap bitmap = BitmapFactory.decodeResource(getResources(), R.mipmap.iv_500x400);
//图片副本
Bitmap mNewBitmap = Bitmap.createBitmap(bitmap.getWidth(), bitmap.getHeight(), bitmap.getConfig());
//用副本生成Canvas
mCanvas = new Canvas(mNewBitmap);
mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mPaint.setStrokeCap(Paint.Cap.ROUND);//直线圆头
mCanvas.drawBitmap(bitmap, new Matrix(), mPaint);
mPaint.setStrokeWidth(10);
mPaint.setColor(Color.parseColor("#88164BE6"));
//设置副本图片到ImageView
mIdIvShow.setImageBitmap(mNewBitmap);
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2018.10.30 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 本篇分为三个等级:一览图:
  • LEVEL1:基础实现
    • 1.成员变量
      • 2.创建画布
        • 3.监听事件
        • 升级版:LEVER2
        • 再升级版:LEVER3
        • 拓展
          • 1.其中可以改变求宽度的等式实现不同的笔触:如
            • 2.在图片上绘画
            领券
            问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档