首页 > 其他分享 >Canvas:实现在线画板操作

Canvas:实现在线画板操作

时间:2024-07-10 20:56:21浏览次数:19  
标签:Canvas 在线 画板 ctx canvas 画布 let active document

想象一下,用几行代码就能创造出如此逼真的图像和动画,仿佛将艺术与科技完美融合,前端开发的Canvas技术正是这个数字化时代中最具魔力的一环,它不仅仅是网页的一部分,更是一个无限创意的画布,一个让你的想象力自由驰骋的平台。

目录

基础页面搭建

绘画操作

按钮点击事件

最后总结


基础页面搭建

接下来借助canvas实现一个简易的画版操作,首先这里我们设置一下画布的一些基础布局样式:

<body>
    <canvas id="canvas" width="800" height="600" style="border: 1px solid #ccc;"></canvas>
    <hr>
    <button id="boldBtn" type="button">粗线条</button>
    <button id="thinBtn" type="button">细线条</button>
    <button id="saveBtn" type="button">保持签名</button>
    <input type="color" name="" id="color" value="">
    <button class="clearBtn">橡皮擦</button>
    <button id="nullBtn">清空画布</button>
</body>

实现的效果如下所示,一个比较简陋的画布,以及一些基础操作的按钮:

绘画操作

接下来我们开始实现绘画操作的功能,通过监听鼠标不同情况的事件来实现画布上书写内容的操作,首先这里我们先绘制画布以及获取对应的dom元素进行操作:

<script>
    // 获取canvas画布绘制上下文对象以及获取2d画笔对象
    let canvas = document.getElementById("canvas");      
    let ctx = canvas.getContext("2d");
    ctx.lineJoin = 'round'; // 线条连接处为圆角
    ctx.lineCap = 'round'; // 线条端点为圆角

    // 获取DOM元素
    let boldBtn = document.getElementById("boldBtn");
    let thinBtn = document.getElementById("thinBtn");
    let saveBtn = document.getElementById("saveBtn");
    let color = document.getElementById("color");
    let clearBtn = document.getElementsByClassName("clearBtn")[0];
    let nullBtn = document.getElementById("nullBtn");
</script>

接下来开始开始监听鼠标的按下、弹起、移动以及离开画布的四种情况的操作,完整代码如下:

<script>
    // 获取canvas画布绘制上下文对象以及获取2d画笔对象
    let canvas = document.getElementById("canvas");      
    let ctx = canvas.getContext("2d");
    ctx.lineJoin = 'round'; // 线条连接处为圆角
    ctx.lineCap = 'round'; // 线条端点为圆角

    // 获取DOM元素
    let boldBtn = document.getElementById("boldBtn");
    let thinBtn = document.getElementById("thinBtn");
    let saveBtn = document.getElementById("saveBtn");
    let color = document.getElementById("color");
    let clearBtn = document.getElementsByClassName("clearBtn")[0];
    let nullBtn = document.getElementById("nullBtn");

    // 设置是否开始绘画的变量
    let isDraw = false;

    // 鼠标按下去函数
    canvas.onmousedown = function(e) {
        isDraw = true; // 开始绘画
        ctx.beginPath(); // 开始路径
        // 获取鼠标点击的坐标
        let x = e.pageX - canvas.offsetLeft; 
        let y = e.pageY - canvas.offsetTop; 
        ctx.moveTo(x, y); // 移动到当前点
    }
    // 鼠标移动函数
    canvas.onmousemove = function(e) {
        if (isDraw) { // 如果开始绘画
            let x = e.pageX - canvas.offsetLeft; 
            let y = e.pageY - canvas.offsetTop; 
            ctx.lineTo(x, y); // 绘制
            ctx.stroke()
        }
    }
    // 鼠标弹起函数
    canvas.onmouseup = function(e) {
        isDraw = false; // 结束绘画
        ctx.closePath(); // 结束路径
    }
    // 鼠标离开画布函数
    canvas.onmouseleave = function(e) {
        isDraw = false; // 结束绘画
        ctx.closePath(); // 结束路径
    }
</script>

最终呈现的效果如下所示:

按钮点击事件

接下来给画布下方的按钮设置对应的点击事件,让其能够实现对应的功能,如下:

