微信小程序基于Canvas实现头像图片裁剪(上)

news2025/7/14 17:13:25

序言

嘿,打工人混迹职场这么久,图片处理肯定都没少碰。不过咱说实话,大部分时候都是直接 “抄近道”,用现成的三方组件😏。就像我,主打一个会用工具,毕竟善用工具可是咱人类的 “超能力”,不会用那可就真成 “原始人” 啦🤣。最近在搞微信小程序开发时,遇到个需求:用户上传图片得能裁剪。我瞅了瞅 UI 给的效果图,好家伙,找遍了都没发现合适的组件,这下没办法,社畜的 “使命感” 上身,只能自己动手,丰衣足食咯😅。

UI图

在这里插入图片描述

咱来拆解一下在微信小程序里实现这功能的流程。首先从上个页面调用小程序选择图片的 API——chooseMedia,这就好比打开了一个 “图片宝库”,通过它能拿到用户选好图片后的临时路径(下面咱就简称 path 啦)。接着带着这个 path “奔赴” 当前页面,然后用Canvas这个神奇的 “画布” 把图片画出来,之后还要让图片能实现移动、缩放这些 “炫酷操作”。

使用 Canvas 画出图片

选择图片这块咱就不细究啦,不是本文的 “主角”,咱直接从拿到 path 之后讲起。

第一步得初始化 Canvas,给它设定宽高。在页面加载完毕时,利用createSelectorQuery这个 “小助手”,拿到 Canvas 的宽高以及 node,然后把这些数据存好,后续其他功能计算时可得靠它们 “大显身手”。

初始化 Canvas
<canvas id="canvas" type="2d" style="width: 100%; height: 100%;"></canvas>
initCanvas() {
    return new Promise((resolve, reject) => {
      const query = wx.createSelectorQuery()
      query.select('#canvas')
        .fields({ node: true, size: true })
        .exec((res) => {
          const { node, width, height } = res[0]
          node.width = width
          node.height = height
          const ctx = node.getContext('2d')
          this.setData({ 
            'canvas.target': node,
            'canvas.ctx': ctx,
            'canvas.width': width,
            'canvas.height': height
          })
          resolve()
      })
    })
  },

可能有的小伙伴要问了:已经设了宽高 100%,为啥还要再设置 node 呢🧐?这是因为不明确设置宽高属性,它就会用默认值,这就像你没给导航设定目的地,它可能就带你去 “奇怪的地方”,绘图效果自然就 “跑偏” 了。但 Canvas 尺寸又不是固定不变的,没法直接简单设置,所以初始化时这一步很关键哦。

初始化图片到Canvas

接下来把图片 “请” 到 Canvas 上。注意啦,图片得在 Canvas 正中间,还得完整地展示在可见区域,可不能按原图尺寸随意 “摆放”,所以得做些简单的 “数学运算”。

initImage() {
    return new Promise((resolve, reject) => {
      const { path, canvas: { target: canvas } } = this.data
      let image = canvas.createImage();
      image.src = path
      image.onload = () => {
        const { ctx, width: canvasWidth, height: canvasHeight } = this.data.canvas
        
        const scale = image.width / image.height
        let width = canvasWidth
        let height = canvasWidth / scale
        if (height > canvasHeight) {
          height = canvasHeight
          width = canvasHeight * scale
        }
    
        const x = (canvasWidth - width) / 2;
        const y = (canvasHeight - height) / 2;
    
        ctx.drawImage(image, x, y, width, height);
        this.setData({
          'image.target': image,
          'image.width': width,
          'image.height': height,
          'image.x': x,
          'image.y': y,
          'image.scale': scale
        })
        resolve()
      }
    })
  },

先用 Canvas 对象创建一个图片对象,再把 path “赋予” 它。等图片加载好,就能获取到图片尺寸,进而算出宽高比。因为要完整展示图片,所以图片高度最大就是 Canvas 的高度,宽度最大是 Canvas 的宽度。通过宽高比,就能算出当绘制宽度是 Canvas 宽度时对应的高度。然后把这个计算出的高度和 Canvas 高度对比,如果超了,就以 Canvas 高度为绘制高度,再算出相应宽度。

