今天给大家带来一个特别有意思的自定义view---涂鸦,先看看效果

效果图

效果看了,挺好玩吧,其实就是利用万能的Path来画路径,颜色,画笔大小,就是一些设置,废话不多说,直接上真家伙

触摸事件处理

这里的触摸事件主要有按下(MotionEvent.ACTION_DOWN)、移动(MotionEvent.ACTION_MOVE)、抬起(MotionEvent.ACTION_UP),需要对其分别做相应的处理

触摸事件处理方法

@Overridepublic boolean onTouchEvent(MotionEvent event) {float x = event.getX();float y = event.getY();switch (event.getAction()) {case MotionEvent.ACTION_DOWN:// 每次down下去重新new一个PathmPath = new Path();//每一次记录的路径对象是不一样的dp = new DrawPath();dp.path = mPath;dp.paint = mPaint;touch_start(x, y);invalidate();break;case MotionEvent.ACTION_MOVE:touch_move(x, y);invalidate();break;case MotionEvent.ACTION_UP:touch_up();invalidate();break;}return true;}复制代码

按下(MotionEvent.ACTION_DOWN)后执行的touch_start(x, y);

    private void touch_start(float x, float y) {mPath.moveTo(x, y);mX = x;mY = y;}复制代码

移动(MotionEvent.ACTION_MOVE)后执行的touch_move(x, y);

     private void touch_move(float x, float y) {float dx = Math.abs(x - mX);float dy = Math.abs(mY - y);if (dx >= TOUCH_TOLERANCE || dy >= TOUCH_TOLERANCE) {// 从x1,y1到x2,y2画一条贝塞尔曲线,更平滑(直接用mPath.lineTo也可以)mPath.quadTo(mX, mY, (x + mX) / 2, (y + mY) / 2);//mPath.lineTo(mX,mY);mX = x;mY = y;}}复制代码

抬起(MotionEvent.ACTION_UP)后执行的touch_up();

    private void touch_up() {mPath.lineTo(mX, mY);mCanvas.drawPath(mPath, mPaint);//将一条完整的路径保存下来(相当于入栈操作)savePath.add(dp);mPath = null;// 重新置空}复制代码

然后看一下onDraw

     @Overridepublic void onDraw(Canvas canvas) {//canvas.drawColor(0xFFAAAAAA);// 将前面已经画过得显示出来canvas.drawBitmap(mBitmap, 0, 0, mBitmapPaint);if (mPath != null) {// 实时的显示canvas.drawPath(mPath, mPaint);}}复制代码

整个过程走完了,接下来就来考虑一下其他过程

撤销、恢复、重做三个操作

撤销操作

    /*** 撤销* 撤销的核心思想就是将画布清空,* 将保存下来的Path路径最后一个移除掉,* 重新将路径画在画布上面。*/public void undo() {if (savePath != null && savePath.size() > 0) {DrawPath drawPath = savePath.get(savePath.size() - 1);deletePath.add(drawPath);savePath.remove(savePath.size() - 1);redrawOnBitmap();}}复制代码

恢复操作

    /*** 恢复,恢复的核心就是将删除的那条路径重新添加到savapath中重新绘画即可*/public void recover() {if (deletePath.size() > 0) {//将删除的路径列表中的最后一个,也就是最顶端路径取出(栈),并加入路径保存列表中DrawPath dp = deletePath.get(deletePath.size() - 1);savePath.add(dp);//将取出的路径重绘在画布上mCanvas.drawPath(dp.path, dp.paint);//将该路径从删除的路径列表中去除deletePath.remove(deletePath.size() - 1);invalidate();}}复制代码

重做操作

    /*** 重做*/public void redo() {if (savePath != null && savePath.size() > 0) {savePath.clear();redrawOnBitmap();}}复制代码
    private void redrawOnBitmap() {/*mBitmap = Bitmap.createBitmap(screenWidth, screenHeight,Bitmap.Config.RGB_565);mCanvas.setBitmap(mBitmap);// 重新设置画布,相当于清空画布*/initCanvas();Iterator<DrawPath> iter = savePath.iterator();while (iter.hasNext()) {DrawPath drawPath = iter.next();mCanvas.drawPath(drawPath.path, drawPath.paint);}invalidate();// 刷新}复制代码

初始化相关

    public void initCanvas() {setPaintStyle();if (screenWidth > 0 && screenHeight > 0) {mBitmapPaint = new Paint(Paint.DITHER_FLAG);//画布大小mBitmap = Bitmap.createBitmap(screenWidth, screenHeight, Bitmap.Config.ARGB_8888);mBitmap.eraseColor(Color.argb(0, 0, 0, 0));mCanvas = new Canvas(mBitmap);  //所有mCanvas画的东西都被保存在了mBitmap中mCanvas.drawColor(Color.TRANSPARENT);}}//初始化画笔样式private void setPaintStyle() {mPaint = new Paint();mPaint.setStyle(Paint.Style.STROKE);mPaint.setStrokeJoin(Paint.Join.ROUND);// 设置外边缘mPaint.setStrokeCap(Paint.Cap.ROUND);// 形状mPaint.setAntiAlias(true);mPaint.setDither(true);if (currentStyle == 1) {mPaint.setStrokeWidth(currentSize);mPaint.setColor(currentColor);} else {//橡皮擦mPaint.setAlpha(0);mPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_IN));mPaint.setColor(Color.TRANSPARENT);mPaint.setStrokeWidth(50);}}复制代码

样式相关修改

    //以下为样式修改内容//设置画笔样式public void selectPaintStyle(int which) {if (which == 0) {currentStyle = 1;setPaintStyle();}//当选择的是橡皮擦时,设置颜色为白色if (which == 1) {currentStyle = 2;setPaintStyle();}}//选择画笔大小public void selectPaintSize(int which) {currentSize = which;setPaintStyle();}//设置画笔颜色public void selectPaintColor(int which) {currentColor = paintColor[which];setPaintStyle();}复制代码

到此为止,涂鸦的效果有了,实际项目中都是对一个照片进行涂鸦操作,因此添加照片进行再次封装,达到实际项目中的那种效果

添加照片进行涂鸦

利用组合自定义view,结合上面的TabletView和ImageView来到实际项目效果

xml布局

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"android:id="@+id/layout"android:layout_width="match_parent"android:layout_height="match_parent"><ImageViewandroid:id="@+id/img"android:layout_width="match_parent"android:layout_height="match_parent" /><com.tabletview.TabletViewandroid:id="@+id/tabletview"android:layout_width="match_parent"android:layout_height="match_parent" /></FrameLayout>复制代码

继承FrameLayout,在onFinishInflate()中添加view

    @Overrideprotected void onFinishInflate() {super.onFinishInflate();View view = LayoutInflater.from(mContext).inflate(R.layout.tuya, null);layout = (FrameLayout) view.findViewById(R.id.layout);img = (ImageView) view.findViewById(R.id.img);tabletview = (TabletView) view.findViewById(R.id.tabletview);addView(view);}复制代码

获得TabletView去执行相关操作

    /*** 去拿到TabletView去执行相关的操作* @return*/public TabletView getTabletview() {return tabletview;}复制代码

设置照片的方法,实际项目中可利用照片、相册选择照片返回设置

    /*** 设置照片,根据实际项目可支持照片选择、相册、拍照* @param bitmap*/public void setImgBitmap(Bitmap bitmap) {img.setImageBitmap(bitmap);}复制代码

保存照片,这里的权限问题根据6.0、7.0、8.0得设置,可以参考其他的权限讲解

    /*** 保存照片*/public void saveToSDCard() {new Thread() {@Overridepublic void run() {SystemClock.sleep(1000);//view层的截图layout.setDrawingCacheEnabled(true);Bitmap newBitmap = Bitmap.createBitmap(layout.getDrawingCache());layout.setDrawingCacheEnabled(false);//获得系统当前时间,并以该时间作为文件名SimpleDateFormat formatter = new SimpleDateFormat("yyyyMMddHHmmss");Date curDate = new Date(System.currentTimeMillis());//获取当前时间String str = formatter.format(curDate) + "paint.png";File file = new File("sdcard/" + str);if (!file.exists()) {try {file.createNewFile();} catch (IOException e) {e.printStackTrace();}}FileOutputStream fos = null;try {fos = new FileOutputStream(file);} catch (Exception e) {e.printStackTrace();}newBitmap.compress(Bitmap.CompressFormat.PNG, 100, fos);}}.start();}复制代码

代码中应用

xml布局

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"android:layout_width="match_parent"android:layout_height="match_parent"tools:context="com.tabletview.MainActivity"><com.tabletview.TuyaViewandroid:id="@+id/tuyaview"android:layout_width="match_parent"android:layout_height="match_parent"/><LinearLayoutandroid:layout_width="match_parent"android:layout_height="wrap_content"android:orientation="vertical"><Buttonandroid:id="@+id/btn_undo"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="撤销" /><Buttonandroid:id="@+id/btn_redo"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="重做" /><Buttonandroid:id="@+id/btn_save"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="保存" /><Buttonandroid:id="@+id/btn_recover"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="恢复" /><Buttonandroid:id="@+id/btn_paintcolor"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="颜色" /><Buttonandroid:id="@+id/btn_paintsize"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="画笔大小" /><Buttonandroid:id="@+id/btn_paintstyle"android:layout_width="wrap_content"android:layout_height="wrap_content"android:text="画笔类型" /><SeekBarandroid:id="@+id/sb_size"android:layout_width="match_parent"android:layout_height="wrap_content" /></LinearLayout></RelativeLayout>复制代码

java代码

    private void initView() {btn_undo = (Button) findViewById(R.id.btn_undo);btn_undo.setOnClickListener(this);btn_redo = (Button) findViewById(R.id.btn_redo);btn_redo.setOnClickListener(this);btn_save = (Button) findViewById(R.id.btn_save);btn_save.setOnClickListener(this);btn_recover = (Button) findViewById(R.id.btn_recover);btn_recover.setOnClickListener(this);sb_size = (SeekBar) findViewById(R.id.sb_size);sb_size.setOnClickListener(this);btn_paintcolor = (Button) findViewById(R.id.btn_paintcolor);btn_paintcolor.setOnClickListener(this);btn_paintsize = (Button) findViewById(R.id.btn_paintsize);btn_paintsize.setOnClickListener(this);btn_paintstyle = (Button) findViewById(R.id.btn_paintstyle);btn_paintstyle.setOnClickListener(this);tuyaview = (TuyaView) findViewById(R.id.tuyaview);//设置照片Bitmap bitmap= BitmapFactory.decodeResource(getResources(),R.mipmap.ic_launcher);tuyaview.setImgBitmap(bitmap);
//tabletView = tuyaview.getTabletview();tabletView.requestFocus();tabletView.selectPaintSize(sb_size.getProgress());sb_size.setOnSeekBarChangeListener(new MySeekChangeListener());}class MySeekChangeListener implements SeekBar.OnSeekBarChangeListener {@Overridepublic void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {tabletView.selectPaintSize(seekBar.getProgress());//Toast.makeText(MainActivity.this,"当前画笔尺寸为"+seekBar.getProgress(),Toast.LENGTH_SHORT ).show();}@Overridepublic void onStartTrackingTouch(SeekBar seekBar) {tabletView.selectPaintSize(seekBar.getProgress());//Toast.makeText(MainActivity.this,"当前画笔尺寸为"+seekBar.getProgress(),Toast.LENGTH_SHORT ).show();}@Overridepublic void onStopTrackingTouch(SeekBar seekBar) {}}@Overridepublic void onClick(View v) {switch (v.getId()) {case R.id.btn_undo:tabletView.undo();break;case R.id.btn_redo:tabletView.redo();break;case R.id.btn_save:
//                tabletView.saveToSDCard();tuyaview.saveToSDCard();break;case R.id.btn_recover:tabletView.recover();break;case R.id.btn_paintcolor:sb_size.setVisibility(View.GONE);showPaintColorDialog(v);break;case R.id.btn_paintsize:sb_size.setVisibility(View.VISIBLE);break;case R.id.btn_paintstyle:sb_size.setVisibility(View.GONE);showMoreDialog(v);break;}}private int select_paint_color_index = 0;private int select_paint_style_index = 0;public void showPaintColorDialog(View parent) {AlertDialog.Builder alertDialogBuilder = new AlertDialog.Builder(MainActivity.this);alertDialogBuilder.setTitle("选择画笔颜色:");alertDialogBuilder.setSingleChoiceItems(R.array.paintcolor, select_paint_color_index, new DialogInterface.OnClickListener() {@Overridepublic void onClick(DialogInterface dialogInterface, int which) {select_paint_color_index = which;tabletView.selectPaintColor(which);dialogInterface.dismiss();}});alertDialogBuilder.setNegativeButton("取消", new DialogInterface.OnClickListener() {@Overridepublic void onClick(DialogInterface dialogInterface, int i) {dialogInterface.dismiss();}});alertDialogBuilder.create().show();}public void showMoreDialog(View parent) {AlertDialog.Builder alertDialogBuilder = new AlertDialog.Builder(this);alertDialogBuilder.setTitle("选择画笔或橡皮擦:");alertDialogBuilder.setSingleChoiceItems(R.array.paintstyle, select_paint_style_index, new DialogInterface.OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {select_paint_style_index = which;tabletView.selectPaintStyle(which);dialog.dismiss();}});alertDialogBuilder.setNegativeButton("取消", new DialogInterface.OnClickListener() {@Overridepublic void onClick(DialogInterface dialog, int which) {dialog.dismiss();}});alertDialogBuilder.create().show();}复制代码

TabletView github demo地址

个人博客

自定义view————涂鸦画板相关推荐

  1. 王姨劝我学HarmonyOS鸿蒙2.0系列教程之六自定义View涂鸦项目实战!

    这一节写个简单的项目,涂鸦,简单来讲就是画什么,显示什么.最好的网站,就是官网,这里再发一下: https://developer.harmonyos.com/cn/docs/documentatio ...

  2. DrawBoard 是一个自定义 View 实现的画板;方便对图片进行各种编辑或涂鸦相关操作

    DrawBoard 项目地址:jenly1314/DrawBoard 简介: :art: DrawBoard 是一个自定义 View 实现的画板:方便对图片进行各种编辑或涂鸦相关操作 更多:作者    ...

  3. 自定义view实现涂鸦(画板)功能

    自定义view实现涂鸦功能,包括撤销.恢复.重做.保存以及橡皮擦(在风格中实现)功能,小模块包括画笔颜色调整.画笔尺寸调整.画笔类型(包括正常画笔以及橡皮擦功能),之后又陆续实现了画圆.画矩形以及画箭 ...

  4. android canvas 手写,自定义view—Canvas实现手写板和涂鸦功能

    学习导航 第一节:http://blog..net/bobo8945510/article/details/53197727 -自定义View-自定义属性及引用 第二节:http://blog..ne ...

  5. 一个涂鸦笔记本的自定义View第三方库

    一个涂鸦笔记本的自定义View第三方库 目录 一个涂鸦笔记本的自定义View第三方库 目录 简介 集成 初始化 其他方法说明 功能演示 源码及jar包 简介 还记得曾经QQ的涂鸦功能吗?现在这个第三方 ...

  6. 【Android】自定义View、画家(画布)Canvas与画笔Paint的应用——画图、涂鸦板app的实现

    利用一个简单的画图app来说明安卓的图形处理类与自定义View的应用. 如下图,有一个供用户自己任意画图.涂鸦的app, 这里不做那么花俏了,仅提供黑白两色,但可以改变笔尖的粗细. 实质上这里的橡皮擦 ...

  7. iOS:quartz2D绘图小项目(涂鸦画板)

    介绍:学了quartz2D的绘图知识后,我根据它的一些功能制作了一个小项目:涂鸦画板. 功能:绘制各种图形,还可以选取相册上的照片做涂鸦,然后保存到相册中.其中,还包括功能有:颜色的选取.线宽的选取. ...

  8. Android涂鸦画板原理详解——从初级到高级(二)

    前言 前面写了<Android涂鸦画板原理详解--从初级到高级(一)>,讲了涂鸦原理初级和中级的应用,现在讲解高级应用.如果没有看过前面一篇文章的同学,建议先去看看哈. 准备 高级涂鸦涉及 ...

  9. Android自定义View--简易画板

    自定义VIew实现简易画板效果,功能包括清空.选择颜色,选择大小,效果如下 画板布局: <?xml version="1.0" encoding="utf-8&qu ...

  10. android 画布实现签名,Android 自定义View手写签名并保存图片

    1.自定义View--支撑设置画笔色彩,画笔宽度,画板色彩,铲除画板,查看是否有签名,保存画板图片(仿制粘贴可直接使用) /***CreatedbyYyyyQon2020/3/5. *电子签名*/pu ...

最新文章

  1. 如何修改可运行Jar包,如何反编译Jar包
  2. linux命令行ps1变量_利用Shell中变量PS1定制Linux Shell命令主提示符
  3. C++ STL 初步介绍01
  4. android转流媒体,android 4.4中的流媒体渲染过程
  5. mysql select count 5万条数据很慢_mysql亿级数据数据库优化方案测试银行交易流水记录的查询...
  6. MySql数据库驱动类
  7. 全国计算机一级12月考试答案,12月到12月全国高校计算机等级考试(广西考区)一级笔试真题及答案...
  8. mask rcnn数据转换为tfrecord数据
  9. android view state,Android状态系统(二)——View状态组合
  10. C语言解释一下BA无标度网络
  11. ROLLUP函数问题
  12. matlab求六自由度机械臂,基于人工势场的六自由度空间机械臂避障路径
  13. 随机过程总结(1)--一些基本概念
  14. Python提取CSV数据统计四分位数
  15. it人才外包公司招人真的很难吗?
  16. DLang 编译实验
  17. CSAPP实验二——二进制炸弹bomb
  18. ultrascale学习笔记之remoteproc启动R5
  19. “快用助手”动起苹果商店奶酪
  20. android释放内存只有1GB,手机内存不够用?教你5秒删掉1G垃圾,提升速度!

热门文章

  1. 【趣味实践】自动补帧算法——RIFE的使用
  2. Spatial Transformer Networks(STN)理解
  3. 一些有意思的js代码
  4. 一个完整的数据挖掘项目-纽约市建筑能源之星预测
  5. 关于while循环终止循环的三种方式
  6. TotalCommander常用操作
  7. python填充三角形颜色_python的pillow用ImageDraw.Draw.polygon如何填充半透明的颜色
  8. android 多媒体封装格式详解---MKV
  9. 剪映怎么导入mkv_mkv用什么播放器打开_什么播放器可以打开mkv格式-系统城
  10. 结对开发项目--石家庄地铁web版