粗线条:这里设置点击事件函数如下,遮盖策略设置在现有画布上下文之上绘制新图形,点击之后宽度增加到20,并添加对应激活类名active,删除其他按钮可能存在的active类名:

// 粗线条函数
boldBtn.onclick = function(e) {
    ctx.globalCompositeOperation = 'source-over'; 
    ctx.lineWidth = 20
    boldBtn.classList.add("active");
    thinBtn.classList.remove("active");
    clearBtn.classList.remove("active");
}

细线条:这里设置点击事件函数如下,遮盖策略设置在现有画布上下文之上绘制新图形,点击之后宽度减少到2,并添加对应激活类名active,删除其他按钮可能存在的active类名:

// 细线条函数
thinBtn.onclick = function(e) {
    ctx.globalCompositeOperation = 'source-over'; 
    ctx.lineWidth = 2
    thinBtn.classList.add("active");
    boldBtn.classList.remove("active");
    clearBtn.classList.remove("active");
}

保存签名:获取图片数据并创建一个a标签进行文件下载:

// 保存签名函数
saveBtn.onclick = function(e) {
    let urlData = canvas.toDataURL(); // 获取图片数据
    // let img = new Image()
    // img.src = urlData; // 图片地址
    // document.body.appendChild(img); // 添加图片到body

    let downloadA = document.createElement("a"); // 创建a标签
    downloadA.setAttribute("download", "签名"); // 设置下载文件名
    downloadA.href = urlData; // 设置a标签的地址
    downloadA.click(); // 点击a标签
}

修改颜色:设置监听函数,动态修改canvas颜色:

// 颜色选择函数
color.onchange = function(e) {
    ctx.strokeStyle = color.value; // 设置线条颜色
}

橡皮擦:这里设置点击事件函数如下,遮盖策略设置现有内容保持在新图形不重叠的地方,点击之后区域呈现空白,并添加对应激活类名active,删除其他按钮可能存在的active类名:

// 橡皮擦函数
clearBtn.onclick = function(e) {
    // 清除画布
    ctx.globalCompositeOperation = 'destination-out';
    ctx.lineWidth = 30
    clearBtn.classList.add("active");
    boldBtn.classList.remove("active");
    thinBtn.classList.remove("active");
}

清空画布:调用canvas函数直接清除整个画布的内容:

// 清空画布函数
nullBtn.onclick = function(e) {
    ctx.clearRect(0, 0, 800, 600);
}

最终呈现的效果如下所示:

最后总结

本次代码实现了一个简单的画板功能,让用户可以在网页上绘制、调整线条粗细、选择颜色、保存签名等操作。以下是代码的实现思路:

1)HTML结构部分

在<body>中包含一个 <canvas> 元素,用于绘制图形。

一组按钮和一个颜色选择器,用于控制画笔的粗细、颜色以及清除和保存绘制的功能。

2)CSS部分

定义了一个按钮的样式 .active,用于表示当前选中状态,例如粗线条、细线条、橡皮擦等按钮会根据点击状态变化背景颜色。

3)JavaScript部分

《1》获取元素:

使用 document.getElementById 和 document.getElementsByClassName 获取 <canvas> 和各种控制按钮。

《2》绘图相关事件:

canvas.onmousedown:鼠标按下时开始绘画,记录起始点。
canvas.onmousemove:鼠标移动时,如果正在绘画(isDraw为真),则根据当前鼠标位置绘制线条。
canvas.onmouseup 和 canvas.onmouseleave:鼠标抬起或离开画布时结束绘画路径。

《3》功能按钮事件:

粗线条 (boldBtn) 和 细线条 (thinBtn):设置线条宽度,并通过 classList 控制按钮的选中状态。
保存签名 (saveBtn):使用 canvas.toDataURL() 获取绘制内容的DataURL,创建一个下载链接,允许用户保存签名。
颜色选择 (color):通过 color.onchange 设置绘制线条的颜色。
橡皮擦 (clearBtn):使用 ctx.globalCompositeOperation = 'destination-out' 实现擦除效果,清除画布上的内容。
清空画布 (nullBtn):使用 ctx.clearRect() 清除整个画布的内容。

《4》绘图上下文:

使用 getContext('2d') 获取 Canvas 2D 渲染上下文对象 ctx,设置线条的样式(圆角连接和端点)。

