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>
相关推荐
天***88527 分钟前
Edge 浏览器离线绿色增强版+官方安装包,支持win7等系统
前端·edge
漫游的渔夫16 分钟前
别再直接 `json.loads` 了!AI 返回的 JSON 坑位指南
前端·人工智能
软件工程师文艺28 分钟前
从0到1:Claude Code如何用React构建CLI应用
前端·react.js·前端框架
M ? A37 分钟前
Vue 迁移 React 实战:VuReact 一键自动化转换方案
前端·vue.js·经验分享·react.js·开源·自动化·vureact
yuki_uix38 分钟前
重排、重绘与合成——浏览器渲染性能的底层逻辑
前端·javascript·面试
Burt1 小时前
我的 2026 全栈选型:Vue3 + Elysia + Bun + AlovaJS
vue.js·全栈·bun
沃尔威武1 小时前
调试黑科技:Chrome DevTools时间旅行调试实战
前端·科技·chrome devtools
小锋java12341 小时前
SpringBoot 4 + Spring Security 7 + Vue3 前后端分离项目设计最佳实践
java·vue.js·spring boot
一 乐1 小时前
校园线上招聘|基于springboot + vue校园线上招聘系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·论文·毕设·校园线上招聘系统
yuki_uix1 小时前
虚拟 DOM 与 Diff 算法——React 性能优化的底层逻辑
前端·react.js·面试