亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > HTML > 正文

canvas像素畫板的實現代碼

2024-08-26 00:21:09
字體:
來源:轉載
供稿:網友

最近項目上要實現一個類似像素風格的畫板,可以像素小格子可以擦除,框選變色,可以擦出各種圖形,這樣一個小項目看似簡單,包含的東西還真不少。

繪制像素格子

我們先定義像素格子類

Pixel = function (option) {    this.x = option.x;    this.y = option.y;    this.shape = option.shape;    this.size = option.size || 8;}

x和y表示中心點坐標,一開始我是這么做的,先定義路徑

createPath: function (ctx) {			if (this.shape === 'circle') {				this.createCircle(ctx);			} else if (this.shape === 'rect') {				this.createRect(ctx);			} else {				this.createCircle(ctx);			}		},		createCircle: function (ctx) {			var radius = this.size / 2;			ctx.arc(this.x,this.y,radius,0,Math.PI*2);		},		createRect: function (ctx) {			var points = this.getPoints();            points.forEach(function (point, i) {                ctx[i == 0 ? 'moveTo' : 'lineTo'](point.x, point.y);            })            ctx.lineTo(points[0].x, points[0].y);		},

像素網格支持圓形和矩形,路徑定義好后,然后進行繪制

draw: function (ctx) {			ctx.save();			ctx.lineWidth=this.lineWidth;			ctx.strokeStyle=this.strokeStyle;			ctx.fillStyle=this.fillStyle;			ctx.beginPath();			this.createPath(ctx);			ctx.stroke();			if(this.isFill){ctx.fill();}			ctx.restore();		}

然后通過循環批量創建像素網格:

for (var i = stepX + .5; i < canvas.width; i+=stepX) {		for (var j = stepY + .5; j < canvas.height; j+=stepY) {			var pixel = new Pixel({				x: i,				y: j,				shape: 'circle'			})			box.push(pixel);			pixel.draw(ctx);		}	}

這樣做看似完美,然而有一個巨大斃命,每畫一個像素都回繪制到上下文中,每一次都在改變canvas的狀態,這樣做會導致渲染性能太差,因為像素點很多,如果畫布比較大,性能很是令人堪憂,并且畫板上面還有一些操作,如此頻繁改變canvas的狀態是不合適的。

canvas,像素畫板,代碼

因此,正確的做法是:我們應該定義好所有的路徑,最好在一次性的批量繪制到canvas中; 

//定義像素的位置for (var i = stepX + .5; i < canvas.width; i+=stepX) {		for (var j = stepY + .5; j < canvas.height; j+=stepY) {			var pixel = new Pixel({				x: i,				y: j,				shape: 'circle'			})			box.push(pixel);		}	}//批量繪制	console.time('time');	ctx.beginPath();	for (var c = 0; c < box.length; c++) {		var circle = box[c];		ctx.moveTo(circle.x + 3, circle.y);		circle.createPath(ctx);	}	ctx.closePath();	ctx.stroke();		console.timeEnd('time');

canvas,像素畫板,代碼

可以看到這個渲染效率很快,盡可能少的改變canvas的狀態,因為每改變一次上下文的狀態,canvas都會重新繪制,這種狀態是全局的狀態。

像素網格交互

項目的需求是,在畫布上鼠標按下移動,可以擦除像素點,這里面包含兩個知識點,一個是如何獲取鼠標移動路徑上的像素網格,二是性能問題,因為我們這個需求的要求是繪制八萬個點,不說別的,光是循環都得幾十上百毫秒,何況還要繪制渲染。我們先來看第一個問題:

獲取鼠標移動路徑下的網格