尺寸确定好后,最后算图片绘制的中心位置。以 Canvas 左上角为 (0,0),X 轴坐标就是 Canvas 宽度减去图片宽度再除以 2,Y 轴坐标同理。最后调用drawImage方法,图片就 “乖乖” 初始化绘制好啦。别忘了把图片相关数据存起来,后续功能还得靠它们 “帮忙” 呢。

绘制裁剪区域

再接着就是绘制裁剪区域,从 UI 图能看出,裁剪区域是正中心的一个长方形。咱先确定好这个长方形尺寸并存起来,然后利用这个尺寸在 Canvas 四周画上四个黑色遮罩长方形,这招就像 “围点打援”,简单又有效😎。

// 裁剪区域的相关数据
crop: {
  scale: 5 / 7,
  width: 200,
  height: 0,
  x: 0,
  y: 0,
},
// 裁剪区域尺寸等信息
getCropInfo() {
    const {
      canvas: { width: canvasWidth, height: canvasHeight },
      crop: { width: cropWidth, scale: cropScale }
    } = this.data
    const result = {
      scale: cropScale,
      width: cropWidth,
      height: cropWidth / cropScale,
      x: (canvasWidth - cropWidth) / 2,
      y: (canvasHeight - cropWidth / cropScale) / 2
    }
    this.setData({
      'crop.height': cropWidth / cropScale,
      'crop.x': (canvasWidth - cropWidth) / 2,
      'crop.y': (canvasHeight - cropWidth / cropScale) / 2,
    })
    return result
  },
  // 绘制裁剪区域
  drawCrop() {
    const { ctx, width: canvasWidth, height: canvasHeight } = this.data.canvas
    const { width: cropWidth, height: cropHeight, x, y} = this.data.crop
    // 该方式会出现 一次绘制成功一次绘制不成功的情况
    // ctx.rect(0, 0, canvasWidth, canvasHeight);
    // ctx.rect(x, y, cropWidth, cropHeight);
    // ctx.fillStyle = 'rgba(0, 0, 0, 0.7)';
    // ctx.fill('evenodd');

    ctx.rect(0, 0, canvasWidth, y);
    ctx.rect(0, y + cropHeight, canvasWidth, y);
    ctx.rect(0, y, x, cropHeight);
    ctx.rect(x + cropWidth, y, x, cropHeight);
    ctx.fillStyle = 'rgba(0, 0, 0, 0.7)';
    ctx.fill()
  },

到这儿,基本雏形已经有啦。不过关于拖拽、缩放这些更精彩的操作,由于篇幅限制,咱下篇再继续 “揭秘”😜。

在这里插入图片描述

🦀🦀感谢看官看到这里,如果觉得文章不错的话🙌,点个关注不迷路⭐。
诚邀您加入我的微信技术交流群🎉,群里都是志同道合的开发者👨‍💻,大家能一起交流分享摸鱼🐟。期待与您在群里相见🚀,咱们携手在开发路上共同进步✨ !
👉点我

感谢各位大侠一路相伴,实在感激! 不瞒您说,在下还有几个开源项目 📦,它们就像精心培育的幼苗 🌱,急需您的浇灌。要是您瞧着还不错,麻烦动动手指,给它们点亮几颗 Star ⭐,您的支持就是它们成长的最大动力,在此谢过各位大侠啦!

  • Nova UI组件库:https://github.com/gmingchen/nova-ui
  • 基于 Vue3 + Element-plus 管理后台基础功能框架
  • 预览:https://admin.gumingchen.icu
    • Github:https://github.com/gmingchen/agile-admin
    • Gitee:https://gitee.com/shychen/agile-admin
    • 基础版后端:https://github.com/gmingchen/java-spring-boot-admin
    • 文档:http://admin.gumingchen.icu/doc/
  • 基于 Vue3 + Element-plus + websocket 即时聊天系统
    • 预览:https://chatterbox.gumingchen.icu/
    • Github:https://github.com/gmingchen/chatterbox
    • Gitee:https://gitee.com/shychen/chatterbox
  • 基于 node 开发的后端服务:https://github.com/gmingchen/node-server

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2328915.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于VMware的Cent OS Stream 8安装与配置及远程连接软件的介绍

