Android实现手写板功能

本文实例为大家分享了Android实现手写板功能的具体代码,供大家参考,具体内容如下

自定义个一个手写板的重点:

笔画为一次down-move-up的集合

撤销笔画并非一次path的动作撤销 应该也是一次down-move -up的撤销

为了更好的笔画需要使用贝塞尔曲线来完成

效果如下:

截图中清楚 的意思是清除 !

具体代码如下:

package com.kyli.base.view;

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Path;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;

import androidx.annotation.Nullable;

import java.util.ArrayList;
import java.util.List;

/**
 * 绘制画板
 */
public class SignBoradView extends View {
    /*4个像素点*/
    private int beierThreshold = 4;
    private float x = 0;
    private float y = 0;
    /*画笔*/
    private Paint mPaint;
    /*宽度*/
    private int strokeWidth = 10;

    /*yanbse*/
    private int color = Color.BLACK;

    /*当前笔画*/
    private Path path;

    private int state = State.CLEAR;

    private interface State {
        /*画板可以使用了*/
        int START = 0;

        /*停止使用画板*/
        int STOP = 1;
        /*清空画板*/
        int CLEAR = 2;
    }

    private List<EveryPenPath> everyPenPaths = new ArrayList<>();

    /*每一个笔画*/
    private static class EveryPenPath {
        public Path path;
    }

    public SignBoradView(Context context) {
        super(context);
    }

    public SignBoradView(Context context, @Nullable AttributeSet attrs) {
        super(context, attrs);
    }

