console
window.onload=function(){
!function(){
class DogAnimation {
constructor (canvas) {
this.canvas = canvas;
canvas.width = window.innerWidth;
window.onresize = () => canvas.width = window.innerWidth;
canvas.height = 200;
this.lastWalkingTime = Date.now();
this.keyFrameIndex = -1;
this.ctx = this.canvas.getContext("2d");
this.RES_PATH = "https://webpublish.oss-cn-beijing.aliyuncs.com/wk/assets/images/dog";
this.IMG_COUNT = 8;
this.dog = {
stepDistance: 9,
speed: 0.15,
mouseX: -1,
frontStopX: -1,
backStopX: window.innerWidth,
};
}
async start () {
await this.loadResources();
this.pictureWidth = this.dogPictures[0].naturalWidth / 2;
this.dog.mouseX = window.innerWidth - this.pictureWidth;
this.recordMousePosition();
window.requestAnimationFrame(this.walk.bind(this));
}
recordMousePosition() {
window.addEventListener("mousemove", event => {
this.dog.frontStopX = event.clientX - this.pictureWidth;
this.dog.backStopX = event.clientX;
});
window.addEventListener("touchstart", event => {
this.dog.frontStopX = event.touches[0].clientX - this.pictureWidth;
this.dog.backStopX = event.touches[0].clientX;
});
}
loadResources () {
let imagesPath = [];
for (let i = 0; i <= this.IMG_COUNT; i++) {
imagesPath.push(`${this.RES_PATH}/${i}.png`);
}
let works = [];
imagesPath.forEach(imgPath => {
works.push(new Promise(resolve => {
let img = new Image();
img.onload = () => resolve(img);
img.src = imgPath;
}));
});
return new Promise(resolve => {
Promise.all(works).then(dogPictures => {
this.dogPictures = dogPictures;
resolve();
});
});
}
walk () {
let now = Date.now();
let diffDistance = (now - this.lastWalkingTime) * this.dog.speed;
if (diffDistance < this.dog.stepDistance) {
window.requestAnimationFrame(this.walk.bind(this));
return;
}
this.keyFrameIndex = ++this.keyFrameIndex % this.IMG_COUNT;
let direct = -1,
stopWalking = false;
if (this.dog.frontStopX > this.dog.mouseX) {
direct = 1;
}
else if (this.dog.backStopX < this.dog.mouseX) {
direct = -1;
}
else {
stopWalking = true;
direct = this.dog.frontStopX === -1 ? -1 :
this.dog.backStopX - this.dog.mouseX
> this.pictureWidth / 2 ? 1 : -1;
this.keyFrameIndex = -1;
}
let ctx = this.ctx;
ctx.clearRect(0, 0, this.canvas.width, this.canvas.height);
ctx.save();
if (!stopWalking) {
this.dog.mouseX += this.dog.stepDistance * direct;
}
if (direct === -1) {
ctx.scale(direct, 1);
}
let img = this.dogPictures[this.keyFrameIndex + 1];
let drawX = 0;
drawX = this.dog.mouseX * direct -
(direct === -1 ? this.pictureWidth : 0);
ctx.drawImage(img, 0, 0, img.naturalWidth, img.naturalHeight,
drawX, 20, 186, 162);
ctx.restore();
this.lastWalkingTime = now;
window.requestAnimationFrame(this.walk.bind(this));
}
}
let canvas = document.querySelector("#dog-walking");
let dogAnimation = new DogAnimation(canvas);
dogAnimation.start();
}();
}
<canvas id="dog-walking" width="0" height="0" style="position:fixed;bottom:0;left:0"></canvas>
body, p, h2{
margin: 0;
padding: 0;
}
h2 {
font-size: 18px;
}
body {
background-color: #ebeced;
font-size: 16px;
}
.body-wrapper {
max-width: 815px;
margin: 0 auto;
}
header {
position: relative;
}
body:before {
content: "";
position: absolute;
left: 0;
top: 0;
width: 100%;
height: 240px;
background-color: rgb(42, 170, 237);
}
.basic-info {
width: 760px;
height: 260px;
background-color: #fff;
margin: 60px auto 0;
position: relative;
}
.portrait-container {
float: left;
width: 180px;
height: 180px;
border: 1px solid #fff;
margin: 40px 20px 0;
border-radius: 50%;
overflow: hidden;
}
.my-portrait {
width: 180px;
margin: 0 auto;
}
#dog-walking {
position: fixed;
bottom: 0;
left: 0;
}