真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

使用canvas怎么實(shí)現(xiàn)一個(gè)像素畫板

本篇文章給大家分享的是有關(guān)使用canvas怎么實(shí)現(xiàn)一個(gè)像素畫板,小編覺得挺實(shí)用的,因此分享給大家學(xué)習(xí),希望大家閱讀完這篇文章后可以有所收獲,話不多說(shuō),跟著小編一起來(lái)看看吧。

為定南等地區(qū)用戶提供了全套網(wǎng)頁(yè)設(shè)計(jì)制作服務(wù),及定南網(wǎng)站建設(shè)行業(yè)解決方案。主營(yíng)業(yè)務(wù)為成都網(wǎng)站設(shè)計(jì)、成都網(wǎng)站建設(shè)、定南網(wǎng)站設(shè)計(jì),以傳統(tǒng)方式定制建設(shè)網(wǎng)站,并提供域名空間備案等一條龍服務(wù),秉承以專業(yè)、用心的態(tài)度為用戶提供真誠(chéng)的服務(wù)。我們深信只要達(dá)到每一位用戶的要求,就會(huì)得到認(rèn)可,從而選擇與我們長(zhǎng)期合作。這樣,我們也可以走得更遠(yuǎn)!

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

x和y表示中心點(diǎn)坐標(biāo),一開始我是這么做的,先定義路徑

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);
		},

像素網(wǎng)格支持圓形和矩形,路徑定義好后,然后進(jìn)行繪制

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();
		}

然后通過(guò)循環(huán)批量創(chuàng)建像素網(wǎng)格:

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);
		}
	}

這樣做看似完美,然而有一個(gè)巨大斃命,每畫一個(gè)像素都回繪制到上下文中,每一次都在改變canvas的狀態(tài),這樣做會(huì)導(dǎo)致渲染性能太差,因?yàn)橄袼攸c(diǎn)很多,如果畫布比較大,性能很是令人堪憂,并且畫板上面還有一些操作,如此頻繁改變canvas的狀態(tài)是不合適的。

使用canvas怎么實(shí)現(xiàn)一個(gè)像素畫板

因此,正確的做法是:我們應(yīng)該定義好所有的路徑,最好在一次性的批量繪制到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怎么實(shí)現(xiàn)一個(gè)像素畫板

可以看到這個(gè)渲染效率很快,盡可能少的改變canvas的狀態(tài),因?yàn)槊扛淖円淮紊舷挛牡臓顟B(tài),canvas都會(huì)重新繪制,這種狀態(tài)是全局的狀態(tài)。

像素網(wǎng)格交互

項(xiàng)目的需求是,在畫布上鼠標(biāo)按下移動(dòng),可以擦除像素點(diǎn),這里面包含兩個(gè)知識(shí)點(diǎn),一個(gè)是如何獲取鼠標(biāo)移動(dòng)路徑上的像素網(wǎng)格,二是性能問(wèn)題,因?yàn)槲覀冞@個(gè)需求的要求是繪制八萬(wàn)個(gè)點(diǎn),不說(shuō)別的,光是循環(huán)都得幾十上百毫秒,何況還要繪制渲染。我們先來(lái)看第一個(gè)問(wèn)題:

獲取鼠標(biāo)移動(dòng)路徑下的網(wǎng)格

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

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 計(jì)算線段與圓是否相交
	 * @param {x: num, y: num} p 圓心點(diǎn)
	 * @param {x: num, y: num} v 線段起始點(diǎn)
	 * @param {x: num, y: num} w 線段終點(diǎn)
	 */
    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));
    }

以上就是使用canvas怎么實(shí)現(xiàn)一個(gè)像素畫板,小編相信有部分知識(shí)點(diǎn)可能是我們?nèi)粘9ぷ鲿?huì)見到或用到的。希望你能通過(guò)這篇文章學(xué)到更多知識(shí)。更多詳情敬請(qǐng)關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。


當(dāng)前標(biāo)題:使用canvas怎么實(shí)現(xiàn)一個(gè)像素畫板
當(dāng)前鏈接:http://weahome.cn/article/pcheie.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部