飛入購物車動畫uniapp+vue3

頁面代碼

<template>
    <view class="main">
        <scroll-view :scroll-top="0" scroll-y="true" style="height:90vh">
            <view class="box-right">
                <view>
                    <block v-for="item in 15">
                        <view @click="addCar($event)" class="add-btn">
                            加入購物車
                        </view>
                    </block>
                </view>
            </view>
        </scroll-view>
        <cartsBall ref="cartBtn" :endPos="{
                x: 80, y:height
            }"></cartsBall>

    </view>

</template>

<script setup>
    import {
        onMounted,
        ref
    } from "vue";
    import cartsBall from './cartsBall.vue'
    const height = ref(0)
    onMounted(() => {
        height.value = uni.getWindowInfo().windowHeight
    })

    const cartBtn = ref()
    const addCar = (e) => {
        cartBtn.value.drop({
            x: e.detail.x,
            y: e.detail.y
        })
    }
</script>

<style>
    .main {
        height: 90vh;
        background: #cdcdcd;
    }

    .box-right {
        display: flex;
        justify-content: end;
    }
 

    .add-btn {
        display: flex;
        width: 80px;
        height: 50px;
        line-height: 50px;
        background: #a0cfff;
        margin-bottom: 40px;
    }
</style>

cartsBall.vue 組件代碼

<template>
    <view>
        <view class="ball-wrap" v-for="(item, k) of balls" :key="k"
            :style="{
                opacity: item.show,
                top: item.start.y + 'px',
                left: item.start.x + 'px',
                transitionDuration: (item.show?(duration/1000):0)+'s',
                'transition-timing-function': xTimeFunction[!item.ani?0:1],
                transform: 'translate3d('+item.offset.x+'px,0,0)',
                zIndex
            }"
        >
            <view class="ball" :class="{ball3d:is3dSheet}"
                :style="{
                    marginLeft: -size/2 + 'px',
                    marginTop: -size/2 + 'px',
                    padding: size + 'px',
                    backgroundImage: ballImage,
                    backgroundColor: ballColor,
                    transitionDuration: (item.show?(duration/1000):0)+'s',
                    transform: 'translate3d(0,'+item.offset.y+'px,0)',
                    'transition-timing-function': yTimeFunction[item.ani?0:1]
                }"
            ></view>
        </view>
    </view>
</template>

<script>
    // 節(jié)流函數
    function debounce(func, delay, context) {
        return function () {
            func.id && clearTimeout(func.id);
            func.id = setTimeout(() => {
                func.apply(context,arguments)
            }, delay)
        }
    }
    export default {
        props: {
            size: {
                type: Number,
                default: 8
            },
            //3D
            is3dSheet: {
                type: Boolean,
                default: true
            },
            endPos: {
                type: Object,
                default(){
                    return {
                        x: 150,
                        y: 150
                    }
                }
            },
            //持續(xù)時間
            duration: {
                type: Number,
                default: 700
            },
            zIndex: {
                type: Number,
                default: 9999
            },
            ballImage: {
                type: String,
                default: ''
            },
            ballColor: {
                type: String,
                default: 'red'
            }
        },
        data() {
            return {
                balls: [],
                xTimeFunction: ['ease-in', 'ease-out'],
                yTimeFunction: ['cubic-bezier(0.23,-0.1, 0.84,-0.01)', 'cubic-bezier(0, 0.59, 0.46, 2.15)']
            };
        },
        mounted() {
            this.initBalls()
        },
        methods: {
            drop(pos){
                let ball 
                let duration=this.duration
                for (var i = 0; i < this.balls.length; i++) {
                    if(this.balls[i].show){continue}
                    ball = this.balls[i]
                }
                ball.start.x = pos.x
                ball.start.y = pos.y
                ball.offset.x = this.endPos.x - pos.x 
                ball.offset.y = this.endPos.y - pos.y 
                if(ball.offset.y>0){
                    ball.ani = 1
                }else{
                    ball.ani = 0
                }
                ball.show = 1
                
                // 無限加載
                this.balls.push({
                    show: 0,
                    start: {
                        x: 0,
                        y: 0
                    },
                    offset: {
                        x: 0,
                        y: 0
                    },
                    ani: 0
                })
                if(ball.offset.y<120){
                    //y低于120時加速的代碼
                } 
                setTimeout(()=>{
                    ball.show = 0
                }, duration)
                debounce(this.initBalls,  duration+200, this)()
            },
            initBalls(){
                const balls = []
                // 解決微信自帶瀏覽器渲染慢問題死遭,先加載幾個
                for (var i = 0; i < 3; i++) {
                    const ball = {
                        show: 0,
                        start: {
                            x: 0,
                            y: 0
                        },
                        offset: {
                            x: 0,
                            y: 0
                        },
                        ani: 0
                    }
                    balls.push(ball)
                }
                this.balls = balls
            }
        }
    }
