public class BezierGestureTrackView extends View {

private Bitmap mBufferBitmap;
private Canvas mBufferCanvas;

public BezierGestureTrackView(Context context) {
super(context);
}
PointF pointF;

Paint paint;
Paint paint2;

Path p,path ;

@ColorInt public static final int YELLOW = 0xFFFFFF00;
@ColorInt
public static final int CYAN = 0xFF00FFFF;
public static final int c1 = 0xEF8A97;
public static final int c2 = 0xFBD2A8;

public BezierGestureTrackView(Context context, AttributeSet attrs) {
super(context, attrs);
paint = new Paint();
paint.setStyle(Paint.Style.STROKE);//设置画笔的填充模式
paint.setStrokeJoin(Paint.Join.ROUND);//设置画笔图形接触时笔迹的形状
paint.setStrokeCap(Paint.Cap.ROUND);//设置画笔离开画板时笔迹的形状
paint.setStrokeWidth(8);//设置画笔粗细 6
paint.setAntiAlias(true);//设置抗锯齿
paint.setColor(Color.WHITE);
initPaint();
// BitmapFactory.decodeResource(getResources(), R.drawable.hbbg);
}

//新建两个SparseArray<Path>用来保存我们画出的Path对象,具体为什么之后会解释
//mActivePointers用来保存当前正在画的Path
SparseArray<Path> mActivePointers = new SparseArray<>();
//truePointers用来保存所有的Path,在onDraw()方法里面我们也是绘制这里的所有Path
SparseArray<Path> truePointers = new SparseArray<>();

//保存单个点
private int i=0;
private float[] a= new float[1000];

//重写onTouchEvent()方法
@Override
public boolean onTouchEvent(MotionEvent event) {

//获取当前DOWN或者UP的是手指的index
int curPointerIndex = event.getActionIndex();
//通过index获得当前手指的id
int curPointerId = event.getPointerId(curPointerIndex);
//获取当前正在发生的事件
int actionMasked = event.getActionMasked();

switch (actionMasked) {
//不管是第一个还是第N个手指落下,都执行以下方法:
case MotionEvent.ACTION_DOWN:

case MotionEvent.ACTION_POINTER_DOWN:
//新建一个PointF对象用来保存点的坐标
pointF = new PointF();
pointF.x = event.getX(curPointerIndex);
pointF.y = event.getY(curPointerIndex);
//每当有手指下落时,都新建一个Path,并移动到手指落下的点
Path p = new Path();
p.moveTo(pointF.x, pointF.y);
//将新建的path添加到mActivePointers中
mActivePointers.append(curPointerId, p);

//保存单个点位置

if (i<998){
a[i++]=event.getX(curPointerIndex);
a[i++]=event.getY(curPointerIndex);
}

break;

case MotionEvent.ACTION_MOVE:
//当有手指移动时,遍历mActivePointers,注意这里遍历的条件并不是mActivePointers.size(),而是event.getPointerCount()
//也就是只监控当前所有正在滑动的手指数,另外在MotionEvent.ACTION_UP事件中,抬起的手指已经从mActivePointers中删除了
for (int size = event.getPointerCount(), i = 0; i < size; i++) {
//获取mActivePointers中的每一个Path
path = mActivePointers.get(event.getPointerId(i));
if (path != null) {
//更新每一个Path将得到的点连线
path.lineTo(event.getX(i), event.getY(i));

}
}

break;
case MotionEvent.ACTION_UP:

case MotionEvent.ACTION_POINTER_UP:
//当手指抬起时,代表已经绘制完毕,取得抬起的手指绘制图线实例,传入truePointers让Drow()绘制
path = mActivePointers.get(curPointerId);
truePointers.append(truePointers.size(), path);
//从mActivePointers中将抬起的手指删除
mActivePointers.remove(curPointerId);
break;

}
//重绘
invalidate();
return true;
}

private void initBuffer(){
mBufferBitmap = Bitmap.createBitmap(getWidth(), getHeight(), Bitmap.Config.ARGB_8888);
mBufferCanvas = new Canvas(mBufferBitmap);
}

@Override
protected void onDraw(Canvas canvas) {
//canvas.drawColor(Color.BLACK);

// Bitmap srcBitmap= BitmapFactory.decodeResource(getResources(), R.drawable.hbbg);
// canvas.drawBitmap(srcBitmap,0f,0f,null);

//遍历mActivePointers,因为只有当手指抬起时Path才被加入truePointers,此时truePointers里面还没有保存当前正在绘制的图线
//如果没有遍历mActivePointers,则图线不会即时显示,只有在抬起手指的瞬间,图线才会一起显示
for (int size = mActivePointers.size(), i = 0; i < size; i++) {
Path path = mActivePointers.valueAt(i);
canvas.drawPath(path, paint);
}
//遍历truePointers,这里保存的是当前正在画的图线
for (int size = truePointers.size(), i = 0; i < size; i++) {
path = truePointers.valueAt(i);
//绘制truePointers里的所有Path
canvas.drawPath(path, paint);
}

if (pointF!=null){
//canvas.drawPoint(pointF.x, pointF.y,paint2);
canvas.drawPoints(a, paint2);

for (int j = 0; j < a.length; j++) {
Log.e("TAG","a:"+a[i]+"a的长度:"+a.length);
}
}

}
// public void clean(){
//// Canvas canvas = sh.lockCanvas();
////
////
//// if(canvas!=null){
//// Paint paint = new Paint();
//// paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR));
//// canvas.drawPaint(paint);
//////canvas.drawColor(Color.BLUE);
//// }
// Log.e("BezierGestureTrackView","CLEAN");
// }

public void clear() {

if (mBufferBitmap != null) {
if (mActivePointers!=null){
mActivePointers.clear();
}
if (truePointers != null) {
truePointers.clear();
}

mBufferBitmap.eraseColor(Color.TRANSPARENT);
invalidate();

}

}

private void initPaint() {
paint2 = new Paint();
//LinearGradient lg=new LinearGradient(0,0,100,100,Color.RED,Color.BLUE, Shader.TileMode.MIRROR); //参数一为渐变起初点坐标x位
LinearGradient lg=new LinearGradient(0,0,100,100,CYAN,YELLOW, Shader.TileMode.MIRROR); //参数一为渐变起初点坐标x位
paint2.setStyle(Paint.Style.STROKE);//设置画笔的填充模式
paint2.setStrokeJoin(Paint.Join.ROUND);//设置画笔图形接触时笔迹的形状
paint2.setStrokeCap(Paint.Cap.ROUND);//设置画笔离开画板时笔迹的形状
paint2.setStrokeWidth(45);//设置画笔粗细 50
paint2.setAntiAlias(true);//设置抗锯齿
paint2.setColor(Color.WHITE);//设置抗锯齿
paint2.setShader(lg);
}
}

