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

首頁 > 開發 > HTML5 > 正文

Canvas圖片分割效果的實現

2024-09-05 07:22:57
字體:
來源:轉載
供稿:網友

之前在逛cssdesignawards時發現了一個把圖片內容分割的效果(網址:https://weareludwig.com),大家可以點進去看看,感覺挺炫酷的,于是自己試著實現了一下,效果還不錯。效果查看https://codepen.io/geeknoble/pen/OQaOVG

分析

首先我們可以發現圖片的內容被分成了一個個小矩形,并對每個矩形進行了隨機平移。Canvas的drawImage函數可以對圖片內容進行裁剪并繪制到Canvas畫布中,所以該效果主要實現原理就是使用drawImage。主要效果有兩個,一個是圖片內容的打亂和復原,一個是和下張圖片的切換,這兩個效果都可以使用drawImage,只是移動的距離不一樣??傮w思路有了那么就可以去著手實現一下。

初始工作

首先我們要初始化一些變量,比如圖片的寬高,矩形的個數,剪切的尺寸等,然后再計算每個矩形的坐標,使用一個二重循環將矩形坐標保存在data中。每個矩形有個隨機位移,這個位移也需要保存起來,存在randoms中。其中x,y表示canvas畫布的坐標,x1,y1表示圖片裁剪的坐標。

init: function (context, width, height, area, img) {            this.context = context;            this.img = img;            this.imgWidth = img[0].width;          //圖片寬高            this.imgHeight = img[0].height;            this.index = 0;                       //當前圖片序號            this.width = width;                  //畫布寬高            this.height = height;            this.area = height/12;                     //小矩形長度            this.countX = width / this.area;             //水平和垂直方向小矩形個數            this.countY = height / this.area;            this.wx = this.imgWidth / this.countX;      //圖片在小矩形中的寬高            this.wy = this.imgHeight / this.countY;            this.state = true;                   //圖片狀態,true表示未拆分            this.dataFlag = true;                //小矩形坐標狀態,true表示未加上隨機值            this.duration = 1000;                 //動畫時間            this.duration2 = 1500;            this.startTime = 0;            this.data = [];                       //小矩形坐標信息            this.randoms = [];                    //位置隨機值            //初始化矩形坐標            var x1 = 0, y1 = 0, x = 0, y = 0;            for (var i = 0; i < this.countY; i++) {                for (var j = 0; j < this.countX; j++) {                    context.drawImage(this.img[this.index], x1, y1, this.wx, this.wy, x, y, this.area, this.area);                    //儲存矩形坐標                    this.data.push({                        x1: x1,                        y1: y1,                        x: x,                        y: y                    });                    //添加隨機值                    this.randoms.push(random(-this.area, this.area));                    x1 += this.wx;                    x += this.area;                }                x1 = 0;                y1 += this.wy;                x = 0;                y += this.area;            }            this.checkMargin();        }

檢測邊緣

在給矩形添加位移之前我們需要判斷一下位移后的坐標是否超過圖片界限,比如在頂部的矩形如果是y軸移動,那么只能夠向上移,判斷的條件為當前坐標加上位移值是否小于0或大于圖片的寬高。如果更新后的坐標小于0,那么這個隨機值一定是負數,需要把隨機值改為正數,如果大于圖片高度,那么改成負數即可。由于每個矩形的移動都是在一個方向上移動,所以我這里寫成偶數位移動x軸,奇數位移動y軸。

//檢測邊緣        checkMargin: function () {            var self = this;            this.data.forEach(function (item, index) {                if (index % 2 == 0) {  // 下標為2的倍數時移動x軸,否則移動y軸                    if ( item.x1 + self.randoms[index] < 0)                        // 改為正數                        self.randoms[index] = -self.randoms[index];                    if (item.x1 + self.wx + self.randoms[index] > self.imgWidth )                        // 改為負數                        self.randoms[index] = -Math.abs(self.randoms[index])                } else {                    if (item.y1 + self.randoms[index] < 0)                        self.randoms[index] = -self.randoms[index];                    if (item.y1 + self.randoms[index] + self.wy > self.imgHeight)                        self.randoms[index] = -Math.abs(self.randoms[index])                }            })        }

分離和復原

動畫的內容的分離和復原就是更新矩形坐標的值,打亂內容只要將data里的坐標加上隨機值,而復原就是減去隨機值,

//檢測邊緣        checkMargin: function () {            var self = this;            this.data.forEach(function (item, index) {                if (index % 2 == 0) {  // 下標為2的倍數時移動x軸,否則移動y軸                    if ( item.x1 + self.randoms[index] < 0)                        // 改為正數                        self.randoms[index] = -self.randoms[index];                    if (item.x1 + self.wx + self.randoms[index] > self.imgWidth )                        // 改為負數                        self.randoms[index] = -Math.abs(self.randoms[index])                } else {                    if (item.y1 + self.randoms[index] < 0)                        self.randoms[index] = -self.randoms[index];                    if (item.y1 + self.randoms[index] + self.wy > self.imgHeight)                        self.randoms[index] = -Math.abs(self.randoms[index])                }            })        }

在儲存好坐標后就可以去實現平移動畫了,移動的過程有一個平滑的過渡,我們可以使用Tween.js的緩動算法,該算法有4個參數分別是當前時間,初始位置,結束位置,動畫時間。詳細內容可以參考張鑫旭的這篇文章https://www.zhangxinxu.com/wordpress/2016/12/how-use-tween-js-animation-easing/ 。通過Tween.js可以算出每一幀要移動的距離,然后再使用requestAnimationFrame去更新坐標。

blockAnimation: function () {            var flag = 1;            if (this.state) {  // 判斷是打亂圖片還是還原圖片                this.update(true)            } else {                flag = -1;                this.update(false);            }            var self = this;            this.startTime = +new Date();  // 獲取當前時間            this.state = !this.state;            (function animation() {                var t = +new Date();                if (t >= self.startTime + self.duration) {  // 動畫結束條件                    return false;                }                self.data.forEach(function (item, index) {                    if (index % 2 == 0) {                        var pos = Math.tween.Expo.easeInOut(t - self.startTime, 0, self.randoms[index] * flag, self.duration);   // 計算出每幀移動的距離                        self.context.drawImage(self.img[self.index], item.x1 + pos, item.y1, self.wx, self.wy, item.x, item.y, self.area, self.area);                    } else {                        var pos = Math.tween.Expo.easeInOut(t - self.startTime, 0, self.randoms[index] * flag, self.duration);                          self.context.drawImage(self.img[self.index], item.x1, item.y1 + pos, self.wx, self.wy, item.x, item.y, self.area, self.area);                    }                });                requestAnimationFrame(animation);            })();        }

到這里就已經實現了分離和復原的動畫了

圖片切換

接下來開始處理圖片切換的部分,這里跟輪播圖有點像,輪播圖動畫是將每個圖片位置移動可視窗口寬度的距離,這里也是一樣,只要將坐標加上圖片高度就可以實現y軸上的切換。和輪播圖不一樣的是,我們這里只有一個canvas標簽,在切換時只需要改變當前圖和下一張圖的坐標,當前圖移動距離為y1 + pos,下張圖移動距離為y1 + pos - imgHeight(為什么要減imgHeight就不用說了吧)。

//垂直滑動動畫        verticalAnimation: function (val) {            if (!this.time2) {                return false;            }            this.checkTime(2);            var self = this;            val ? val = 1 : val = -1;  //判斷上滑還是下滑            if ((this.index + val) < 0 || (this.index + val) >= (this.img.length)) {   //判斷圖片序號是否到底                return false;            }            this.state ? this.update(true) : this.update(false);            this.startTime = +new Date();            (function animation() {                var t = +new Date();                if (t >= self.startTime + self.duration2) {                    val === 1 ? self.index++ : self.index--;  //調整圖片順序                    self.index < 0 ? self.index = self.img.length - 1 : self.index;                    self.index >= self.img.length ? self.index = 0 : self.index;                    return false;                }                self.data.forEach(function (item) {                    var pos = Math.tween.Cubic.easeInOut(t - self.startTime, 0, (self.imgHeight) * val, self.duration2);                    // 更新當前圖片坐標                    self.context.drawImage(self.img[self.index], item.x1, item.y1 + pos, self.wx, self.wy, item.x, item.y, self.area, self.area);                    // 更新下張圖片坐標                    self.context.drawImage(self.img[self.index + val], item.x1, item.y1 + pos - self.imgHeight * val, self.wx, self.wy, item.x, item.y, self.area, self.area);                });                requestAnimationFrame(animation);            })()        }

x軸的切換也是同理,現在所有功能都差不多完成了,完整代碼可以在codepen里查看。

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

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲a级在线播放观看| 欧美乱妇高清无乱码| 欧美激情一区二区三级高清视频| 欧美日韩国产中字| 亚洲小视频在线观看| 视频一区视频二区国产精品| 91国偷自产一区二区三区的观看方式| 91黑丝高跟在线| 日日骚av一区| 国产精品亚洲综合天堂夜夜| 日韩中文字幕视频在线| 亚洲欧美中文字幕在线一区| 亚洲精品aⅴ中文字幕乱码| 国产婷婷成人久久av免费高清| 午夜精品三级视频福利| 欧美激情久久久| 久久久久一本一区二区青青蜜月| 俺去亚洲欧洲欧美日韩| 亚洲深夜福利在线| 日韩激情在线视频| 欧美一级视频在线观看| 国产精品成人一区| 国产丝袜高跟一区| 精品动漫一区二区三区| 久久99视频精品| 日韩中文字幕视频在线| 欧美视频在线观看免费| 亚洲免费av片| 国产亚洲精品一区二区| 欧美一区二区影院| 热99久久精品| 隔壁老王国产在线精品| 精品一区二区三区四区| 精品久久在线播放| 日韩精品在线播放| 精品一区精品二区| 国产精品国产自产拍高清av水多| 国内精品一区二区三区| 国产成人久久精品| 亚洲欧美在线磁力| 97免费中文视频在线观看| 精品久久久久久久久久| 中文字幕亚洲一区二区三区| 亚洲精品久久久久中文字幕二区| 91亚洲国产成人精品性色| 欧美视频在线视频| 亚洲欧美国产精品| 久久久久久久999精品视频| 久久国产加勒比精品无码| 91沈先生作品| 国产欧美在线观看| 欧美与黑人午夜性猛交久久久| 国内免费精品永久在线视频| 国产精品日韩在线观看| 永久555www成人免费| 国产成人高潮免费观看精品| 精品视频在线导航| 亚洲无av在线中文字幕| 黑人巨大精品欧美一区二区一视频| 国产成人jvid在线播放| 欧美黄色片在线观看| 亚洲区一区二区| 亚洲一区二区三区在线免费观看| 亚洲欧美日韩国产中文专区| 久久久久久久香蕉网| 久久久亚洲国产| 欧美性猛交xxxx久久久| 久久久久久久久久亚洲| 国内精久久久久久久久久人| 国产精品色视频| 97久久伊人激情网| 美女啪啪无遮挡免费久久网站| 92裸体在线视频网站| 亚洲成人久久网| 国产精品美女久久久久久免费| 国产日韩中文字幕| 精品亚洲一区二区三区在线播放| 亚洲成人网久久久| 日韩综合中文字幕| 欧美性xxxx极品hd满灌| 亚洲电影免费观看高清完整版在线观看| 欧美午夜精品久久久久久人妖| 久久综合国产精品台湾中文娱乐网| 26uuu另类亚洲欧美日本一| 日韩在线视频二区| 亚洲精品久久久久| 亚洲国产精久久久久久| 亚洲精品wwww| 欧美极品xxxx| 亚洲自拍偷拍视频| 久久久噜久噜久久综合| 成人亲热视频网站| 国产精品久久久久999| 亚洲免费中文字幕| 亚洲美女喷白浆| 日韩成人av一区| 欧美性受xxxx白人性爽| 日韩av免费在线看| 日韩激情av在线播放| 精品国产31久久久久久| 欧美日韩国产中文精品字幕自在自线| 57pao国产成人免费| 欧美日韩国产一区二区| 精品美女久久久久久免费| 日韩一区二区av| 欧美成人手机在线| 国产精品人人做人人爽| 午夜精品一区二区三区在线播放| 欧美天天综合色影久久精品| 亚洲午夜久久久久久久| 国产精品视频yy9099| 91国自产精品中文字幕亚洲| 久久久久久中文| 精品国内产的精品视频在线观看| 26uuu另类亚洲欧美日本老年| 国产在线999| 亚洲最大福利网站| 欧美激情在线播放| 4438全国亚洲精品在线观看视频| 精品成人国产在线观看男人呻吟| 久久91亚洲精品中文字幕| 91精品国产91| 亚洲激情在线观看视频免费| 欧美大秀在线观看| 全亚洲最色的网站在线观看| 久久久久久噜噜噜久久久精品| 69av在线播放| 欧美精品在线网站| 亚洲第一区第二区| 国产精品a久久久久久| 欧美视频一二三| 欧美精品免费在线观看| 国产精品视频网址| 国产欧美日韩视频| 国产日韩欧美在线视频观看| 国产成人精品免高潮费视频| 日韩亚洲欧美中文高清在线| 91精品国产综合久久香蕉的用户体验| 国产拍精品一二三| 国产在线播放不卡| 亚洲精品福利资源站| 日本久久久久久| 国产aⅴ夜夜欢一区二区三区| 欧美性高跟鞋xxxxhd| 亚洲精品国产拍免费91在线| 日本成人激情视频| 色香阁99久久精品久久久| 亚洲大胆美女视频| 国产精品美女www爽爽爽视频| 久久久久久国产精品三级玉女聊斋| 欧美乱大交xxxxx| 日韩精品免费在线视频观看| 成人国产精品一区二区| 久久久久久国产| 欧美一区二区大胆人体摄影专业网站| 亚洲色图日韩av| 久久久999国产| 7777精品视频| 亚洲精品欧美日韩专区| 68精品久久久久久欧美| 亚洲精品白浆高清久久久久久| 欧美激情手机在线视频| 欧美老女人性生活| 日本一区二区三区四区视频|