1.VMware Workstation 简介&#xff1a; VMware Workstation&#xff08;中文名“威睿工作站”&#xff09;是一款功能强大的桌面虚拟计算机软件&#xff0c;提供用户可在单一的桌面上同时运行不同的操作系统&#xff0c;和进行开发、测试 、部署新的应用程序的最佳解决方案。…

Ubuntu环境基于Ollama部署DeepSeek+Open-Webui实现本地部署大模型-无脑部署

Ollama介绍 Ollama是一款简单好用的模型部署工具,不仅可以部署DeepSeek,市面上开源模型大部分都可以一键部署,这里以DeepSeek为例 官网 DeepSeek 版本硬件要求 安装Ollama 环境 sudo apt update sudo apt install curl sudo apt install lsof1.命令一键安装 在官网点击…

goto在Java中的用法

说明&#xff1a;goto 在一些编程语言&#xff08;如C语言&#xff09;中&#xff0c;是用来表示跳转的&#xff0c;即代码执行到此处跳转到对应位置继续执行。 举例 举个例子&#xff0c;如下&#xff0c;是一个双层嵌套循环。现在我需要代码在内层循环符合某条件时跳出双层…

Vue3+Vite+TypeScript+Element Plus开发-03.主页设计与router配置

系列文档目录 Vue3ViteTypeScript安装 Element Plus安装与配置 主页设计与router配置 静态菜单设计 Pinia引入 文章目录 目录 系列文档目录 文章目录 前言 一、主页设计 二、配置代替别名 三、配置router 四、运行效果 五、参考文献 前言 本文将重点介绍如何使用…

无限滚动(Infinite Scroll)页面谷歌不收录!必须改回分页吗?

近三年&#xff0c;全球超过58%的网站采用无限滚动设计&#xff08;数据来源&#xff1a;PageTraffic 2023&#xff09; 谷歌官方数据显示&#xff0c;动态加载内容的索引失败率高达73%&#xff08;Google Webmaster Report 2022&#xff09;&#xff0c;而采用纯无限滚动的页…

Git相关笔记1 - 本地文件上传远程仓库

Git相关笔记 目录 Git相关笔记Git上传相关文件第一步创建一个仓库&#xff1a;第二步本地创建空文件夹&#xff1a;第三步开始在gitbush上传文件&#xff1a;解决外网网络连接的问题&#xff1a;中文文件的编码问题&#xff1a;参考资料 Git上传相关文件 第一步创建一个仓库&a…

如何计算财富自由所需要的价格?

写在前面&#xff1a;​【财富自由计算器】已上线&#xff0c;快算算财富自由要多少​ 多少钱&#xff0c;才能实现你的财富梦想&#xff1f; 需要多少&#xff0c;才能实现财务安全、财务独立&#xff0c;甚至财务自由&#xff1f; 看到结尾&#xff0c;你会很清楚地看到&…

thinkphp每条一级栏目中可自定义添加多条二级栏目,每条二级栏目包含多个字段信息

