微信小程序使用画布实现飘落泡泡功能

微信小程序使用画布实现飘落泡泡功能:从组件封装到页面调用的完整实践

先看示例截图:

一、背景与技术选型

在微信小程序中实现类似于飘落的泡泡或者樱花飘落的功能,一般主要有 Canvas 和图片两种方案:

(1)Canvas 方案:性能优异,资源占用小,但视觉表现依赖绘图 API

(2)图片方案:视觉效果真实,但资源加载与内存占用较高

本文将采用Canvas 方案实现飘落的泡泡组件,通过组件化设计提升代码复用性,并分享性能优化经验,帮助开发者在真实感与性能间找到平衡点。
二、组件设计与实现
2.1 组件结构设计

首先创建组件文件夹components/down-rain,目录结构如下:

bash 复制代码
components/
└─ down-rain/
   ├── index.js         // 逻辑层
   ├── index.wxml       // 视图层
   ├── index.wxss       // 样式层
   ├── index.json       // 配置文件

2.2 组件核心代码实现

以下是组件的完整实现:

javascript 复制代码
Component({
  properties: {
    // 数量
    petalCount: { type: Number, value: 60 },
    //  大小
    petalSize: { type: Array, value: [3, 8] },
    // 下落速度
    speed: { type: Number, value: 2 },
    // 风力影响
    wind: { type: Number, value: 0.3 },
  },

  data: {
    canvasWidth: 0,
    canvasHeight: 0,
    ctx: null,
    animationTimer: null,
    petals: []
  },

  lifetimes: {
    attached() {
      this.initCanvas();
    },
    detached() {
      this.stopAnimation();
    }
  },

  methods: {
    // 初始化Canvas
    async initCanvas() {
      const query = this.createSelectorQuery();
      query.select('.canvas').boundingClientRect();
      const { windowWidth, windowHeight } = wx.getSystemInfoSync();
      const canvas = wx.createCanvasContext('downCanvas', this);
      
      this.setData({
        canvasWidth: windowWidth,
        canvasHeight: windowHeight,
        ctx: canvas,
        petals: this.createPetals()
      });
      
      this.startAnimation();
    },

    // 创建花瓣数组 - 修改:从随机位置开始下落
    createPetals() {
      const { petalCount, petalSize } = this.properties;
      const { canvasWidth, canvasHeight } = this.data;
      const vanishLine = canvasHeight * 2 / 3; // 消失线位置
      
      return Array(petalCount).fill().map(() => {
        // 随机生成初始位置,y可以在画布上方
        const y = Math.random() * canvasHeight * 1.5 - canvasHeight * 0.5;
        return {
          x: Math.random() * canvasWidth,
          y,
          size: petalSize[0] + Math.random() * (petalSize[1] - petalSize[0]),
          speed: 0.5 + Math.random() * this.properties.speed,
          angle: Math.random() * Math.PI * 2,
          wind: (Math.random() - 0.5) * this.properties.wind,
          alpha: 0.5 + Math.random() * 0.5,
          startY: y, // 记录起始Y坐标用于计算消失
          visible: y < vanishLine // 初始可见性判断
        };
      });
    },

    // 开始动画
    startAnimation() {
      this.data.animationTimer = setInterval(() => {
        this.updatePetals();
        this.drawPetals();
      }, 30);
    },

    // 停止动画
    stopAnimation() {
      if (this.data.animationTimer) {
        clearInterval(this.data.animationTimer);
      }
    },

    // 更新位置
    updatePetals() {
      const { canvasWidth, canvasHeight, petals } = this.data;
      const vanishLine = canvasHeight * 2 / 3; 
      
      this.setData({
        petals: petals.map(petal => {
          petal.y += petal.speed;
          petal.x += Math.sin(petal.angle) * petal.wind;
          petal.angle += 0.02;
          
          // 计算与消失线的距离比例
          const distanceRatio = Math.max(0, (petal.y - vanishLine) / (canvasHeight - vanishLine));
          
          // 超过消失线后逐渐消失
          if (distanceRatio > 0) {
            petal.alpha = Math.max(0, petal.alpha * (1 - distanceRatio));
            petal.visible = petal.alpha > 0;
          } else {
            petal.visible = true;
            petal.alpha = 0.5 + Math.random() * 0.5; // 重置透明度
          }
          
          // 完全消失后重置位置
          if (!petal.visible || petal.y > canvasHeight) {
            return {
              x: Math.random() * canvasWidth,
              y: -10, // 从顶部重新开始
              size: petal.size, // 保持原有大小
              speed: 0.5 + Math.random() * this.properties.speed,
              angle: Math.random() * Math.PI * 2,
              wind: petal.wind, // 保持原有风力影响
              alpha: 0.5 + Math.random() * 0.5,
              startY: -10,
              visible: true
            };
          }
          
          return petal;
        })
      });
    },

    // 绘制
    drawPetals() {
      const { ctx, petals } = this.data;
      ctx.clearRect(0, 0, this.data.canvasWidth, this.data.canvasHeight);
      petals.forEach(petal => {
        if (petal.visible) {
          ctx.beginPath();
          ctx.arc(petal.x, petal.y, petal.size, 0, Math.PI * 2);
          ctx.fillStyle = `rgba(255, 192, 203, ${petal.alpha})`;
          ctx.fill();
        }
      });

      ctx.draw(false);
    },
  }
});

2.3 视图层实现

xml 复制代码
<canvas class="canvas" canvas-id="downCanvas"></canvas>

2.4 样式层实现

css 复制代码
.canvas {
  position: fixed;
  top: 0;
  left: 0;
  width: 100%;
  height: 100%;
  pointer-events: none;
  z-index: 999;
}

三、页面调用与集成
3.1 页面配置

在需要调用的界面的json文件处引入组件

javascript 复制代码
{
  "usingComponents": {
    "down-rain": "/components/down-rain/index"
  },
  "navigationStyle": "custom"
}

3.2 页面布局

xml 复制代码
<down-rain petalCount="50" speed="5"></down-rain>

四、总结与拓展

本文通过组件化设计实现了微信小程序中基于Canvas 的飘落泡泡的效果。实际项目中,可根据活动预算和性能要求选择合适的实现方案:

(1)对性能要求高、视觉要求低的场景推荐使用 Canvas 方案

(2)对视觉效果要求高、预算充足的场景推荐使用图片方案

编写不易,谢谢点赞+收藏+关注,后续更新更多示例呦~

相关推荐
千百元6 分钟前
jenkins打包问题jar问题
前端
喝拿铁写前端8 分钟前
前端批量校验还能这么写?函数式校验器组合太香了!
前端·javascript·架构
巴巴_羊12 分钟前
6-16阿里前端面试记录
前端·面试·职场和发展
我是若尘14 分钟前
前端遇到接口批量异常导致 Toast 弹窗轰炸该如何处理?
前端
该用户已不存在37 分钟前
8个Docker的最佳替代方案,重塑你的开发工作流
前端·后端·docker
然我39 分钟前
面试官最爱的 “考试思维”:用闭包秒杀递归难题 🚀
前端·javascript·面试
明月与玄武1 小时前
HTML知识全解析:从入门到精通的前端指南(上)
前端·html
teeeeeeemo1 小时前
CSS place-items: center; 详解与用法
前端·css·笔记
未来之窗软件服务1 小时前
html读取身份证【成都鱼住未来身份证】:CyberWinApp-SAAS 本地化及未来之窗行业应用跨平台架构
前端·html·身份证读取
木木jio1 小时前
🧹 前端日志查询组件的重构实践:从 1600 行巨型组件到模块化 hooks
前端·react.js