摘要:第二步,消除涂鴉鋸齒的辦法簡(jiǎn)單的繪制和圖片保存完成了,但是在這種情況下,線條會(huì)有很明顯的鋸齒靈魂畫手來(lái)了。在經(jīng)過(guò)搜索查閱之后,發(fā)現(xiàn)有一個(gè)繪制辦法可以降低鋸齒的問(wèn)題。橡皮擦的原理是,將橡皮繪制的路徑覆蓋到原來(lái)的畫筆上。
第一步,我們先實(shí)現(xiàn)簡(jiǎn)單的繪制,并且在繪制之后將圖片保存到本地
var canvas = document.getElementById("canvas"), ctx = canvas.getContext("2d"), iptColor = document.getElementById("iptColor"),//畫筆顏色 iptSize = document.getElementById("iptSize"),//畫筆大小 btnClear = document.getElementById("btnClear"), //清除按鈕 btnSave = document.getElementById("btnSave"),//保存按鈕 canvasWidth = 800, canvasHeight = 600; canvas.setAttribute("width",canvasWidth); canvas.setAttribute("height",canvasHeight); iptSize.oninput = function() { document.querySelector(".txt-size").innerHTML = iptSize.value } canvas.addEventListener("mousedown",function(e){ var e = e || window.event; ctx.lineWidth = iptSize.value; ctx.strokeStyle = iptColor.value; ctx.lineCap = "round"; ctx.lineJoin = "round"; ctx.beginPath(); ctx.moveTo(e.clientX - canvas.offsetLeft,e.clientY - canvas.offsetTop); //線條起始位置 document.onmousemove = function(e) { var e = e || window.event; ctx.lineTo(e.clientX - canvas.offsetLeft,e.clientY - canvas.offsetTop); ctx.stroke();//繪制線條 }; canvas.onmouseup = function() { document.onmousemove = null; document.onmouseup = null; }; }) //清除畫布 btnClear.addEventListener("click",function(){ ctx.clearRect(0, 0, canvasWidth, canvasHeight); }) //保存圖片到本地 btnSave.addEventListener("click",function(){ var imgData = canvas.toDataURL("image/png").replace("image/png", "image/octet-stream"); var filename = "圖片.png"; saveFile(imgData,filename) }) var saveFile = function(data, filename) { var save_link = document.createElementNS("http://www.w3.org/1999/xhtml", "a"); save_link.href = data; save_link.download = filename; var event = document.createEvent("MouseEvents"); event.initMouseEvent("click", true, false, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null); save_link.dispatchEvent(event); };
這樣看起來(lái)雖然說(shuō)好像沒(méi)問(wèn)題,但是這一句有一些問(wèn)題:
e.clientX - canvas.offsetLeft,e.clientY - canvas.offsetTop
如果頁(yè)面比較高,產(chǎn)生了滾動(dòng)條,當(dāng)滾動(dòng)條向上滑的時(shí)候,繪圖的位置就不準(zhǔn)確了,示例如:https://codepen.io/jianxiujiu...。
所以此處應(yīng)該使用:
e.clientX - canvas.getBoundingClientRect().left,e.clientY - canvas.getBoundingClientRect().top
來(lái)替換。
第二步,消除涂鴉鋸齒的辦法簡(jiǎn)單的繪制和圖片保存完成了,但是在這種情況下,線條會(huì)有很明顯的鋸齒(靈魂畫手來(lái)了)。
有一個(gè)簡(jiǎn)單粗暴的方法,將線條的邊緣進(jìn)行模糊:
ctx.shadowBlur = 1; ctx.shadowColor = iptColor.value;
示例:https://codepen.io/jianxiujiu...
加上邊緣模糊之后,線條明顯柔和了許多,但是還是有一個(gè)問(wèn)題,就是在畫筆收筆的地方,線條會(huì)變?。`魂畫手又來(lái)了)。
在經(jīng)過(guò)搜索查閱之后,發(fā)現(xiàn)有一個(gè)繪制辦法可以降低鋸齒的問(wèn)題。
原理如圖:
繪制一個(gè)圓點(diǎn),在下一個(gè)圓點(diǎn)之間,繪制一個(gè)矩形進(jìn)行填充。這樣繪制出來(lái)的線條的銜接會(huì)是圓滑的。
var canvas = document.getElementById("canvas"), ctx = canvas.getContext("2d"), btnClear = document.getElementById("btnClear"), btnSave = document.getElementById("btnSave"), iptColor = document.getElementById("iptColor"), iptSize = document.getElementById("iptSize"), canvasWidth = 800, canvasHeight = 600; canvas.setAttribute("width",canvasWidth); canvas.setAttribute("height",canvasHeight); iptSize.oninput = function() { document.querySelector(".txt-size").innerHTML = iptSize.value } canvas.addEventListener("mousedown",function(e){ var e = e || window.event; var x1 = e.clientX - canvas.getBoundingClientRect().left, y1 = e.clientY - canvas.getBoundingClientRect().top; lineSize = iptSize.value; lineColor = iptColor.value; ctx.beginPath(); ctx.fillStyle = lineColor; ctx.lineCap = "round"; ctx.lineJoin = "round"; ctx.arc(x1, y1, lineSize, 0, 2 * Math.PI); ctx.fill(); ctx.closePath(); document.onmousemove = function(e) { var e = e || window.event; var x2 = e.clientX - canvas.getBoundingClientRect().left, y2 = e.clientY - canvas.getBoundingClientRect().top; var asin = lineSize * Math.sin(Math.atan((y2 - y1) / (x2 - x1))); var acos = lineSize * Math.cos(Math.atan((y2 - y1) / (x2 - x1))); //分別獲取矩形的四個(gè)點(diǎn)的xy軸位置 var x3 = x1 + asin; var y3 = y1 - acos; var x4 = x1 - asin; var y4 = y1 + acos; var x5 = x2 + asin; var y5 = y2 - acos; var x6 = x2 - asin; var y6 = y2 + acos; ctx.beginPath(); ctx.fillStyle = lineColor; ctx.arc(x2, y2, lineSize, 0, 2 * Math.PI); ctx.fill(); ctx.closePath(); ctx.beginPath(); ctx.fillStyle = lineColor; ctx.moveTo(x3, y3); ctx.lineTo(x5, y5); ctx.lineTo(x6, y6); ctx.lineTo(x4, y4); ctx.fill(); ctx.closePath(); x1 = x2; y1 = y2; }; canvas.onmouseup = function() { document.onmousemove = null; document.onmouseup = null; }; }) btnClear.addEventListener("click",function(){ ctx.clearRect(0, 0, canvasWidth, canvasHeight); }) btnSave.addEventListener("click",function(){ var imgData = canvas.toDataURL("image/png").replace("image/png", "image/octet-stream"); var filename = "圖片.png"; saveFile(imgData,filename) }) var saveFile = function(data, filename) { var save_link = document.createElementNS("http://www.w3.org/1999/xhtml", "a"); save_link.href = data; save_link.download = filename; var event = document.createEvent("MouseEvents"); event.initMouseEvent("click", true, false, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null); save_link.dispatchEvent(event); };
DEMO:
https://codepen.io/jianxiujiu...
用這個(gè)方法,我們還可以加上橡皮擦。
橡皮擦的原理是,將橡皮繪制的路徑覆蓋到原來(lái)的畫筆上。
這里我們將用到canvas的一個(gè)屬性globalCompositeOperation,詳解參照:
http://www.w3school.com.cn/ta...
var canvas = document.getElementById("canvas"), ctx = canvas.getContext("2d"), btnClear = document.getElementById("btnClear"), btnSave = document.getElementById("btnSave"), btnRestore = document.getElementById("btnRestore"), iptColor = document.getElementById("iptColor"), iptSize = document.getElementById("iptSize"), eraser = document.getElementById("eraser") pen = document.getElementById("pen"), canvasWidth = 800, canvasHeight = 600, isClear = 0; canvas.setAttribute("width",canvasWidth); canvas.setAttribute("height",canvasHeight); iptSize.oninput = function() { document.querySelector(".txt-size").innerHTML = iptSize.value } eraser.addEventListener("click",function(){ isClear = 1; this.classList.add("cho"); pen.classList.remove("cho") }) pen.addEventListener("click",function(){ isClear = 0; this.classList.add("cho"); eraser.classList.remove("cho") }) canvas.addEventListener("mousedown",function(e){ var e = e || window.event; var x1 = e.clientX - canvas.getBoundingClientRect().left, y1 = e.clientY - canvas.getBoundingClientRect().top; lineSize = iptSize.value; if(isClear == 0){ lineColor = iptColor.value; ctx.beginPath(); ctx.fillStyle = lineColor; ctx.arc(x1, y1, lineSize, 0, 2 * Math.PI); ctx.fill(); ctx.closePath(); document.onmousemove = function(e) { draw(); }; btnClear.classList.remove("dis"); btnSave.classList.remove("dis"); }else{ ctx.strokeStyle = "rgba(250,250,250,0)"; document.onmousemove = function() { ctx.globalCompositeOperation = "destination-out"; draw(); ctx.globalCompositeOperation = "source-over" } } canvas.onmouseup = function() { document.onmousemove = null; document.onmouseup = null; }; draw = function(e){ var e = e || window.event, x2 = e.clientX - canvas.getBoundingClientRect().left, y2 = e.clientY - canvas.getBoundingClientRect().top, asin = lineSize * Math.sin(Math.atan((y2 - y1) / (x2 - x1))), acos = lineSize * Math.cos(Math.atan((y2 - y1) / (x2 - x1))), x3 = x1 + asin, y3 = y1 - acos, x4 = x1 - asin, y4 = y1 + acos, x5 = x2 + asin, y5 = y2 - acos, x6 = x2 - asin, y6 = y2 + acos; ctx.beginPath(); ctx.arc(x2, y2, lineSize, 0, 2 * Math.PI); ctx.fill(); ctx.closePath(); ctx.beginPath(); ctx.fillStyle = lineColor; ctx.moveTo(x3, y3); ctx.lineTo(x5, y5); ctx.lineTo(x6, y6); ctx.lineTo(x4, y4); ctx.fill(); ctx.closePath(); x1 = x2; y1 = y2; } }) // 清除 btnClear.addEventListener("click",function(){ ctx.clearRect(0, 0, canvasWidth, canvasHeight); this.classList.add("dis"); btnSave.classList.add("dis"); }) //保存圖片 btnSave.addEventListener("click",function(){ var imgData = canvas.toDataURL("image/png").replace("image/png", "image/octet-stream"); var filename = "圖片.png"; saveFile(imgData,filename); this.classList.add("dis"); }) var saveFile = function(data, filename) { var save_link = document.createElementNS("http://www.w3.org/1999/xhtml", "a"); save_link.href = data; save_link.download = filename; var event = document.createEvent("MouseEvents"); event.initMouseEvent("click", true, false, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null); save_link.dispatchEvent(event); };
地址:https://codepen.io/jianxiujiu...
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/89025.html
摘要:方案背景需求需要對(duì)圖片進(jìn)行標(biāo)注,導(dǎo)出圖片。對(duì)應(yīng)方案用實(shí)現(xiàn)涂鴉圓形矩形的繪制,最終生成圖片編碼用于上傳大量圖片批量上傳很耗時(shí)間,為了提高用戶體驗(yàn),改為只實(shí)現(xiàn)圓形矩形繪制,最終保存成坐標(biāo),下次顯示時(shí)根據(jù)坐標(biāo)再繪制。 方案背景 需求 需要對(duì)圖片進(jìn)行標(biāo)注,導(dǎo)出圖片。 需要標(biāo)注N多圖片最后同時(shí)保存。 需要根據(jù)多邊形區(qū)域數(shù)據(jù)(區(qū)域、顏色、名稱)標(biāo)注。 對(duì)應(yīng)方案 用canvas實(shí)現(xiàn)涂鴉、圓形、...
摘要:撤銷清空等操作撤銷前進(jìn)清空清空前后數(shù)據(jù)恢復(fù)到默認(rèn)數(shù)據(jù)地址查看代碼 效果展示 showImg(https://segmentfault.com/img/bVHJXf?w=1550&h=846); Canvas API簡(jiǎn)介 調(diào)用方法 getImageData() 返回ImageData對(duì)象,該對(duì)象為畫布上指定的矩形復(fù)制像素?cái)?shù)據(jù) putImageData() 把圖像數(shù)據(jù)(從指定的 Imag...
摘要:開啟掃描時(shí)需要設(shè)備處于配網(wǎng)狀態(tài)一分類配網(wǎng)子設(shè)備可以通過(guò)使用手機(jī)藍(lán)牙直接掃描獲取設(shè)備到設(shè)備基礎(chǔ)信息,再使用配網(wǎng)接口實(shí)現(xiàn)設(shè)備的本地配網(wǎng)。 ? (一)分類 ? (二)設(shè)備配置 ? (三)設(shè)備管理 ? ? 設(shè)備管理,大體分為兩類,mesh 和 其他 ? ? 獲取設(shè)備列表,給涂鴉sdk發(fā)送當(dāng)前房間id...
閱讀 2240·2019-08-30 10:51
閱讀 793·2019-08-30 10:50
閱讀 1475·2019-08-30 10:49
閱讀 3139·2019-08-26 13:55
閱讀 1605·2019-08-26 11:39
閱讀 3419·2019-08-26 11:34
閱讀 1949·2019-08-23 18:30
閱讀 3387·2019-08-23 18:22