92国产精品视频_亚洲a级在线观看_国产精品电影观看_国产精品免费观看在线_精品伊人久久97_亚洲人成在线观_尤物九九久久国产精品的特点_成人激情在线播放_成人黄色大片在线免费观看_亚洲成人精品久久久_久久免费视频在线观看_久久精品国产一区_国产一区二区三区18_亚洲欧美中文字幕在线一区_日韩美女中文字幕_日韩视频免费在线

手把手教你利用js給圖片打馬賽克

2021-5-17    前端達人

效果演示

在這里插入圖片描述

Canvas簡介

這個 HTML 元素是為了客戶端矢量圖形而設計的。它自己沒有行為,但卻把一個繪圖 API 展現給客戶端 JavaScript 以使腳本能夠把想繪制的東西都繪制到一塊畫布上。

HTML5 標簽用于繪制圖像(通過腳本,通常是 JavaScript)

不過, 元素本身并沒有繪制能力(它僅僅是圖形的容器) - 您必須使用腳本來完成實際的繪圖任務

getContext() 方法可返回一個對象,該對象提供了用于在畫布上繪圖的方法和屬性

本手冊提供完整的 getContext(“2d”) 對象屬性和方法,可用于在畫布上繪制文本、線條、矩形、圓形等等

標記和 SVG 以及 VML 之間的差異:

標記和 SVG 以及 VML 之間的一個重要的不同是, 有一個基于 JavaScript 的繪圖 API,而 SVG 和 VML 使用一個 XML 文檔來描述繪圖。

這兩種方式在功能上是等同的,任何一種都可以用另一種來模擬。從表面上看,它們很不相同,可是,每一種都有強項和弱點。例如,SVG 繪圖很容易編輯,只要從其描述中移除元素就行。

要從同一圖形的一個 標記中移除元素,往往需要擦掉繪圖重新繪制它。
在這里插入圖片描述

知識點簡介

  • 利用js創建圖片
let img = new Image() //可以給圖片一個鏈接 img.src = 'https://ss0.bdstatic.com/70cFuHSh_Q1YnxGkpoWK1HF6hhy/it/u=826495019,1749283937&fm=26&gp=0.jpg' //或者本地已有圖片的路徑 //img.src = './download.jpg' //添加到HTML中 document.body.appendChild(img)  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8

  • canvas.getContext(“2d”)

語法:
參數 contextID 指定了您想要在畫布上繪制的類型。當前唯一的合法值是 “2d”,它指定了二維繪圖,并且導致這個方法返回一個環境對象,該對象導出一個二維繪圖 API

let ctx = Canvas.getContext(contextID)  
  • 1

  • ctx.drawImage()

JavaScript 語法 1:
在畫布上定位圖像:

context.drawImage(img,x,y);  
  • 1

JavaScript 語法 2:
在畫布上定位圖像,并規定圖像的寬度和高度:

context.drawImage(img,x,y,width,height);  
  • 1

JavaScript 語法 3:
剪切圖像,并在畫布上定位被剪切的部分:

context.drawImage(img,sx,sy,swidth,sheight,x,y,width,height);  
  • 1

  • ctx.getImageData()

JavaScript 語法
getImageData() 方法返回 ImageData 對象,該對象拷貝了畫布指定矩形的像素數據。
對于 ImageData 對象中的每個像素,都存在著四方面的信息,即 RGBA 值:
R - 紅色 (0-255)
G - 綠色 (0-255)
B - 藍色 (0-255)
A - alpha 通道 (0-255; 0 是透明的,255 是完全可見的)
color/alpha 以數組形式存在,并存儲于 ImageData 對象的 data 屬性中

var imgData=context.getImageData(x,y,width,height);  
  • 1

  • ctx.putImageData()

putImageData() 方法將圖像數據(從指定的 ImageData 對象)放回畫布上。

接下來跟著我一步一步做完這個小功能叭~
在這里插入圖片描述

step-by-step

準備好我們的圖片,并添加上我們的方法

<body> <img src="./download.jpg"> <button onclick="addCanvas()">生成Canvas</button> <button onclick="generateImg()">生成圖片</button> </body>  
  • 1
  • 2
  • 3
  • 4
  • 5

在這里插入圖片描述
接下來寫addCanvas方法