这段代码整合了 HTML、CSS 和 JavaScript,提供了一个简单而功能丰富的在线绘图工具,用户可以通过点击不同的按钮和操作来绘制、擦除、保存签名等。完整代码如下:

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
    <style>
        button.active {
            color: #fff;
            background-color: orange;
        }
    </style>
</head>

<body>
    <canvas id="canvas" width="800" height="600" style="border: 1px solid #ccc;"></canvas>
    <hr>
    <button id="boldBtn" type="button">粗线条</button>
    <button id="thinBtn" type="button">细线条</button>
    <button id="saveBtn" type="button">保存签名</button>
    <input type="color" name="" id="color" value="">
    <button class="clearBtn">橡皮擦</button>
    <button id="nullBtn">清空画布</button>
<script>
    // 获取canvas画布绘制上下文对象以及获取2d画笔对象
    let canvas = document.getElementById("canvas");      
    let ctx = canvas.getContext("2d");
    ctx.lineJoin = 'round'; // 线条连接处为圆角
    ctx.lineCap = 'round'; // 线条端点为圆角

    // 获取DOM元素
    let boldBtn = document.getElementById("boldBtn");
    let thinBtn = document.getElementById("thinBtn");
    let saveBtn = document.getElementById("saveBtn");
    let color = document.getElementById("color");
    let clearBtn = document.getElementsByClassName("clearBtn")[0];
    let nullBtn = document.getElementById("nullBtn");

    // 设置是否开始绘画的变量
    let isDraw = false;

    // 鼠标按下去函数
    canvas.onmousedown = function(e) {
        isDraw = true; // 开始绘画
        ctx.beginPath(); // 开始路径
        // 获取鼠标点击的坐标
        let x = e.pageX - canvas.offsetLeft; 
        let y = e.pageY - canvas.offsetTop; 
        ctx.moveTo(x, y); // 移动到当前点
    }
    // 鼠标移动函数
    canvas.onmousemove = function(e) {
        if (isDraw) { // 如果开始绘画
            let x = e.pageX - canvas.offsetLeft; 
            let y = e.pageY - canvas.offsetTop; 
            ctx.lineTo(x, y); // 绘制
            ctx.stroke()
        }
    }
    // 鼠标弹起函数
    canvas.onmouseup = function(e) {
        isDraw = false; // 结束绘画
        ctx.closePath(); // 结束路径
    }
    // 鼠标离开画布函数
    canvas.onmouseleave = function(e) {
        isDraw = false; // 结束绘画
        ctx.closePath(); // 结束路径
    }

    // 粗线条函数
    boldBtn.onclick = function(e) {
        ctx.globalCompositeOperation = 'source-over'; 
        ctx.lineWidth = 20
        boldBtn.classList.add("active");
        thinBtn.classList.remove("active");
        clearBtn.classList.remove("active");
    }
    // 细线条函数
    thinBtn.onclick = function(e) {
        ctx.globalCompositeOperation = 'source-over'; 
        ctx.lineWidth = 2
        thinBtn.classList.add("active");
        boldBtn.classList.remove("active");
        clearBtn.classList.remove("active");
    }
    // 保存签名函数
    saveBtn.onclick = function(e) {
        let urlData = canvas.toDataURL(); // 获取图片数据
        // let img = new Image()
        // img.src = urlData; // 图片地址
        // document.body.appendChild(img); // 添加图片到body

        let downloadA = document.createElement("a"); // 创建a标签
        downloadA.setAttribute("download", "签名"); // 设置下载文件名
        downloadA.href = urlData; // 设置a标签的地址
        downloadA.click(); // 点击a标签
    }
    // 颜色选择函数
    color.onchange = function(e) {
        ctx.strokeStyle = color.value; // 设置线条颜色
    }
    // 橡皮擦函数
    clearBtn.onclick = function(e) {
        // 清除画布
        ctx.globalCompositeOperation = 'destination-out';
        ctx.lineWidth = 30
        clearBtn.classList.add("active");
        boldBtn.classList.remove("active");
        thinBtn.classList.remove("active");
    }
    // 清空画布函数
    nullBtn.onclick = function(e) {
        ctx.clearRect(0, 0, 800, 600);
    }
</script>
</body>
</html>

标签:Canvas,在线,画板,ctx,canvas,画布,let,active,document
From: https://blog.csdn.net/qq_53123067/article/details/140332017