小程序客户端需要展示团购详情这种结构的内容,后台会新增多条套餐,每条套餐可以新增多条菜品信息,每条菜品信息包含菜品名称,价格,份数等字段信息,类似于购物网的商品多规格属性,数据表中以json类型存储,手写了一个后台添加和编辑的demo 添加页面 编辑页面(json数据…

深入解析ARM与RISC-V架构的Bring-up核心流程

深入解析ARM与RISC-V架构的Bring-up核心流程 作者&#xff1a;嵌入式架构探索者 | 2023年10月 引言 在嵌入式开发中&#xff0c;处理器的Bring-up&#xff08;启动初始化&#xff09;是系统运行的第一道门槛。ARM和RISC-V作为两大主流架构&#xff0c;其Bring-up流程既有共性…

Lettuce与Springboot集成使用

一、Lettuce核心优势与Spring Boot集成背景 Lettuce特性 基于Netty的非阻塞I/O模型&#xff0c;支持同步/异步/响应式编程线程安全&#xff1a;共享单连接实现多线程并发操作&#xff0c;性能衰减低原生支持Redis集群、哨兵、主从架构&#xff0c;自动重连机制保障高可用Spring…

《系统分析师-基础篇-1-6章总结》

第1章 绪论 系统分析师角色 职责&#xff1a;需求分析、系统设计、项目管理、技术协调。 能力要求&#xff1a;技术深度&#xff08;架构设计、开发方法&#xff09; 业务理解&#xff08;企业流程、行业知识&#xff09; 沟通能力。 系统开发生命周期 传统模型&#xf…

【YOLO系列(V5-V12)通用数据集-电梯内电动车检测数据集】

YOLO格式的电梯内电动车检测数据集&#xff0c;适用于YOLOv5-v11所有版本&#xff0c;可以用于本科毕设、发paper、做课设等等&#xff0c;有需要的在这里获取&#xff1a; 电梯内电动车检测数据集 数据集专栏地址&#xff1a;https://blog.csdn.net/qq_41304809/category_1290…

算法题(114):矩阵距离

审题&#xff1a; 本题需要我们找出所有0距离最近的1的曼哈顿距离 思路&#xff1a; 方法一&#xff1a;多源bfs 分析曼哈顿距离&#xff1a; 求法1&#xff1a;公式法&#xff0c;带入题目公式&#xff0c;利用|x1-x2||y1-y2|求出 求法2&#xff1a;曼哈顿距离就是最短距离 本…

0102-web架构网站搭建-基础入门-网络安全

文章目录 1. 常规2 站库分离3 前后端分离4 集成环境5 docker6 分配站结语 1. 常规 结构&#xff1a;源码数据都在同服务器 影响&#xff1a;无&#xff0c;常规安全测试手法 2 站库分离 结构&#xff1a;源码和数据库不在同服务器 存储&#xff1a;其他服务器上数据库或者…

Linux系统编程:进程管理、内存对比与树莓派应用

一、认识进程和线程&#xff0c;在Linux系统下查看系统中各进程的编号pid并终止一个进程pid 1.进程和线程 ​​进程​​&#xff1a;操作系统分配资源&#xff08;如内存、CPU时间片&#xff09;的基本单位。每个进程有独立的内存空间&#xff0c;进程间通信需要较复杂的机制…

ue5 仿鬼泣5魂类游戏角色和敌人没有碰撞

UE5系列文章目录 文章目录 UE5系列文章目录前言一、问题原因二、设置碰撞2.读入数据 总结 前言 ue5 仿鬼泣5魂类游戏角色和敌人没有碰撞 一、问题原因 在UE5中&#xff0c;角色和敌人没有碰撞可能是由多种原因导致的&#xff0c;以下是一些可能的原因及解决方法&#xff1a…

基于Flask的MBA考生成绩查询系统设计与实现

基于Flask的MBA考生成绩查询系统设计与实现 序言 2024年吉林大学MBA在职研究生考试成绩公布后&#xff0c;考生收到的成绩单为PDF格式文档。为方便考生快速查询个人成绩及排名信息&#xff0c;笔者基于Python Flask框架开发了本查询系统。该系统支持关键词模糊查询、序号范围…

DHCP之报文格式

字段说明&#xff1a; op (op code): 表示报文的类型&#xff0c;取值为 1 或 2&#xff0c;含义如下 1:客户端请求报 2:服务器响应报文 Secs (seconds):由客户端填充&#xff0c;表示从客户端开始获得 IP 地址或 IP 地址续借后所使用了的秒数&#xff0c;缺省值为 3600s。 F…

React 文件上传新玩法:Aliyun OSS 加持的智能上传组件

文件上传是前端开发中的“老朋友”&#xff0c;但如何让它既简单又强大&#xff0c;还能无缝对接云端存储&#xff1f;今天&#xff0c;我要带你认识一个超酷的 React 组件 AliUploader&#xff0c;它不仅支持拖拽上传、批量编辑和文件排序&#xff0c;还直接把文件传到 Aliyun…

群体智能优化算法-变色龙优化算法(Chameleon Swarm Algorithm, CSA,含Matlab源代码)

摘要 变色龙优化算法&#xff08;Chameleon Swarm Algorithm, CSA&#xff09;是一种受变色龙行为启发的群体智能优化算法。该算法模拟了变色龙在自然界中通过变换颜色来适应环境的能力&#xff0c;以此为基础&#xff0c;设计了一个适应性强、搜索能力广泛的优化算法。CSA 通…