JS canvas实现画板和签字板功能


Posted in Javascript onFebruary 23, 2021

本文实例为大家分享了JS canvas实现画板/签字板功能的具体代码,供大家参考,具体内容如下

前言

常见的电子教室里的电子黑板。

本文特点:

原生JS
封装好的模块

最简代码样例

<!DOCTYPE html>
<html lang="en">
<head>
 <meta charset="UTF-8">
 <meta http-equiv="X-UA-Compatible" content="IE=edge">
 <meta name="viewport" content="width=device-width, initial-scale=1.0">
 <title>Document</title>
</head>
<body>
 <canvas id="canvas"></canvas>
 <script>
 let c = document.getElementById('canvas');
 c.width = window.innerWidth;
 c.height = window.innerHeight;
 let ctx = c.getContext('2d');

 // draw one black board
 ctx.fillStyle = "black";
 ctx.fillRect(0,0,600,300);

 // 按下标记
 let onoff = false,
  oldx = -10,
  oldy = -10;

 // 设置颜色
 let linecolor = "white";

 // 设置线宽
 let linw = 4;

 // 添加鼠标事件
 // 按下
 c.addEventListener('mousedown', event => {
  onoff = true;
  // 位置 - 10是为了矫正位置,把绘图放在鼠标指针的顶端
  oldx = event.pageX - 10;
  oldy = event.pageY - 10;
 },false);
 // 移动
 c.addEventListener('mousemove', event => {
  if(onoff == true){
  let newx = event.pageX - 10,
   newy = event.pageY - 10;

  // 绘图
  ctx.beginPath();
  ctx.moveTo(oldx,oldy);
  ctx.lineTo(newx,newy);
  ctx.strokeStyle = linecolor;
  ctx.lineWidth = linw;
  ctx.lineCap = "round";
  ctx.stroke();
  // 每次移动都要更新坐标位置
  oldx = newx,
  oldy = newy;
  }
 }, true);
 // 弹起
 c.addEventListener('mouseup', ()=> {
  onoff = false;
 },false);
 </script>
</body>
</html>

结果展示

JS canvas实现画板和签字板功能

代码讲解

思路

1、鼠标按下,开始描画。鼠标按下事件。
2、鼠标弹起,结束描画。鼠标弹起事件。
3、鼠标按下移动,路径画线。鼠标移动事件。

代码讲解

整体思路:按下鼠标,触发移动的开关,移动后开始记录线条(用移动后的坐标-移动前的坐标,然后绘线),每次移动都会更新旧坐标。松开鼠标后,释放移动开关。

1、只有在鼠标按下,才会触发移动绘图的效果,所以需要增加一个状态判断。
2、因为鼠标指针和实际位置有一个偏移量,所以在坐标定位的时候,需要增加pagex-10从而使坐标位于指针的尖端处。
3、每次移动都要更新坐标位置,用小段的线段来模拟不规则的线。

封装模块

<canvas id="canvas"></canvas>
<script>
 class Board{
 constructor(canvasName = 'canvas', data = new Map([
  ["onoff", false],
  ["oldx", -10],
  ["oldy", -10],
  ["fillStyle", "black"],
  ["lineColor", "white"],
  ["lineWidth", 4],
  ["lineCap", "round"],
  ["canvasWidth", window.innerWidth],
  ["canvasHeight", window.innerHeight]
 ])){
  // this.data = data;
  this.c = document.getElementById(canvasName);
  this.ctx = this.c.getContext('2d');
  this.onoff = data.get("onoff");
  this.oldx = data.get("oldx");
  this.oldy = data.get("oldy");
  this.lineColor = data.get("lineColor");
  this.lineWidth = data.get("lineWidth");
  this.lineCap = data.get("lineCap");

  this.c.width = data.get("canvasWidth");
  this.c.height = data.get("canvasHeight");

  this.ctx.fillStyle = data.get("fillStyle");
  this.ctx.fillRect(0,0,600,300);
 }

 eventOperation(){
  // 添加鼠标事件
  // 按下
  this.c.addEventListener('mousedown', event => {
  this.onoff = true;
  // 位置 - 10是为了矫正位置,把绘图放在鼠标指针的顶端
  this.oldx = event.pageX - 10;
  this.oldy = event.pageY - 10;
  },false);
  // 移动
  this.c.addEventListener('mousemove', event => {
  if(this.onoff == true){
   let newx = event.pageX - 10,
   newy = event.pageY - 10;

   // 绘图
   this.ctx.beginPath();
   this.ctx.moveTo(this.oldx,this.oldy);
   this.ctx.lineTo(newx,newy);

   this.ctx.strokeStyle = this.lineColor;
   this.ctx.lineWidth = this.lineWidth;
   this.ctx.lineCap = this.lineCap;
   
   this.ctx.stroke();
   // 每次移动都要更新坐标位置
   this.oldx = newx,
   this.oldy = newy;
  }
  }, true);
  // 弹起
  this.c.addEventListener('mouseup', ()=> {
  this.onoff = false;
  },false);
 }

 }

 let board = new Board();
 board.eventOperation();