相关文章

  • 字节面试题:在线表格功能怎么实现?怎么测?
    最近有小伙伴私信问我怎么不更新了,期待更新,甚是感动。简述下自己近况:还在干测试,最近忙活的事情大概是自动化测试、性能测试以及业务等等,主打一个啥活都干。业余时间,尝试在短视频赛道搞一些个人兴趣领域的创作,所以博客不太更新,想分享的东西还是有的,后续仍然会记录一些工作心得,感......
  • 记一次【PPT制作、PPT在线编辑预览】功能开发过程
    背景    早上,正愉快写着helloWord(Java)。老大对我来了句,搞个【根据用户输入的markdow文本内容,将数据插入到PPT模板中,生成新的PPT文件。并且要可以在线预览以及编辑】    。(ps:就是直接照着人家的功能抄一个出来,参考网址:https://www.*******.vip)。并且跟我说Apach......
  • 基于SpringBoot + SpringCloud+ElasticSear的在线教育管理系统设计与实现(MySQL、Mongo
    本项目适合做计算机相关专业的毕业设计,课程设计,技术难度适中、工作量比较充实。完整资源获取点击下载完整资源1、资源项目源码均已通过严格测试验证,保证能够正常运行;2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通;3、本项目比较适合计算......
  • 【全开源】2024最新版在线客服系统PHP源码(全新UI+终身使用+安装教程)
    PHP在线客服系统主要功能:1.用户信息用户提交:新用户可以通过表单留言输入相关信息,如用户名、密码、邮箱等,完成后获得唯一的用户ID和密码。2.客服管理客服信息管理:管理客服人员的基本信息,如姓名、工号、权限等。客服工作状态:实时显示客服人员的在线/离线状态,方便客户选择合......
  • 在线博客
    新知识汇总在SQL中,YEAR 函数:计算年份。写上asyear:因为默认函数返回年份在结果集无列名,所以自定义一个,避免它默认生成不直观的。最后按类型去映射。加入jwtt依赖(JSONWebTokens)jwt三部分:A:定义类型,签发算法B:body部分即负载(userid)C:自定义密钥,不能被解密定义jwt工具类:cr......
  • Echarts:渲染成Canvas还是SVG,该如何抉择?
    ECharts从初始一直使用Canvas绘制图表。而EChartsv4.0发布了SVG渲染器,从而提供了一种新的选择。在初始化图表实例时,只需设置renderer参数为'canvas'或'svg'即可指定渲染器,比较方便。贝格前端工场带领大家看下如何选择。一、Canvas和SVG的区分Canvas和SVG都是......
  • 网络安全专家的在线工具箱:顶级资源和平台推荐
    前言:在当前信息技术快速发展的时代,网络安全已成为一个无法忽视的重要议题。面对日益复杂的网络威胁,网络安全人员需不断寻找新的工具和方法以防御潜在的攻击。本文将为网络安全专业人员推荐几个实用的在线工具和平台,这些资源能帮助他们在日常工作中提高效率,加强安全防护。简介:在......
  • springboot信息化在线教学平台的设计与实现(文档+源码)
    大家好,我是永钊,一个混迹在java圈的码农,今天要和大家聊的是一款基于springboot的信息化在线教学平台,项目源码请联系永钊,目前有各类成品毕设javawebsshssmspringboot等等项目框架,源码丰富,欢迎咨询。 本网站系统是一个B/S模式系统,采用SpringBoot框架,MYSQL数据库设计开发......
  • 古典舞在线交流平台的设计与实现
    摘 要随着互联网技术的发展,各类网站应运而生,网站具有新颖、展现全面的特点。因此,为了满足用户古典舞在线交流的需求,特开发了本古典舞在线交流平台。本古典舞在线交流平台应用Java技术,MYSQL数据库存储数据,基于SpringBoot框架开发。在网站的整个开发过程中,首先对系统进行了......
  • 学习canvas(一些常用api)
    当然,以下是这些常用CanvasAPI的总结,按照Markdown格式编写:常用CanvasAPI总结1.获取绘图上下文constcanvas=document.getElementById('myCanvas');constctx=canvas.getContext('2d');//或'webgl'2.绘制矩形ctx.fillStyle='blue';ctx.fillRect(1......