看到這個問題,我們很容易想到,寫個函數,通過鼠標的位置獲取下所在的位置包含那個網格,然后每次移動都重新更新位置計算,這樣看是可以完成需求,但是如果鼠標移動過快,是無法做到,每個點的位置都可以計算到的,效果會不連貫。我們換種思路,鼠標經過的路徑,我們可以很明確的知道起始和終點,我們把整個繪制路徑想象成一段段的線段,那么問題就變成,線段與原相交的一個算法了,線段就是畫筆的粗細,線段經過的路徑就是鼠標運動的路徑,與之相交的圓就是需要變化樣式的網格。轉換成代碼就是如下:

function sqr(x) { return x * x }    function dist2(p1, p2) { return sqr(p1.x - p2.x) + sqr(p1.y - p2.y) }    function distToSegmentSquared(p, v, w) {        var l2 = dist2(v, w);        if (l2 == 0) return dist2(p, v);        var t = ((p.x - v.x) * (w.x - v.x) + (p.y - v.y) * (w.y - v.y)) / l2;        if (t < 0) return dist2(p, v);        if (t > 1) return dist2(p, w);        return dist2(p, {            x: v.x + t * (w.x - v.x),            y: v.y + t * (w.y - v.y)        });    }	/**	 * @description 計算線段與圓是否相交	 * @param {x: num, y: num} p 圓心點	 * @param {x: num, y: num} v 線段起始點	 * @param {x: num, y: num} w 線段終點	 */    function distToSegment(p, v, w) {        var offset = pathHeight;        var minX = Math.min(v.x, w.x) - offset;        var maxX = Math.max(v.x, w.x) + offset;        var minY = Math.min(v.y, w.y) - offset;        var maxY = Math.max(v.y, w.y) + offset;        if ((p.x < minX || p.x > maxX) && (p.y < minY || p.y > maxY)) {            return Number.MAX_VALUE;        }        return Math.sqrt(distToSegmentSquared(p, v, w));    }

具體邏輯就不詳述,各位看官可以自行看代碼。然后通過獲取到的相交網格的,然后刪除box里面的數據,重新render一遍,就可以看到效果了。

canvas,像素畫板,代碼

同樣的道理,我們可以做成染色效果,那么我們就可能實現一個canvas像素畫板的小demo了。不過做成染色效果就必須使用第一種繪制方法了,每個像素必須是一個對象,因為每個對象的狀態是獨立的,不過這個不用擔心性能,像素點不多,基本不會有卡頓感。實現效果大體如下:

canvas,像素畫板,代碼

最近又有點懶,先這樣了,后面有時間添加一個上傳圖片,圖片像素畫的功能和導出功能。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。