function addCanvas() { let bt = document.querySelector('button') let img = new Image(); //1.準備賦值復制一份圖片 img.src = './download.jpg'; img.onload = function() { //2.待圖片加載完成 let width = this.width let height = this.height let canvas = document.createElement('canvas') //3.創建畫布 let ctx = canvas.getContext("2d"); //4.獲得該畫布的內容 canvas.setAttribute('width', width) //5.為了統一,設置畫布的寬高為圖片的寬高 canvas.setAttribute('height', height) ctx.drawImage(this, 0, 0, width, height); //5.在畫布上繪制該圖片 document.body.insertBefore(canvas, bt) //5.把canvas插入到按鈕前面 } }  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21

成功在畫布上得到圖片:
在這里插入圖片描述

嗯,我們已經成功走出了成功的一小步,接下來是干什么呢?…嗯,我們需要利用原生的onmouseuponmousedown事件,代表我們按下鼠標這個過程,那么這兩個事件添加到哪呢?

沒錯,既然我們要在canvas上進行馬賽克操作,那我們必然要給canvas元素添加這兩個事件

考慮到我們創建canvas的過程復雜了一點,我們做一個模塊封裝吧!

function addCanvas() { let bt = document.querySelector('button') let img = new Image(); img.src = './download.jpg'; //這里放自己的圖片 img.onload = function() { let width = this.width let height = this.height let { canvas, ctx } = createCanvasAndCtx(width, height) //對象解構接收canvas和ctx ctx.drawImage(this, 0, 0, width, height); document.body.insertBefore(canvas, bt) } } function createCanvasAndCtx(width, height) { let canvas = document.createElement('canvas') canvas.setAttribute('width', width) canvas.setAttribute('height', height) canvas.setAttribute('onmouseout', 'end()') //修補鼠標不在canvas上離開的補丁 canvas.setAttribute('onmousedown', 'start()') //添加鼠標按下 canvas.setAttribute('onmouseup', 'end()') //添加鼠標彈起 let ctx = canvas.getContext("2d"); return { canvas, ctx } } function start() { let canvas = document.querySelector('canvas') canvas.onmousemove = () => { console.log('你按下了并移動了鼠標') } } function end() { let canvas = document.querySelector('canvas') canvas.onmousemove = null }  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46

測試一下我們的start()end()是否生效了
在這里插入圖片描述
嗯,目前來看,我們的代碼依然如我們所愿的正常工作

接下來的挑戰更加嚴峻,我們需要去獲取像素和處理像素,讓我們再重寫start()函數

 function start() { let img = document.querySelector('img') let canvas = document.querySelector('canvas') let ctx = canvas.getContext("2d"); imgData = ctx.getImageData(0, 0, img.clientWidth, img.clientHeight); canvas.onmousemove = (e) => { let w = imgData.width; //1.獲取圖片寬高 let h = imgData.height; //馬賽克的程度,數字越大越模糊 let num = 10; //獲取鼠標當前所在的像素RGBA let color = getXY(imgData, e.offsetX, e.offsetY); for (let k = 0; k < num; k++) { for (let l = 0; l < num; l++) { //設置imgData上坐標為(e.offsetX + l, e.offsetY + k)的的顏色 setXY(imgData, e.offsetX + l, e.offsetY + k, color); } } //更新canvas數據 ctx.putImageData(imgData, 0, 0); } } //這里為你提供了setXY和getXY兩個函數,如果你有興趣,可以去研究獲取的原理 function setXY(obj, x, y, color) { var w = obj.width; var h = obj.height; var d = obj.data; obj.data[4 * (y * w + x)] = color[0]; obj.data[4 * (y * w + x) + 1] = color[1]; obj.data[4 * (y * w + x) + 2] = color[2]; obj.data[4 * (y * w + x) + 3] = color[3]; } function getXY(obj, x, y) { var w = obj.width; var h = obj.height; var d = obj.data; var color = []; color[0] = obj.data[4 * (y * w + x)]; color[1] = obj.data[4 * (y * w + x) + 1]; color[2] = obj.data[4 * (y * w + x) + 2]; color[3] = obj.data[4 * (y * w + x) + 3]; return color; }  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49

嗯,我們離成功不遠拉,最后一步就是生成圖片

好在canavs給我們提供了直接的方法,可以直接將畫布導出為Base64編碼的圖片:

function generateImg() { let canvas = document.querySelector('canvas') var newImg = new Image(); newImg.src = canvas.toDataURL("image/png"); document.body.insertBefore(newImg, canvas) }  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

最終效果:
在這里插入圖片描述

是不是無比輕松呢~,來看看你手寫的代碼是否和下面一樣叭:

完整代碼

<!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> <body> <img src="./download.jpg"> <button onclick="addCanvas()">生成Canvas</button> <button onclick="generateImg()">生成圖片</button> </body> <script> function addCanvas() { let bt = document.querySelector('button') let img = new Image(); img.src = './download.jpg'; //這里放自己的圖片 img.onload = function() { let width = this.width let height = this.height let { canvas, ctx } = createCanvasAndCtx(width, height) ctx.drawImage(this, 0, 0, width, height); document.body.insertBefore(canvas, bt) } } function createCanvasAndCtx(width, height) { let canvas = document.createElement('canvas') canvas.setAttribute('width', width) canvas.setAttribute('height', height) canvas.setAttribute('onmouseout', 'end()') canvas.setAttribute('onmousedown', 'start()') canvas.setAttribute('onmouseup', 'end()') let ctx = canvas.getContext("2d"); return { canvas, ctx } } function start() { let img = document.querySelector('img') let canvas = document.querySelector('canvas') let ctx = canvas.getContext("2d"); imgData = ctx.getImageData(0, 0, img.clientWidth, img.clientHeight); canvas.onmousemove = (e) => { let w = imgData.width; //1.獲取圖片寬高 let h = imgData.height; //馬賽克的程度,數字越大越模糊 let num = 10; //獲取鼠標當前所在的像素RGBA let color = getXY(imgData, e.offsetX, e.offsetY); for (let k = 0; k < num; k++) { for (let l = 0; l < num; l++) { //設置imgData上坐標為(e.offsetX + l, e.offsetY + k)的的顏色 setXY(imgData, e.offsetX + l, e.offsetY + k, color); } } //更新canvas數據 ctx.putImageData(imgData, 0, 0); } } function generateImg() { let canvas = document.querySelector('canvas') var newImg = new Image(); newImg.src = canvas.toDataURL("image/png"); document.body.insertBefore(newImg, canvas) } function setXY(obj, x, y, color) { var w = obj.width; var h = obj.height; var d = obj.data; obj.data[4 * (y * w + x)] = color[0]; obj.data[4 * (y * w + x) + 1] = color[1]; obj.data[4 * (y * w + x) + 2] = color[2]; obj.data[4 * (y * w + x) + 3] = color[3]; } function getXY(obj, x, y) { var w = obj.width; var h = obj.height; var d = obj.data; var color = []; color[0] = obj.data[4 * (y * w + x)]; color[1] = obj.data[4 * (y * w + x) + 1]; color[2] = obj.data[4 * (y * w + x) + 2]; color[3] = obj.data[4 * (y * w + x) + 3]; return color; } function end() { let canvas = document.querySelector('canvas') canvas.onmousemove = null } </script> </body> </html>  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • 101
  • 102
  • 103
  • 104
  • 105
  • 106
  • 107
  • 108
  • 109
  • 110
  • 111
  • 112
  • 113
  • 114
  • 115
  • 116

當然,你可以做更多創作,比如上面打的馬賽克是正方形的,你可以利用你的數學知識讓其變為圓形,以圓心為鼠標中心擴散

你也可以選擇完善一些過程,例如馬賽克位置打錯了,可以選擇將畫布清空然后重新開始~
或者做一些善后處理,導出圖片后隱藏canvas畫布
點個贊吧

藍藍設計建立了UI設計分享群,每天會分享國內外的一些優秀設計,如果有興趣的話,可以進入一起成長學習,請掃碼藍小助,報下信息,藍小助會請您入群。歡迎您加入噢~~希望得到建議咨詢、商務合作,也請與我們聯系。

截屏2021-05-13 上午11.41.03.png


文章來源:csdn   作者: VGtime

分享此文一切功德,皆悉回向給文章原作者及眾讀者.
免責聲明:藍藍設計尊重原作者,文章的版權歸原作者。如涉及版權問題,請及時與我們取得聯系,我們立即更正或刪除。

藍藍設計www.skdbbs.com )是一家專注而深入的界面設計公司,為期望卓越的國內外企業提供卓越的UI界面設計、BS界面設計 、 cs界面設計 、 ipad界面設計 、 包裝設計 、 圖標定制 、 用戶體驗 、交互設計、 網站建設 、平面設計服務


日歷

鏈接

個人資料

藍藍設計的小編 http://www.skdbbs.com

存檔

92国产精品视频_亚洲a级在线观看_国产精品电影观看_国产精品免费观看在线_精品伊人久久97_亚洲人成在线观_尤物九九久久国产精品的特点_成人激情在线播放_成人黄色大片在线免费观看_亚洲成人精品久久久_久久免费视频在线观看_久久精品国产一区_国产一区二区三区18_亚洲欧美中文字幕在线一区_日韩美女中文字幕_日韩视频免费在线
色综合男人天堂| 久久久久久一区二区三区| 国产剧情久久久久久| 中文字幕亚洲一区二区三区五十路| 成人av色在线观看| 亚洲国产欧美在线成人app| 精品久久久久久中文字幕大豆网| 国产精品久久久精品| 97超级碰在线看视频免费在线看| 国产精品入口尤物| 中文字幕免费精品一区| 日韩欧美在线网址| 亚洲日本成人女熟在线观看| 夜夜嗨av色综合久久久综合网| 精品国产一区久久久| 大胆人体色综合| 国产精品视频导航| 久久久www成人免费精品| 97免费视频在线播放| 色婷婷成人综合| 最近中文字幕mv在线一区二区三区四区| 97国产精品免费视频| 欧美高清视频在线观看| 欧美激情精品久久久久久大尺度| 在线观看日韩www视频免费| 亚洲性夜色噜噜噜7777| 91精品国产91久久久久久吃药| 亚洲精品电影网站| 国产精品亚洲自拍| 日韩免费高清在线观看| 亚洲欧洲美洲在线综合| 亚洲第一精品电影| 精品国产999| 在线成人激情黄色| 91免费看国产| 97热在线精品视频在线观看| 久久理论片午夜琪琪电影网| 久久久久久久久久亚洲| 永久免费精品影视网站| 超碰97人人做人人爱少妇| 成人黄色在线观看| 精品久久久在线观看| 欧美激情一二三| 国产精品影院在线观看| 国产精品劲爆视频| 国产乱人伦真实精品视频| 亚洲欧洲日产国码av系列天堂| 欧美性xxxx| 亚洲天堂男人的天堂| 欧美激情精品久久久久久变态| 国产精品久久婷婷六月丁香| 欧美视频免费在线观看| 欧美精品在线免费| 成人中文字幕+乱码+中文字幕| 亚洲天堂av在线免费观看| 欧美精品在线极品| 日韩有码视频在线| 激情亚洲一区二区三区四区| 成人中文字幕在线观看| 国产色婷婷国产综合在线理论片a| 欧美www在线| 日韩中文字幕在线播放| 国产视频福利一区| 国产成人精品午夜| 久久99国产精品久久久久久久久| 在线精品国产成人综合| 毛片精品免费在线观看| 91久久久久久久久久久| 日本久久中文字幕| 国产精品一区二区三区久久| 日本欧美国产在线| 45www国产精品网站| 最近2019中文字幕第三页视频| 亚洲国产精品高清久久久| 国内精品久久久久久影视8| 亚洲片在线观看| 久久精品精品电影网| 欧美日韩在线观看视频小说| 精品一区电影国产| 国产精品电影一区| 久久在精品线影院精品国产| 精品久久久久久久久久| 4k岛国日韩精品**专区| 久久深夜福利免费观看| 国产精品久久久久久久久久ktv| 精品久久久免费| 成人写真福利网| 欧美一区二区视频97| 欧美激情欧美激情| 亚洲精品美女久久久| 日韩精品视频在线| 性色av一区二区咪爱| 日韩av片永久免费网站| 国产极品jizzhd欧美| 久久亚洲一区二区三区四区五区高| 国语自产在线不卡| 欧美大片免费观看在线观看网站推荐| 亚洲女人被黑人巨大进入| 国产精品国产三级国产专播精品人| 日韩欧美视频一区二区三区| 91久久精品国产91久久| 久久99精品久久久久久青青91| 久久久久久网址| 九九九久久久久久| 欧美日韩在线看| 国产91色在线播放| 欧美肥老妇视频| 日韩中文字幕在线观看| 中文字幕在线观看日韩| 97国产一区二区精品久久呦| 亚洲天堂日韩电影| 91香蕉嫩草神马影院在线观看| 91av免费观看91av精品在线| 欧美视频13p| 国内成人精品视频| 亚洲欧美中文字幕| 欧美日韩成人精品| 亚洲欧美激情在线视频| 亚洲欧美日韩久久久久久| 777国产偷窥盗摄精品视频| 欧美激情在线一区| 久久久人成影片一区二区三区| 成人av在线天堂| 久久99国产精品久久久久久久久| 亚洲精品一区中文| 欧美大片大片在线播放| 国产精品扒开腿做爽爽爽的视频| 伊人精品在线观看| 国产成人高潮免费观看精品| 欧美高跟鞋交xxxxhd| 久久精品国产亚洲精品2020| 亚洲精品欧美日韩| 亚洲成色999久久网站| 国产精品日韩电影| 97avcom| 欧美成人精品一区二区| 91精品久久久久久久久久久| 久久久精品视频在线观看| 国内偷自视频区视频综合| 国产亚洲精品久久久久久777| 日韩欧美国产视频| 国产精品都在这里| 久久视频免费在线播放| 日韩免费高清在线观看| 国产亚洲一级高清| 最近2019免费中文字幕视频三| 福利精品视频在线| 亚洲日韩欧美视频一区| 91国偷自产一区二区三区的观看方式| 日韩大片在线观看视频| 亚洲japanese制服美女| 中文字幕无线精品亚洲乱码一区| 2019中文字幕在线| 久久男人资源视频| 国产精品人成电影在线观看| 狠狠色香婷婷久久亚洲精品| 中文字幕精品在线视频| 国内精品久久久久影院优| 日本三级韩国三级久久| 国产亚洲精品久久| 亚洲人成亚洲人成在线观看| 51ⅴ精品国产91久久久久久| 久久免费精品日本久久中文字幕| 国产91精品久久久久久|