    public SignBoradView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }

    private void initPaint() {
        if (mPaint == null) {
            mPaint = new Paint();
            mPaint.setStrokeWidth(strokeWidth);
            mPaint.setColor(color);
            mPaint.setStyle(Paint.Style.STROKE);
            mPaint.setAntiAlias(true);
            mPaint.setFlags(Paint.ANTI_ALIAS_FLAG);
        }
    }

    public void start() {
        state = State.START;
        initPaint();

    }

    /*停止使用*/
    public void stop() {
        state = State.STOP;
    }

    /*清空画板*/
    public void clear() {
        state = State.CLEAR;
        for (int i = everyPenPaths.size() - 1; i >= 0; i--) {
            EveryPenPath everyPenPath = everyPenPaths.get(i);
            everyPenPath.path.reset();
            everyPenPath.path.close();
            everyPenPath.path = null;

        }
        everyPenPaths.clear();
        invalidate();

    }

    public void back() {
        int count = everyPenPaths.size();
        if (count < 1)
            return;
        EveryPenPath everyPenPath = everyPenPaths.get(count - 1);
        everyPenPath.path.reset();
        everyPenPath.path.close();
        everyPenPath.path = null;
        everyPenPaths.remove(count - 1);
        invalidate();
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        if (state == State.START) {

            /*先绘制完整笔画*/
            for (EveryPenPath e : everyPenPaths) {
                canvas.drawPath(e.path, mPaint);
            }
            //当前进行中的  path!=null
            if (path != null) {
                canvas.drawPath(path, mPaint);
            }

        }

    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        if (state == State.START) {
            if (event.getAction() == MotionEvent.ACTION_UP) {
                actionUp(event);
                invalidate();
                return true;
            }
            if (event.getAction() == MotionEvent.ACTION_MOVE) {
                actionMove(event);
                invalidate();
                return true;
            }
            if (event.getAction() == MotionEvent.ACTION_DOWN) {
                actionDown(event);
                invalidate();
                return true;
            }
        }
        return super.onTouchEvent(event);
    }

    private void actionUp(MotionEvent event) {
        actionMove(event);
        /*构成一个笔画*/
        EveryPenPath everyPenPath = new EveryPenPath();
        everyPenPath.path = path;
        everyPenPaths.add(everyPenPath);
        //将当前画笔置位null;
        path = null;

    }

    /**/
    private void actionMove(MotionEvent event) {
        /*每次移动去绘制贝塞尔曲线*/
        float cX = event.getX();
        float cY = event.getY();
        float dX = Math.abs(cX - x);//变化量
        float dY = Math.abs(cY - y);

        if (dX >= beierThreshold || dY >= beierThreshold) {
            float rX = x + (cX - x) / 2;
            float rY = y + (cY - y) / 2;
            path.quadTo(rX, rY, cX, cY);
            //下次的x 域y 将重新计算
            x = cX;
            y = cY;
        }
    }

    /*开始时*/
    private void actionDown(MotionEvent event) {
        path = new Path();
        x = event.getX();
        y = event.getY();
        path.moveTo(x, y);
    }

    public void setBeierThreshold(int beierThreshold) {
        this.beierThreshold = beierThreshold;
    }

    public void setStrokeWidth(int strokeWidth) {
        this.strokeWidth = strokeWidth;
    }

    public void setColor(int color) {
        this.color = color;
    }

    public Bitmap getResult(int bgColor) {
        if (everyPenPaths.size() == 0)
            return null;
        Bitmap bitmap = Bitmap.createBitmap(getWidth(), getHeight(), Bitmap.Config.ARGB_8888);
        Canvas canvas = new Canvas(bitmap);
        canvas.drawColor(bgColor);
        for (int i = 0; i < everyPenPaths.size(); i++) {
            if (mPaint == null) {
                initPaint();

            }
            canvas.drawPath(everyPenPaths.get(i).path, mPaint);
        }
        return bitmap;
    }

    public Bitmap getResult() {
        return getResult(Color.WHITE);
    }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android实现签名涂鸦手写板

    本文实例为大家分享了Android实现签名涂鸦手写板的具体代码,供大家参考,具体内容如下 布局文件 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:

  • Android编程实现的手写板和涂鸦功能

    本文实例讲述了Android编程实现的手写板和涂鸦功能.分享给大家供大家参考,具体如下: 下面仿一个Android手写板和涂鸦的功能,直接上代码: write_pad.xml <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:greendroid="http://schemas.android.com/apk/res/com.cyrilmottier.androi

  • Android实现手写板功能

    本文实例为大家分享了Android实现手写板功能的具体代码,供大家参考,具体内容如下 自定义个一个手写板的重点: 笔画为一次down-move-up的集合 撤销笔画并非一次path的动作撤销 应该也是一次down-move -up的撤销 为了更好的笔画需要使用贝塞尔曲线来完成 效果如下: 截图中清楚 的意思是清除 ! 具体代码如下: package com.kyli.base.view; import android.content.Context; import android.graphic

  • Android 偷拍功能实现(手机关闭依然拍照)详解及实例代码

     Android 偷拍功能/手机关闭能拍照 效果如下: 其实偷拍与偷录实现方式是一样的,都是使用到的WindowManager来绘制桌面小控件的原理.那我就不多说了- 一.首先我们需要一个SurfaceView: <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android&quo

  • Android实现记事本功能(26)

    本文实例为大家分享了Android实现记事本功能的具体代码,供大家参考,具体内容如下 MainActivity.java代码: package siso.smartnotef.activity; import android.app.AlertDialog; import android.content.DialogInterface; import android.content.Intent; import android.os.Bundle; import android.support.

  • Android登录注册功能 数据库SQLite验证

    本文实例为大家分享了Android登录注册功能的具体代码,供大家参考,具体内容如下 展示效果 代码区 MainActivity(登录方法) public class MainActivity extends AppCompatActivity { @BindView(R.id.editText) EditText editText; @BindView(R.id.editText2) EditText editText2; @BindView(R.id.button) Button button

  • Android实现搜索功能并本地保存搜索历史记录

    本文实例为大家分享了Android实现搜索功能,并且需要显示搜索的历史记录,供大家参考,具体内容如下 效果图: 本案例实现起来很简单,所以可以直接拿来嵌入项目中使用,涉及到的知识点: - 数据库的增删改查操作 - ListView和ScrollView的嵌套冲突解决 - 监听软键盘回车按钮设置为搜索按钮 - 使用TextWatcher( )实时筛选 - 已搜索的关键字再次搜索不重复添加到数据库 - 刚进入页面设置软键盘不因为EditText而自动弹出 代码 RecordSQLiteOpenHel

  • Android ActionBar搜索功能用法详解

    本文实例讲述了Android ActionBar搜索功能用法.分享给大家供大家参考,具体如下: 使用ActionBar SearchView时的注意点: 首先要吐槽一下Android的官方Guide文档 ,关于用法讲得不明确,可能是一直没更新的原因吧. 本来照着文档搞了一下,hint死活出不来,也无法跳转到搜索结果Activity. StackOverflow也有人提出了这个问题,答案说得很明白 - 参考链接. 正确用法 1. 在AndroidManifest.xml中为提供SearchView

  • Android使用Sensor感应器实现线程中刷新UI创建android测力计的功能

    本文实例讲述了Android使用Sensor感应器实现线程中刷新UI创建android测力计的功能.分享给大家供大家参考,具体如下: 前面一篇<Android基于Sensor感应器获取重力感应加速度的方法>我们介绍了sensor的基本知识以及一个使用其中加速度感应器获取数据的例子. 前面提到过一个问题,就是说感应器刷新频率太快,假如我们要做一个UI中,需要根据方向数据绘制一个一个移动的箭头,那么就要太过频繁的刷新绘制界面,占用很多的资源,体验性也会很差,<android 2高级编程>

  • Android拨打电话功能实例详解

    本文实例分析了Android拨打电话功能.分享给大家供大家参考,具体如下: 打电话是手机的一个最基本的功能,现在android智能手机非常流行,里面有多种多样的精彩的手机功能,但是android手机如何实现打电话这个基本功能呢?现以实例说明如下.首先呈上程序: import java.util.regex.Matcher; import java.util.regex.Pattern; import android.app.Activity; import android.content.Int

  • Android 编辑头像功能简单实现实例(图片选取,裁剪)

    本文介绍了Android 编辑头像功能的简单实例,可以实现拍照,图片选取,裁剪. 拍照 public static void startCamera(Fragment fragment){ File file=new File(Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DCIM)+ File.separator+"user_icon.jpg"); Intent intent=new Inten

  • 安卓(Android)实现选择时间功能

    前言 由于大部分android初级教程都没教怎么选择时间,初学者碰到这种难免会有些不知所措,难道要让用户自己输入日期时间?先不说用户体验不好,处理用户输入各式各样的日期格式也要花好大一番功夫,下面看看如何实现android选择时间的功能. android 6.0 中的运行效果 TimePickerDialog和DatePickerDialog介绍 系统封装好了两个类可以供我们直接调用,TimepickerDialog用于选择时间,DatePickerDialog用于选择日期. TimePikck

随机推荐