</script>

<style lang="scss" scoped>
.ball-wrap {
  position: fixed;
  transform: translateZ(0);
  z-index: 999;
  transition-property: transform;
  pointer-events: none;
  .ball {
    width: 0;
    height: 0;
    padding: 40rpx;
    border-radius: 40rpx;
    overflow: hidden;
    background-repeat: no-repeat;
    background-size: cover;
    transform: translateZ(0);
    transition-property: transform opacity;
    &.ball3d {
      position: relative;
      &::after {
        content: "";
        display: block;
        position: absolute;
        left: -15rpx;
        right: -15rpx;
        top: -15rpx;
        bottom: -15rpx;
        opacity: 0.35;
        background-image: radial-gradient(30% 30%, white, transparent, black);
        background-size: 100%;
        background-repeat: no-repeat;
        background-position: -5rpx -5rpx;
      }
    }
  }
}
</style>

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市腺律,隨后出現的幾起案子娱俺,更是在濱河造成了極大的恐慌洁段,老刑警劉巖廓脆,帶你破解...
    沈念sama閱讀 207,113評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件或油,死亡現場離奇詭異浮禾,居然都是意外死亡,警方通過查閱死者的電腦和手機尺借,發(fā)現死者居然都...
    沈念sama閱讀 88,644評論 2 381
  • 文/潘曉璐 我一進店門绊起,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人燎斩,你說我怎么就攤上這事虱歪。” “怎么了瘫里?”我有些...
    開封第一講書人閱讀 153,340評論 0 344
  • 文/不壞的土叔 我叫張陵实蔽,是天一觀的道長荡碾。 經常有香客問我谨读,道長,這世上最難降的妖魔是什么坛吁? 我笑而不...
    開封第一講書人閱讀 55,449評論 1 279
  • 正文 為了忘掉前任劳殖,我火速辦了婚禮铐尚,結果婚禮上,老公的妹妹穿的比我還像新娘哆姻。我一直安慰自己宣增,他們只是感情好,可當我...
    茶點故事閱讀 64,445評論 5 374
  • 文/花漫 我一把揭開白布矛缨。 她就那樣靜靜地躺著爹脾,像睡著了一般。 火紅的嫁衣襯著肌膚如雪箕昭。 梳的紋絲不亂的頭發(fā)上灵妨,一...
    開封第一講書人閱讀 49,166評論 1 284
  • 那天,我揣著相機與錄音落竹,去河邊找鬼泌霍。 笑死,一個胖子當著我的面吹牛述召,可吹牛的內容都是我干的朱转。 我是一名探鬼主播,決...
    沈念sama閱讀 38,442評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼积暖,長吁一口氣:“原來是場噩夢啊……” “哼藤为!你這毒婦竟也來了?” 一聲冷哼從身側響起夺刑,我...
    開封第一講書人閱讀 37,105評論 0 261
  • 序言:老撾萬榮一對情侶失蹤凉蜂,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后性誉,有當地人在樹林里發(fā)現了一具尸體窿吩,經...
    沈念sama閱讀 43,601評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 36,066評論 2 325
  • 正文 我和宋清朗相戀三年错览,在試婚紗的時候發(fā)現自己被綠了纫雁。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,161評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡倾哺,死狀恐怖轧邪,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情羞海,我是刑警寧澤忌愚,帶...
    沈念sama閱讀 33,792評論 4 323
  • 正文 年R本政府宣布,位于F島的核電站却邓,受9級特大地震影響硕糊,放射性物質發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,351評論 3 307
  • 文/蒙蒙 一简十、第九天 我趴在偏房一處隱蔽的房頂上張望檬某。 院中可真熱鬧,春花似錦螟蝙、人聲如沸恢恼。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽场斑。三九已至,卻和暖如春牵署,著一層夾襖步出監(jiān)牢的瞬間和簸,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評論 1 261
  • 我被黑心中介騙來泰國打工碟刺, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留锁保,地道東北人。 一個月前我還...
    沈念sama閱讀 45,618評論 2 355
  • 正文 我出身青樓半沽,卻偏偏與公主長得像爽柒,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子者填,可洞房花燭夜當晚...
    茶點故事閱讀 42,916評論 2 344

推薦閱讀更多精彩內容