</script>

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

Javascript 相关文章推荐
ymPrompt的doHandler方法来实现获取子窗口返回值的方法
Jun 25 Javascript
JavaScript EasyPager 分页函数
May 25 Javascript
javascript 应用小技巧方法汇总
Jul 05 Javascript
微信支付如何实现内置浏览器的H5页面支付
Sep 25 Javascript
前端性能优化及技巧
May 06 Javascript
JavaScript版经典游戏之扫雷游戏完整示例【附demo源码下载】
Dec 12 Javascript
Vue.js中用v-bind绑定class的注意事项
Dec 13 Javascript
使用AngularJS 跨站请求如何解决jsonp请求问题
Jan 16 Javascript
ng2学习笔记之bootstrap中的component使用教程
Mar 09 Javascript
原生js简单实现放大镜特效
May 16 Javascript
Javascript实现秒表计时游戏
May 27 Javascript
详解JavaScript之Array.reduce源码解读
Nov 01 Javascript
基于vue-simple-uploader封装文件分片上传、秒传及断点续传的全局上传插件功能
Feb 23 #Vue.js
js实现验证码干扰(动态)
Feb 23 #Javascript
js实现验证码干扰(静态)
Feb 22 #Javascript
JavaScript实现H5接金币功能(实例代码)
Feb 22 #Javascript
nestjs返回给前端数据格式的封装实现
Feb 22 #Javascript
NestJs使用Mongoose对MongoDB操作的方法
Feb 22 #Javascript
linux服务器快速卸载安装node环境(简单上手)
Feb 22 #Javascript
You might like
十天学会php之第七天
2006/10/09 PHP
php随机输出名人名言的代码
2012/10/07 PHP
对象失去焦点时自己动提交数据的实现代码
2012/11/06 PHP
浅谈PHP错误类型及屏蔽方法
2017/05/27 PHP
PHP钩子与简单分发方式实例分析
2017/09/04 PHP
php支付宝APP支付功能
2020/07/29 PHP
cookie 最近浏览记录(中文escape转码)具体实现
2013/06/08 Javascript
Knockout数组(observable)使用详解示例
2013/11/15 Javascript
Ajax局部更新导致JS事件重复触发问题的解决方法
2014/10/14 Javascript
jQuery插件EnPlaceholder实现输入框提示文字
2015/06/05 Javascript
jQuery简单实现两级下拉菜单效果代码
2015/09/15 Javascript
谈谈我对JavaScript原型和闭包系列理解(随手笔记9)
2015/12/24 Javascript
jQuery实现左侧导航模块的显示与隐藏效果
2016/07/04 Javascript
基于JavaScript实现本地图片预览
2017/02/08 Javascript
jQuery常见面试题之DOM操作详析
2017/07/05 jQuery
浅谈vue.js导入css库(elementUi)的方法
2018/03/09 Javascript
angularjs 的数据绑定实现原理
2018/07/02 Javascript
ng-zorro-antd 入门初体验
2018/12/03 Javascript
Vue数据绑定实例写法
2019/08/06 Javascript
vue 实现Web端的定位功能 获取经纬度
2019/08/08 Javascript
Jquery Datatables的使用详解
2020/01/30 jQuery
python threading模块操作多线程介绍
2015/04/08 Python
Python连接phoenix的方法示例
2017/09/29 Python
python3利用smtplib通过qq邮箱发送邮件方法示例
2017/12/03 Python
python3利用Dlib19.7实现人脸68个特征点标定
2018/02/26 Python
对python csv模块配置分隔符和引用符详解
2018/12/12 Python
对Pycharm创建py文件时自定义头部模板的方法详解
2019/02/12 Python
Python3.5内置模块之random模块用法实例分析
2019/04/26 Python
scrapy结合selenium解析动态页面的实现
2020/09/28 Python
安装不同版本的tensorflow与models方法实现
2021/02/20 Python
HTML5地理定位与第三方工具百度地图的应用
2016/11/17 HTML / CSS
2014年庆祝国庆65周年演讲稿
2014/09/21 职场文书
公司人事任命通知
2015/04/20 职场文书
2016年公司新年寄语
2015/08/17 职场文书
护理工作心得体会
2016/01/22 职场文书
用Python的绘图库(matplotlib)绘制小波能量谱
2021/04/17 Python