自定义view,用来测试屏幕的更多相关文章

  1. 自定义View和ViewGroup

    为了扫除学习中的盲点,尽可能多的覆盖Android知识的边边角角,决定对自定义View做一个稍微全面一点的使用方法总结,在内容上面并没有什么独特的地方,其他大神们的博客上面基本上都有讲这方面的内容,如 ...

  2. Android 自定义View修炼-Android实现圆形、圆角和椭圆自定义图片View(使用BitmapShader图形渲染方法)

    一.概述 Android实现圆角矩形,圆形或者椭圆等图形,一般主要是个自定义View加上使用Xfermode实现的.实现圆角图片的方法其实不少,常见的就是利用Xfermode,Shader.本文直接继 ...

  3. 自定义View实现五子棋游戏

    成功的路上一点也不拥挤,因为坚持的人太少了. ---简书上看到的一句话 未来请假三天顺带加上十一回家结婚,不得不说真是太坑了,去年婚假还有10天,今年一下子缩水到了3天,只能赶着十一办事了. 最近还在 ...

  4. Android自定义View(一、初体验自定义TextView)

    转载请标明出处: http://blog.csdn.net/xmxkf/article/details/51454685 本文出自:[openXu的博客] 目录: 继承View重写onDraw方法 自 ...

  5. Android的自定义View及View的绘制流程

    目标:实现Android中的自定义View,为理清楚Android中的View绘制流程“铺路”. 想法很简单:从一个简单例子着手开始编写自定义View,对ViewGroup.View类中与绘制View ...

  6. 自定义View和ViewGroup(有这一篇就够了)

    为了扫除学习中的盲点,尽可能多的覆盖Android知识的边边角角,决定对自定义View做一个稍微全面一点的使用方法总结,在内容上面并没有什么独特的地方,其他大神们的博客上面基本上都有讲这方面的内容,如 ...

  7. 【朝花夕拾】Android自定义View篇之(五)Android事件分发机制(上)Touch三个重要方法的处理逻辑

    前言 转载请注明,转自[https://www.cnblogs.com/andy-songwei/p/10998855.html]谢谢! 在自定义View中,经常需要处理Android事件分发的问题, ...

  8. 【朝花夕拾】Android自定义View篇之(四)自定义View的三种实现方式及自定义属性使用介绍

    前言 转载请声明,转自[https://www.cnblogs.com/andy-songwei/p/10979161.html],谢谢! 尽管Android系统提供了不少控件,但是有很多酷炫效果仍然 ...

  9. Android进阶之绘制-自定义View完全掌握(二)

    这是自定义View系列的第二篇博客,我们继续来学习关于自定义View的知识. 今天我们来实现一下广告条案例. 我们要实现的是这样的一个效果. 要想实现这样的效果,我们可以借助ViewPager控件,然 ...

  10. 【转载】自定义View,有这一篇就够了

    为了扫除学习中的忙点,尽可能多的覆盖Android知识的边边角角,决定对自定义View做一个稍微全面一点的使用方法总结,在内容上面并没有什么独特的地方,其他大神们博客上面基本上都有讲这方面的内容,如果 ...

随机推荐

  1. 正则提取器和beanshell处理器组合,将提取的所有id拼接成字符串

    1.添加正则表达式,提取所有id值 2.添加beanshell处理器将所有的id值拼接成字符串 方法一: int N = Integer.parseInt(vars.get("build_m ...

  2. ob_DES_艺恩

    网站 aHR0cHM6Ly93d3cuZW5kYXRhLmNvbS5jbi9Cb3hPZmZpY2UvQk8vWWVhci9pbmRleC5odG1s 抓包返回密文  点到initator,PostA ...

  3. Ubuntu:Docker 容器操作

    创建容器 1.docker run [option] 镜像名 [向启动容器中传入的命令] 常用可选说明 -i 表示以"交互模式"运行容器 -t 表示容器启动后会进入其命令行.加入这 ...

  4. day12-功能实现11

    家居网购项目实现011 以下皆为部分代码,详见 https://github.com/liyuelian/furniture_mall.git 27.功能25-事务管理 27.1下订单问题思考 在生成 ...

  5. [OpenCV实战]21 使用OpenCV的Eigenface

    目录 1 PCA 1.1 方差是什么 1.2 什么是PCA 1.3 什么是矩阵的特征向量和特征值? 1.4 如何计算PCA 2 什么是EigenFaces? 2.1 将图像作为向量 2.2 如何计算如 ...

  6. flask_apscheduler

    网上很多文章包括官网给出来的,更多的是执行的函数和初始话app在同一文件 按照app 路由 模型类 视图 往下延申,项目已经成型,所以上述的方法不适用. (不用操心模型类与app互相导入的错误)下面分 ...

  7. [WPF]数据绑定失效的问题

    有可能的原因如下: 绑定的属性一定是属性而不能是字段,比如: public int data; 是无法生效的,而 public int data {get;set;} 这样才能生效

  8. 《线段树学习笔记》 AC代码索引

    P3372 [模板]线段树 1 | LibreOJ#132. 树状数组 3 :区间修改,区间查询 #include <bits/stdc++.h> #define int long lon ...

  9. 揭开华为云CodeArts TestPlan启发式测试设计神秘面纱!

    摘要:质量是产品的生死线. 本文分享自华为云社区<揭开华为云CodeArts TestPlan启发式测试设计神秘面纱!>,作者:华为云PaaS服务小智 . 2019年12月20日,是美国波 ...

  10. JS实现excel数据透析,形成关系图

    网上查了好多例子,都没有找到答案,只能自己硬着头皮写了 想要的样子: 下面是DEMO,已经实现效果了!!!! 举例  导入 <!DOCTYPE html> <html lang=&q ...