uniapp/微信小程序实现加入购物车点击添加飞到购物车动画

1、预期效果

2、实现思路

每次点击添加按钮时,往该按钮上方添加一个悬浮元素,通过位移动画将元素移到目标位置。

  1. 为每个点击元素设置不同的class,才能通过uni.createSelectorQuery来获取每个元素的节点信息;

  2. 添加一个与点击元素一模一样的动画元素;

  3. 获取点击元素的节点信息将动画元素放置到点击元素上方;

  4. 计算动画元素到目标位置的距离,获得xy坐标执行位移动画;

  5. 等待每个动画元素执行动画完毕后移除该元素。

3、核心代码

TypeScript 复制代码
<template>
  <view>
    <!-- 商品列表 -->
    <view v-for="item in goodsList" :key="item.id">
      <view :class="[`add-cart-${item.id}`]" @click="addToCart(item)">加购</view>
    </view>
    <!-- 动画元素列表 -->
    <view
      v-for="item in anims" :key="item.key"
      style="position: fixed; transition: transform 0.5s linear;"
      :style="{
        top: `${item.top}px`,
        left: `${item.left}px`,
        transform: `translate(${item.x}px, ${item.y}px)`,
      }"
    >
      加购
    </view>
  </view>
</template>

<script setup lang="ts">
import { ref } from 'vue';
import uniqueId from 'lodash-es/uniqueId';

const sys = uni.getSystemInfoSync();
const anims = ref<any[]>([]);
const goodsList = ref([{ id: 1 }, { id: 2 }, { id: 3 }])

function addToCart(item) {
  // 添加动画元素
  const key = uniqueId();
  anims.value.push({
    key,
    id: item.id,
    left: 0,
    top: 0,
    y: 0,
    x: 0,
  });
  // 获取点击元素的节点信息
  uni.createSelectorQuery().select(`.add-cart-${item.id}`)
    .boundingClientRect((e: any) => {
      // 初始化起始位置
      anims.value.some((citem) => {
        if (citem.key === key) {
          citem.top = e.top;
          citem.left = e.left;
          return true;
        }
        return false;
      });
      nextTick(() => {
        // 设置目标位置
        anims.value.some((citem) => {
          if (citem.key === key) {
            citem.y = sys.windowHeight - citem.top - 50;
            citem.x = -sys.windowWidth * 0.30;
            setTimeout(() => { // 等待动画执行完毕移除元素
              anims.value.splice(anims.value.findIndex((v: any) => v.key === key), 1);
            }, 500);
            return true;
          }
          return false;
        });
      });
    }).exec();
}

</script>
相关推荐
来恩10032 分钟前
jQuery选择器
前端·javascript·jquery
前端繁华如梦4 分钟前
树上挂苹果还是挂玻璃球?Three.js 程序化果实的完整实现指南
前端·javascript
墨痕诉清风11 分钟前
Web浏览器客户端检测网站网络健康(代码)
前端·网络·测试工具
IMPYLH13 分钟前
Linux 的 wc 命令
linux·运维·服务器·前端·bash
happybasic30 分钟前
Python库升级标准流程~
linux·前端·python
川冰ICE35 分钟前
前端工程化深度实战:从Webpack5到Vite5的构建工具演进与选型决策
前端
CDwenhuohuo36 分钟前
优惠券组件直接用 uview plus
前端·javascript·vue.js
用户740904723627544 分钟前
我用 curl 排查了一次 OpenAI-compatible API 连接失败:401、403、404 分别怎么定位
前端
kft13141 小时前
XSS深度剖析:从弹窗到持久化窃取Cookie
前端·web安全·xss·安全测试
烬羽1 小时前
《前端三权分立:HTML、CSS、JS为什么不能“乱搞”》
前端