成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

小記-用canvas完成圖像液化(向前變形)過程

Cheng_Gang / 639人閱讀

摘要:前幾天由于團隊需要,折騰了一下圖像液化的處理過程?,F(xiàn)在來整理一下思路,做個記錄。用到公式如下,網(wǎng)上拿來的話不多說,上代碼本來想盡量寫出點逼格。。。后來發(fā)現(xiàn)怎么寫也還是幾個搞定,就那樣了。然后再用的把轉(zhuǎn)換后的輸出到中效果圖如下

前幾天由于團隊需要,折騰了一下圖像液化的處理過程。
現(xiàn)在來整理一下思路,做個記錄。

  

用到公式如下,網(wǎng)上拿來的

  

話不多說,上代碼
本來想盡量寫出點逼格。。。后來發(fā)現(xiàn)怎么寫也還是幾個function搞定,就那樣了。

(function(global) {

    // 計算兩點距離平方
    function distanceSqr( x1, y1, x2, y2 ) { return sqr(x1-x2) + sqr(y1-y2); }
    // 計算平方
    function sqr(x) { return x*x; }

    // 遍歷一個指定圓內(nèi)的所有點
    // 通過callback傳入回調(diào)方法,回調(diào)傳出每一個點的相關(guān)信息
    function eachCircleDot( imageData, ox, oy, r, callback ) {

        var imgWidth    = imageData.width,
            imgHeight   = imageData.height,
            data        = imageData.data,
            left        = ox-r,
            right       = ox+r,
            top         = oy-r,
            bottom      = oy+r,
            dotRedOffset,dotGreenOffset,dotBlueOffset,alphaOffset;

        for( var x = left; x < right; x++ )
            for( var y = top; y < bottom; y++ )

                if( distanceSqr( x, y, ox, oy ) <= sqr(r) ) {

                    dotRedOffset      = y*imgWidth*4+x*4;
                    dotGreenOffset    = dotRedOffset    + 1;
                    dotBlueOffset     = dotGreenOffset  + 1;
                    alphaOffset       = dotBlueOffset   + 1;

                    callback(
                        // 當(dāng)前點的坐標(biāo)
                        { x:x, y:y },
                        // 點的RGBA四個分量對應(yīng)字節(jié)的下標(biāo)
                        {
                            r: dotRedOffset,
                            g: dotGreenOffset,
                            b: dotBlueOffset,
                            a: alphaOffset,
                        },
                        // 傳進來的ImageData的data部分
                        data

                    );

                }

    }

    // 復(fù)制一個imageData的data到一個buff里
    function copyImageDataBuff( imgData ) {

        var data = imgData.data,
            imgDataBuff = [];

        for( var i in data )
            imgDataBuff[i] = data[i];

        return imgDataBuff;

    }

    // 從buff按照指定坐標(biāo)復(fù)制像素點數(shù)據(jù)到目標(biāo)imageData里
    function moveDot( imgData, dataBuff, x, y, srcX, srcY ) {

        var imgWidth    = imgData.width,
            imgHeight   = imgData.height,

            data = imgData.data;

        x = Math.floor(x);
        y = Math.floor(y);

        srcX = Math.floor(srcX);
        srcY = Math.floor(srcY);

        var targetStartOffset   = y*imgHeight*4 + x*4,
            srcStartOffset      = srcY*imgHeight*4 + srcX*4;


        for( var i = 0; i < 4; i++ )
            data[ targetStartOffset + i ] = dataBuff[ srcStartOffset + i ];

    }

    // 執(zhí)行液化過程
    // imgData  通過canvas的getImageData方法得到的數(shù)據(jù)對象
    // cx,cy    圓心坐標(biāo)
    // mx,my    移動目標(biāo)坐標(biāo)
    // r        作用半徑
    // strength 力度百分比(1-100)
    function liquify( imgData, cx, cy, mx, my, r, strenth ) {

        var imgDataBuff = copyImageDataBuff(imgData);

        eachCircleDot( imgData, cx, cy, r, function( posi ) {

            var tx = posi.x,
                ty = posi.y;


            var u = transFormula( cx, cy, mx, my, tx, ty, r, strenth );


            moveDot( imgData, imgDataBuff, tx, ty, u.x, u.y );


            function transFormula( cx, cy, mx, my, tx, ty, r, strenth ) {


                strenth = strenth || 100;

                var relativity = sqr(r) - distanceSqr( tx, ty, cx, cy );

                var distanceMovedSqr    = distanceSqr( mx, my, cx, cy );

                var rate = sqr( relativity / ( relativity + distanceMovedSqr*(100/strenth) ) );


                var ux = tx - rate * (mx-cx),
                    uy = ty - rate * (my-cy);

                return { x:ux, y:uy };

            }

        });

    }

    // 掛到全局對象
    global.LiquifyFilter = {
        liquify: liquify
    };

})(window);
  

