使用Vue transition 實現(xiàn)點贊動畫效果

效果一覽

image

點贊時愛心縮小變大,變大時略微大一點再變正常。
取消點贊時愛心無動畫

同時數(shù)字滾動。+1 時向上滾動,-1 時向下滾動。

愛心效果

材料:愛心圖標(biāo)兩個,沒有我這種 icon 組件的用 png 圖片代替

<transition :name=" isLike ? 'zoom' : '' " mode="out-in">
    <!-- 愛心圖標(biāo) -->
    <icon data="@icon/like.svg" color="#FF0000" v-if="isLike" key="like"></icon>
    <icon data="@icon/unlike.svg" color="#333333" v-else key="unlike"></icon>
</transition>

因為在點贊時有動畫,取消點贊不要動畫,所以 transition 的 name 屬性需要根據(jù) isLike 變量變化,當(dāng) isLiketrue 時,給他 zoom 動畫,否則不給動畫。動畫模式是 out-in 即先出后進(jìn)。原來的圖標(biāo)先從大變小,然后新圖標(biāo)再從小變大。

這里注意當(dāng)兩個切換的組件名稱一致時,需要加 key 屬性,以區(qū)分兩個組件,否則動畫不會生效

接下來寫 css

/** 動畫進(jìn)行時的class **/
.zoom-enter-active, .zoom-leave-active {
    transition: all .15s cubic-bezier(0.42, 0, 0.34, 1.55);
}

/** 設(shè)置進(jìn)場開始的狀態(tài)和離場結(jié)束的狀態(tài),都是縮放到0 **/
.zoom-enter, .zoom-leave-to {
    transform: scale(0);
}

/** 設(shè)置進(jìn)場結(jié)束的狀態(tài)和離場開始的狀態(tài), 都是縮放到1 **/
.zoom-enter-to, .zoom-leave {
    transform: scale(1);
}

根據(jù)官方文檔描述,.name-enter-active 和 .name-leave-active 會在動畫進(jìn)行時設(shè)置到 icon 組件的 class 中,所以這里我們設(shè)置 transition 的動畫屬性,時間以及曲線。

因為我們需要在放大時略微比 scale(1) 要大一點再回到正常大小,所以要自定義動畫曲線 cubic-bezier(0.42, 0, 0.34, 1.55) 。這個曲線怎么來的呢?

打開 chrome 的調(diào)試面板,隨便找個 dom 設(shè)置 transition-timing-function: ease; 然后點擊 ease 旁邊的曲線小圖標(biāo)

image

拖動拉桿就可以調(diào)整曲線

image

在動畫快結(jié)尾的時候,使曲線超出結(jié)束點即可。

image

然后復(fù)制曲線面板下面的值 cubic-bezier(0.25, 0.1, 0.27, 1.32) 即可。

更多關(guān)于動畫時間曲線的知識,這里就不贅述了,網(wǎng)上有很多相關(guān)知識。

關(guān)于縮放部分,按照上述 css 設(shè)定以及 mode="out-in" 動畫模式是先出后進(jìn),在點贊時。

  1. 原來的愛心開始離場,此時原來愛心縮放狀態(tài)為 1,即大小的 100%
  2. 愛心離場動畫進(jìn)行中,此時開始從 1 縮放到 0,即大小的 0%
  3. 原愛心離場完成,新愛心開始進(jìn)場,此時新愛心縮放狀態(tài)為 0
  4. 新愛心進(jìn)場動畫進(jìn)行中,此時開始從 0 動畫縮放到結(jié)束狀態(tài) 1

當(dāng)取消點贊的時候,isLikefalse,transition 的 name 就等于空字符串,此時就不會有動畫了。

數(shù)字滾動動畫

因為只是數(shù)字的變化,因此 transition 里只需要一個 div 即可。只是要注意 div 設(shè)置 key,以標(biāo)識數(shù)據(jù)變動。

<div class="like-num-wrapper">
    <transition :name="item.is_like ? 'plus' : 'minus'">
        <div
                class="like-num"
                :style="{color: item['is_like'] ? 'red': '#333'}"
                :key="item['like_num']"
        >
            {{item['like_num']}}
        </div>
    </transition>
</div>
  .like-num-wrapper {
    position: relative;
    margin-left: 16px;
    text-align: end;
    font-size: 13px;
    height: 17px;
    overflow-y: hidden;

    .like-num {
      top: 0;
      left: 0;
      position: relative;
      line-height: 17px;
    }
  }

需要注意我們?yōu)榱撕糜嬎闵舷聺L動的距離,所以需要把數(shù)字的高度定死在 17px,接下來寫 transition 的動畫 class。我們通過點贊狀態(tài)來判斷應(yīng)該用那一套動畫,點贊時 transition 的 name 是 plus ,取消點贊時是 minus 。

// 點贊數(shù)字+1動畫
.plus-enter-active, .plus-leave-active {
  transition: all .3s ease-in;
}

.plus-enter, .plus-leave {
  transform: translateY(0);
}

.plus-enter-to, .plus-leave-to {
  transform: translateY(-17px);
}

// 點贊數(shù)字-1動畫
.minus-enter-active, .minus-leave-active {
  transition: all .3s ease-in;
}

.minus-enter {
  transform: translateY(-34px);
}

.minus-enter-to {
  transform: translateY(-17px);
}

.minus-leave {
  transform: translateY(0);
}

.minus-leave-to {
  transform: translateY(17px);
}

點贊動畫

點贊動畫很簡單,在點贊時會在老的數(shù)字div下面生成一個新數(shù)字div。此時只需要讓他們都往上移動 17px 即可。

image.png

看起來大概就是這樣。1 和 2 都在 .like-num-wrapper div 下,而這個 div 我們設(shè)置了overflow-y: hidden css 屬性,所以 2 此時是被隱藏的。讓 1 和 2 這兩個 div 都往上移動 17px 即可。

取消點贊動畫

當(dāng)點擊取消點贊時,情形跟點贊一樣,只是需要對 2 和 1 做不同的位移。

image.png

因為取消點贊是數(shù)字從上往下滾動,所以需要讓數(shù)字 1 的初始位置在 2 上面。所以寫出如下代碼,設(shè)定數(shù)字 1 的動畫初始位置

.minus-enter {
  transform: translateY(-34px);
}

為什么是 -34px ?因為數(shù)字div的高度是 17px 往上移 17px 會和 2 重疊,那么數(shù)字 1 再往上移動 17px 就會出現(xiàn)在 2 的上面。-17-17 = 34 這一切都發(fā)生在電光火石之間。

接下來在 .minus-enter-to 中位移到 -17px 就可以實現(xiàn) 1 往下滾動到 2 這個效果了。

數(shù)字 2 的離場動畫就簡單多了 直接從 0 到 17px 就可以滾出去了。

至此整個點贊效果就完成了,歡迎點贊我的這篇文章????

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容