天天看點

JS 實作抛物線動畫

author: 陳家賓
email: [email protected]
date: 2018/2/24
           

在做小程式的項目中,需要在添加購物車的時候,增加抛物線小球動畫。

先給大家看下效果圖(其實已經是實作後的效果了,順便給自己公司打廣告了哈哈)

JS 實作抛物線動畫

分析

這種不固定起始位置的動畫,自然不能用 gif 圖,是以隻能用原生代碼實作

那我們有什麼工具來實作動畫呢?

  • 小程式提供了 JS API

    createAnimation

    來建立動畫
  • CSS transition

工具有了,我們再看一下什麼是抛物線。

這裡我們隻讨論水準抛物線,水準抛物線從數學原理上來說就是【水準勻速、垂直加速的運動】,轉換成代碼層面就是在動畫效果

timingFunction

中,水準動畫采用

linear

,垂直動畫采用

ease-in

是以我們需要把這個抛物線動畫分解成 兩個 同時 進行但 不同動畫效果 的動畫。

實作

(一)小程式的實作

JS:

cartAnimation(x, y) { // x y 為手指點選的坐标,即球的起始坐标
    let self = this,
        cartY = app.globalData.winHeight - , // 結束位置(購物車圖檔)縱坐标
        cartX = , // 結束位置(購物車圖檔)的橫坐标
        animationX = flyX(cartX, x), // 建立球的橫向動畫
        animationY = flyY(cartY, y) // 建立球的縱向動畫
    this.setData({
        ballX: x,
        ballY: y,
        showBall: true
    })
    setTimeoutES6().then(() => { // 100 秒延時,確定球已經到位并顯示
        self.setData({
            animationX: animationX.export(),
            animationY: animationY.export(),
        })
        return setTimeoutES6() // 400 是球的抛物線動畫時長
    }).then(() => { // 400 秒延時後隐藏球
        this.setData({
            showBall: false,
        })
    })
}

function setTimeoutES6(sec) { // Promise 化 setTimeout
    return new Promise((resolve, reject) => {
        setTimeout(() => {resolve()}, sec)
    })
}

function flyX(cartX, oriX) { // 水準動畫
    let animation = wx.createAnimation({
        duration: ,
        timingFunction: 'linear',
    })
    animation.left(cartX).step()
    return animation
}

function flyY(cartY, oriY) { // 垂直動畫
    let animation = wx.createAnimation({
        duration: ,
        timingFunction: 'ease-in',
    })
    animation.top(cartY).step()
    return animation
}
           

HTML:

<view animation="{{animationY}}" style="position:fixed;top:{{ballY}}px;" hidden="{{!showBall}}">
    <view class="ball" animation="{{animationX}}" style="position:fixed;left:{{ballX}}px;"></view>
</view>
           

translate 優化

據我所知,用

transform: translate()

來實作的動畫會比 top & left 性能更優,但實作下來卻沒那麼容易咯。

研究來研究去,發現 translate 的做法比 top & left 的做法多了一步,就是需要将小球的 translate 位移還原(否則 translate 一直有值),才能保證下一次的位移從點選的位置開始

cartAnimation(x, y) {
    let self = this,
        cartY = app.globalData.winHeight - ,
        cartX = ,
        animationX = flyX(cartX, x),
        animationY = flyY(cartY, y)
    this.setData({
        leftNum: x,
        topNum: y,
        showBall: true
    })
    setTimeoutES6().then(() => {
        self.setData({
            animationDataX: animationX.export(),
            animationDataY: animationY.export(),
        })
        return setTimeoutES6()
    }).then(() => {
        this.setData({
            showBall: false,
            animationX: flyX(, , ).export(), // 還原小球位置,即 translate 恢複預設值
            animationY: flyY(, , ).export(),
        })
    })
}

function flyX(cartX,oriX,duration) {
    let animation = wx.createAnimation({
        duration: duration||,
        timingFunction: 'linear',
    })
    animation.translateX(cartX-oriX).step()
    return animation
}
function flyY(cartY,oriY,duration) {
    let animation = wx.createAnimation({
        duration: duration||,
        timingFunction: 'ease-in',
    })
    animation.translateY(cartY-oriY).step()
    return animation
}
           

HTML 部分不變

(二)H5 的實作

除了小程式之外,前端日常開發更多的當然還是 H5,下面我将用 CSS3 transition 的方法來實作

<!DOCTYPE html>
<html lang="en" style="width:100%;height:100%;">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width">
    <style>
        * {
            padding: ;
            margin: ;
        }
        #ball {
            width:px;
            height:px;
            background: #5EA345;
            border-radius: %;
            position: fixed;
            transition: left s linear, top s ease-in;
        }
    </style>
    <title>CSS3 水準抛物線動畫</title>
</head>
<body style="width:100%;height:100%;">
    <div id="ball"></div>
</body>
<script>
    var $ball = document.getElementById('ball');
    document.body.onclick = function (evt) {
        console.log(evt.pageX,evt.pageY)
        $ball.style.top = evt.pageY+'px';
        $ball.style.left = evt.pageX+'px';
        $ball.style.transition = 'left 0s, top 0s';
        setTimeout(()=>{
            $ball.style.top = window.innerHeight+'px';
            $ball.style.left = '0px';
            $ball.style.transition = 'left 1s linear, top 1s ease-in';
        }, )
    }
</script>
</html>
           

還有體驗連結哦,點我

至此,水準抛物線動畫的實作就介紹得差不多啦,嘻嘻!!

繼續閱讀