使用它

先用canvas的

getImageData();

方法獲取到要處理圖片的imageData

全局作用域下調(diào)用

LiquifyFilter.liquify( imageData, 圓心X, 圓心Y, 目標(biāo)點X, 目標(biāo)點Y, 作用半徑, [力度百分比] );

完成轉(zhuǎn)換。

然后再用canvas的

puImageData();

把轉(zhuǎn)換后的imageData輸出到canvas中

  

效果圖如下

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/85412.html

相關(guān)文章

  • 小記-canvas完成圖像液化向前變形過程

    摘要:前幾天由于團隊需要,折騰了一下圖像液化的處理過程。現(xiàn)在來整理一下思路,做個記錄。用到公式如下,網(wǎng)上拿來的話不多說,上代碼本來想盡量寫出點逼格。。。后來發(fā)現(xiàn)怎么寫也還是幾個搞定,就那樣了。然后再用的把轉(zhuǎn)換后的輸出到中效果圖如下 前幾天由于團隊需要,折騰了一下圖像液化的處理過程。 現(xiàn)在來整理一下思路,做個記錄。 用到公式如下,網(wǎng)上拿來的 showImg(https://segm...

    jone5679 評論0 收藏0
  • 瘦臉之液化算法

    摘要:對上面公式進行改進,加入變形程度控制變量,改進后瘦臉公式如下,優(yōu)缺點優(yōu)點形變思路簡單直接缺點局部變形算法,只能基于一個中心點,向另外一個點的方向啦。單點拉伸的變形,可以實現(xiàn)瘦臉的效果,但是效果自然度有待提升。 論文:Interactive Image Warping(1993年Andreas...

    BWrong 評論0 收藏0
  • 前端小記4——高性能mobile web開發(fā)

    摘要:高性能動畫與端場景需要相比,移動端需要考慮的因素也相對復(fù)雜,重點考慮流量功耗與流暢度。而在移動端,我們選擇性能更優(yōu)瀏覽器原生實現(xiàn)方案動畫。然而,動畫在移動多終端設(shè)備場景下,相比會面對更多的性能問題,主要體現(xiàn)在動畫的卡頓與閃爍。1.高性能CSS3動畫 與PC端場景需要相比,移動web端需要考慮的因素也相對復(fù)雜,重點考慮:流量、功耗與流暢度。在pc端上考慮更多的是流暢度,而mobile web中...

    番茄西紅柿 評論0 收藏0
  • 使vue完成微信公眾號網(wǎng)頁小記

    摘要:前言公司最近有一個頁面的功能,比較簡單的一個調(diào)查表功能,嵌套在我們微信公眾號里面。同時用到了微信的登錄和分享接口。參考鏈接使用微信接口前端部分我們用微信接口主要是做的登錄和分享功能,首先是上微信公眾平臺上邊看看,把權(quán)限搞好之后后端配置。 showImg(https://segmentfault.com/img/bVbrOkH); 前言: 公司最近有一個H5頁面的功能,比較簡單的一個調(diào)查...

    notebin 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<