注:相關教程知識閱讀請移步到HTML教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲人成在线一二| 亚洲人成欧美中文字幕| 日韩亚洲第一页| 亚洲精品视频免费在线观看| 亚洲影视九九影院在线观看| www亚洲精品| 亚洲欧洲av一区二区| 日本欧美国产在线| 亚洲精品97久久| 欧美午夜精品久久久久久浪潮| 欧美在线视频网站| 中日韩午夜理伦电影免费| 欧美成人免费在线视频| 国产欧美精品在线| 97在线观看免费高清| 国产福利视频一区二区| 欧美激情精品久久久久久| 亚洲一区亚洲二区亚洲三区| 91香蕉亚洲精品| 亚洲美女黄色片| 欧美成人午夜剧场免费观看| 欧美精品国产精品日韩精品| 蜜臀久久99精品久久久久久宅男| 日韩欧美中文免费| 自拍亚洲一区欧美另类| 国产精品欧美一区二区三区奶水| 5252色成人免费视频| 亚洲乱码一区二区| 国产精品人成电影在线观看| 亚洲色图色老头| 97精品伊人久久久大香线蕉| 亚洲精品456在线播放狼人| 欧美xxxwww| 亚洲成人网在线观看| 日韩三级影视基地| 国产精品aaa| 欧美激情视频免费观看| 国产午夜精品一区二区三区| 欧美在线视频一区二区| 欧美激情精品在线| 国产精品电影在线观看| 日韩在线观看免费全集电视剧网站| 热99久久精品| 亚洲最大成人免费视频| 日韩av免费看| 欧美激情奇米色| 亚洲欧美在线x视频| 国产精品日韩在线观看| 在线视频亚洲欧美| 精品久久久久久亚洲国产300| 欧洲精品在线视频| 欧美电影电视剧在线观看| 精品中文字幕久久久久久| 国产成人精品久久二区二区91| 日韩免费观看网站| 欧美成人中文字幕在线| 亚洲综合在线做性| 国产精品福利在线观看| 亚洲人成啪啪网站| 在线视频一区二区| 亚洲一级黄色片| 日韩欧美在线视频日韩欧美在线视频| 亚洲男人的天堂在线| 精品小视频在线| www欧美xxxx| 18一19gay欧美视频网站| 欧美日韩国产成人| 精品人伦一区二区三区蜜桃网站| 欧美做受高潮1| 亚洲日本欧美日韩高观看| 91在线高清视频| 亚洲аv电影天堂网| 国产精品com| 成人激情电影一区二区| 最近2019免费中文字幕视频三| 97国产精品视频人人做人人爱| 精品毛片网大全| 色综合伊人色综合网| 色综合伊人色综合网站| 欧美性猛交xxxx偷拍洗澡| 久热爱精品视频线路一| 国产精品户外野外| 国产精品日韩在线一区| 自拍偷拍亚洲在线| 精品国产依人香蕉在线精品| 久久久国产精彩视频美女艺术照福利| 美女久久久久久久| 亚洲综合一区二区不卡| 国产精品视频26uuu| 久久久97精品| 欧美怡红院视频一区二区三区| 亚洲精品suv精品一区二区| 一区二区av在线| 久久久亚洲福利精品午夜| 国产精品久久久久久网站| 日韩毛片中文字幕| 国产成人啪精品视频免费网| 日韩av快播网址| 亚洲精品自拍视频| 国产在线日韩在线| 久久精品国产亚洲| 欧美精品生活片| 日韩av片电影专区| 久久人体大胆视频| 韩国三级电影久久久久久| 亚洲精品资源在线| 久久6免费高清热精品| 国内外成人免费激情在线视频网站| 国产精品久久久一区| 国产精品高清免费在线观看| 国产精品电影网站| 日韩精品中文字幕在线观看| 在线免费观看羞羞视频一区二区| 欧美精品videosex牲欧美| 尤物tv国产一区| 欧美大学生性色视频| 国产精品一区二区电影| 欧美疯狂xxxx大交乱88av| 97国产真实伦对白精彩视频8| 欧美激情影音先锋| 欧美激情亚洲国产| 国产日韩换脸av一区在线观看| 亚洲高清av在线| 一区二区三区亚洲| 亚洲男人av电影| 欧美亚洲在线播放| 精品精品国产国产自在线| 国产视频久久久久| 欧美综合在线第二页| 精品国产一区二区三区四区在线观看| 91av免费观看91av精品在线| 国产精品成人免费电影| 亚洲在线视频观看| 久久青草福利网站| 最近中文字幕mv在线一区二区三区四区| 国产精品99一区| 91av中文字幕| 亚洲精品www| 日韩av有码在线| 日韩av成人在线观看| 日韩精品中文字幕在线播放| 超碰97人人做人人爱少妇| 国产成人精品网站| 国内揄拍国内精品少妇国语| 成年无码av片在线| 一区二区成人av| 2019中文字幕全在线观看| 欧美精品免费播放| 久久精品欧美视频| 国产日韩欧美黄色| 久久综合亚洲社区| 国产亚洲精品日韩| 精品久久久91| 色av吧综合网| 亚洲精选一区二区| 欧美性视频网站| 久久久精品视频成人| 国产亚洲一区二区在线| 日韩成人在线观看| 一区二区三区视频观看| 乱亲女秽乱长久久久| 国内精品视频一区| 久久综合久久美利坚合众国| 性欧美视频videos6一9|