canvas实现画布拖拽效果 适配Uniapp和Vue (开箱即用)

需求:我司是做AIGC项目最近和地铁项目有关需要实现海报效果图,并且需要使用画布拖拽和修改上传删除等等功能 当时连续加班花了10个工作日搓出来 实现挺简单的但是Canvas数据处理还是挺麻烦的  大概功能如图下 

首先我们需要引入Fabric.js 这个库封装好了原生的Canvas的Api使用uniapp和vue项目

Fabric.js能做的事情
  • 在Canvas上创建、填充图形(包括图片、文字、规则图形和复杂路径组成图形)。
  • 给图形填充渐变颜色。
  • 组合图形(包括组合图形、图形文字、图片等)。
  • 设置图形动画集用户交互。
  • 生成JSON, SVG数据等。
  • 生成Canvas对象自带拖拉拽功能。

 步骤一:引入Fabric.js库

npm i fabric --save

步骤二:安装less或者sass因为需要更改canvas样式需要用到deep(使用sass注意更改style的lang)

npm i less 
npm i less-loader

步骤三:创建DOM节点 uniapp和纯vue项目的canvas创建是不一样的 (巨坑)

Uniapp的canvas外层会嵌套uniapp-canvas会阻碍到Fabric.js读取但是纯vue项目则不会

因为Canvas代码量比较多 我这边把坑就额外标记出来 建议复制到vscode查看更佳 (下列代码复制即可使用 亲测有效 一定要按照步骤来)

<template>
	<!-- 父级节点 -->
	<view id="card-canvas-content">
	</view>
</template>

<script>
import {
	fabric
} from 'fabric'; // 引入库
let canvas = null // 最好设置成全局的不要定义到data里面
export default {
	data() {
		return {
		}
	},
	mounted() {
		this.createCanvasDom('card-canvas') // 初始化节点
		this.init() //初始化画布
	},
	methods: {
		// 创建画布
		createCanvasDom(elID) {
			var cardContent = document.getElementById('card-canvas-content'); //选取父节点
			var cardCanvas = document.createElement('canvas'); // 创建canvas标签
			cardCanvas.setAttribute('id', elID); // 设置canvas的ID
			cardContent.appendChild(cardCanvas); // 插入canvas到父节点中
		},
		// 初始化画布
		init() {
			// 画布初始配置
			canvas = new fabric.Canvas('card-canvas', { // 获取画布节点传入 初始化节点ID
				width: 200, // 后续可以更改
				height: 200,
			});

		}
	}
}
</script>
<style lang="less" scoped>
/deep/ #card-canvas{
	border: 1px solid red ;
}
</style>

 步骤四:canvas基础配置(这些配置是可选的可以根据项目情况自行配置)

// 初始化画布
		init() {
			// 画布初始配置
			canvas = new fabric.Canvas('card-canvas', { // 获取画布节点传入 初始化节点ID
				width: 200, // 后续可以更改
				height: 200,
			});
			    fabric.Object.prototype.padding = 10;
				fabric.Object.prototype.controls.mtr.withConnection = false;
				// 解决动态生成文字模糊问题
				fabric.Object.prototype.objectCaching = false;
				fabric.Object.prototype.borderColor = 'dodgerblue';
				// 修改控制点的形状,默认为`rect`矩形,可选的值还有`circle`圆形
				fabric.Object.prototype.cornerStyle = 'circle';
				// 修改控制点的填充色为白色
				fabric.Object.prototype.cornerColor = 'white';
				// 修改控制点的大小为10px
				fabric.Object.prototype.cornerSize = 10;
				// 设置控制点不透明,即可以盖住其下的控制线
				fabric.Object.prototype.transparentCorners = false;
				// 修改控制点的边框颜色为`gray`灰色
				fabric.Object.prototype.cornerStrokeColor = 'gray';
				// 单独修改旋转控制点距离主体的纵向距离为-20px
				fabric.Object.prototype.controls.mtr.offsetY = -20;
				// 单独修改旋转控制点,光标移动到该点上时的样式为`pointer`,一个手的形状
				fabric.Object.prototype.controls.mtr.cursorStyle = 'pointer';

		}

第五步:初始化数据建议抽离一个方法出去避免都写在init里面过于冗杂 (在init里面调用放置末尾

init() {
     this.initDate()
}
// 初始化数据
		initDate() {
			//更改画布后续宽高 可以根据后台的图片动态设置canvas大小
			canvas.setDimensions({
				width: this.width,
				height: this.height
			});
			// fabric内置的方法 用于重新渲染canvas 在字体以及大小更换 图片替换都需要调用此方法 否则canvas不会更新新数据
			canvas.renderAll();
			this.$nextTick(() => {
				// canvas渲染完成之后最后就会执行一般是处理loading效果的
			});
		}

步骤六:导入背景图 (巨坑 当时我研究了贼久 一直报错图片污染导致后续图片无法导出canvas)  

我们需要把远程的背景图片转成base64的格式不然导出canvas会报错 这也是跨域 网上有很多解决方案都没用

代码比较长我就抽离出去写注释 这些方法统一在initDate里面调用 

// 图片转码
			getURLBase64(url) {
				//将远程图片下载本地成为base64图片
				let _this = this;
				return new Promise((resolve, reject) => {
					var xhr = new XMLHttpRequest();
					xhr.open('get', url, true);
					xhr.responseType = 'blob';
					xhr.onload = function() {
						if (this.status === 200) {
							var blob = this.response;
							var fileReader = new FileReader();
							fileReader.onloadend = function(e) {
								var result = e.target.result;
								_this.imageUrl = result;
								resolve(result);
							};
							fileReader.readAsDataURL(blob);
						}
					};
					xhr.onerror = function() {
						reject();
					};
					xhr.send();
				});
			},

 设置Canvas背景图片

// 设置/修改画布背景图  bgUrl:转码后的背景图片 canvasDOM:canvas的实例变量名
		setBackgroundImage(bgUrl, canvasDOM) {
			canvasDOM.setBackgroundImage(bgUrl, canvasDOM.renderAll.bind(canvasDOM), { // canvas实例变量
				originX: 'left',
				originY: 'top',
				left: 0,
				top: 0,
				level: 1, // 层级
				bgUrl // 路径
			});
		},

 使用:

async initDate() {
			//更改画布后续宽高 可以根据后台的图片动态设置canvas大小
			canvas.setDimensions({
				width: this.width,
				height: this.height
			});
			// fabric内置的方法 用于重新渲染canvas 在字体以及大小更换 图片替换都需要调用此方法 否则canvas不会更新新数据
			canvas.renderAll();
			// 背景地址 自己随便填一个线上的即可
			let bgUrl = await this.getURLBase64('https://img1.baidu.com/it/u=395218774,2205974617&fm=253&app=138&size=w931&n=0&f=GIF&fmt=auto?sec=1718730000&t=15d143401501c32f552f4157563acaf4');
			// 设置背景图片
			this.setBackgroundImage(bgUrl, canvas)
		},

 此时是页面呈现的样式

 

步骤六:插入文字

// 插入文字 text:文本 canvasDOM:canvas的实例变量名
		setTextFont(text,canvasDOM) {
			const textDOM = new fabric.Textbox(text, {
				top: 0, // 位置
				left: 0,
				fontSize: 19, // 文字大小
				fill: 'red', // 颜色
				level: 0, // 层级
				fontFamily: '思源黑体 CN Heavy',
				splitByGrapheme: true
			});
			canvasDOM.add(textDOM); //添加文字
			canvasDOM.bringToFront(textDOM); //优先级
		}

使用

async initDate() {
			//更改画布后续宽高 可以根据后台的图片动态设置canvas大小
			canvas.setDimensions({
				width: this.width,
				height: this.height
			});
			// fabric内置的方法 用于重新渲染canvas 在字体以及大小更换 图片替换都需要调用此方法 否则canvas不会更新新数据
			canvas.renderAll();
			// 背景地址 自己随便填一个线上的即可
			let bgUrl = await this.getURLBase64('https://img1.baidu.com/it/u=395218774,2205974617&fm=253&app=138&size=w931&n=0&f=GIF&fmt=auto?sec=1718730000&t=15d143401501c32f552f4157563acaf4');
			// 设置背景图片
			this.setBackgroundImage(bgUrl, canvas)
			// 插入文本
			this.setTextFont('这是一行文本',canvas)
		},

 页面效果 此时页面的文字是可以进行拖拽等操作的

 

(内容有点多 先更新三分之一 后面再抽空更新)

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Apifox 快速入门教程

访问示例项目​ 可访问Apifox官网&#xff0c;下载并打开 Apifox 后&#xff0c;你将会看到由系统自动创建的“示例团队”&#xff0c;其中内含一个“示例项目”。 项目中自动生成了与宠物商店有关的数条接口。 手动新建接口​ 新建接口是开发者们最常用的功能之一。Apifox 能…

硫化物固态电解质在全固态锂电池制造领域发展潜力大

硫化物固态电解质在全固态锂电池制造领域发展潜力大 固态电解质主要包括氧化物、硫化物、聚合物等类型。氧化物固态电解质由于研发难度相对较低&#xff0c;是目前主流技术路线。硫化物固态电解质研发难度较高&#xff0c;但性能优异&#xff0c;特别适合制造全固态锂电池&…

小程序wx.uploadFile异步问题

问题&#xff1a;小程序上传文件后我需要后端返回的一个值&#xff0c;但这个值总是在最后面导致需要这个值的方法总是报错&#xff0c;打印测试后发现这它是异步的。但直接使用 await来等待也不行。 uploadImg.wxml <view class"upload-wrap"><view clas…

黑神话悟空-吉吉国王版本【抢先版】

在中国的游戏市场中&#xff0c;一款名为“黑神话悟空”的游戏引起了广泛的关注。这款游戏以中国传统的神话故事“西游记”为背景&#xff0c;创造了一个令人震撼的虚拟世界。今天&#xff0c;我们要来介绍的是这款游戏的一种特殊版本&#xff0c;那就是吉吉国王版本。 在吉吉国…

邮件推送服务商有哪些核心功能?怎么选择?

邮件推送服务商支持哪些营销工具&#xff1f;推送性能如何评估&#xff1f; 邮件推送服务商的核心功能可以帮助企业更高效地管理和优化其电子邮件营销活动&#xff0c;从而提升客户参与度和转化率。AokSend将详细介绍邮件推送服务商的一些核心功能。 邮件推送服务商&#xff…

鸿蒙仓颉编程语音来了,ArkTs语言危矣?

鸿蒙仓颉编程语言来了&#xff0c;请允许我哭会~~呜呜呜~~我的arkts啊 仓颉编程语言文档地址文档中心 鸿蒙直播大会开始一个小时了&#xff0c;地址华为开发者大会&#xff08;HDC 2024&#xff09;主题演讲 同时api12 不再是秘密了 各位&#xff01;公开啦 api12 公开地址…

如何开启Claude 3的Artifacts功能以及如何注册Claude3

就很突然&#xff0c;Claude 3.5&#xff0c;它来了&#xff01; Anthropic发布3.5系列第一个版本Claude 3.5 Sonnet。在多个关键指标中&#xff0c;GPT-4o几乎被吊打&#xff01; 另外Claude 3.5 Sonnet是免费的&#xff0c;提供了跟gpt-4o一样的次数。更高的速度和次数&…

maxwell源码编译安装部署

目录 1、组件环境 2、maxwell安装前提 3、maxwell安装 3.1、maxwell下载 3.1.1、最新版本下载 ​编辑 3.1.2、历史版本下载 3.2、maxwell安装 3.3、maxwell配置 3.2.1、mysql开启binlog 3.3.2、maxwell元数据配置 3.3.3、maxwell配置任务 4、maxwell部署问题 4.1、utf…

Jenkins macos 下 failed to create dmg 操作不被允许hdiutil: create failed - 操作不被允许?

解决方案&#xff1a; 打开设置&#xff0c;选择“隐私与安全”&#xff0c;选择“完全磁盘访问权限”&#xff0c;点击“”&#xff0c;选择jenkins的路径并添加。 同理&#xff0c;添加java的访问权限。

【Android】记录在自己的AMD处理器无法使用Android studio 虚拟机处理过程

文章目录 问题&#xff1a;无法在AMD平台打开Android studio 虚拟机&#xff0c;已解决平台&#xff1a;AMD 5700g系统&#xff1a;win10专业版1、在 amd平台上使用安卓虚拟机需要安装硬件加速器2、关闭win10上的系统服务 问题&#xff1a;无法在AMD平台打开Android studio 虚拟…

java 八股文最集全网站

弟弟快看-教程&#xff0c;程序员编程资料站 | DDKK.COM

关于微信小程序取消获取用户昵称的一些思考

官方说明&#xff0c;有部分小程序乱用授权&#xff0c;强迫用户提交头像和昵称。 核心是微信担心用户信息被滥用。 其一 &#xff0c;微信头像经常是本人真是照片&#xff0c;在现在人工智能算法的加持下&#xff0c;人脸数据太容易被套取。 其二&#xff0c;微信名称同理&…

使用Python发送电子邮件:轻松实现自动化沟通

哈喽&#xff0c;大家好&#xff0c;我是木头左&#xff01; 1. 为什么使用Python发送电子邮件&#xff1f; 在当今这个信息爆炸的时代&#xff0c;电子邮件已经成为了日常生活中不可或缺的一部分。无论是工作还是生活&#xff0c;都可能需要通过电子邮件与他人进行沟通。而Py…

通过InoDriverShop伺服调试软件连接汇川SV660F系列伺服的具体方法(以太网)

通过InoDriverShop伺服调试软件连接汇川SV660F系列伺服的具体方法(以太网) 具体连接伺服驱动器的步骤可参考以下内容: 启动InoDriverShop, 新建或打开工程 如下图所示,选择在线,选中SV660F图标,右侧通信类型选择“TCP—DCP”,点击下一步,同时要选择自己当前使用的网卡…

【文档+源码+调试讲解】国风彩妆网站springboot

摘 要 二十一世纪我们的社会进入了信息时代&#xff0c;信息管理系统的建立&#xff0c;大大提高了人们信息化水平。传统的管理方式对时间、地点的限制太多&#xff0c;而在线管理系统刚好能满足这些需求&#xff0c;在线管理系统突破了传统管理方式的局限性。于是本文针对这一…

如何使用 NFTScan NFT API 在 Sei 网络上开发 Web3 应用

Sei Network 是一个专为交易而设计的 Layer 1 区块链。它建立在 Cosmos SDK 上&#xff0c;使用一种称为 Tendermint BFT 的新型共识机制。不仅专攻 DeFi 领域的加密资产交易&#xff0c;更在游戏、社交媒体和 NFTs 等热门 verticals 构建了多功能区块链生态系统。Sei Network …

【盘点】8大电商选品思路,实操策略大公开!

1、以人选品 顾名思义&#xff0c;先确定想做的目标人群&#xff0c;再挖掘人群的需求。比如&#xff0c;小个子&#xff0c;这种细分市场&#xff0c;这里的人代表的是一个群体&#xff0c;可以是职业&#xff0c;可以是年龄段可以是一种称呼。如果未能明确目标市场和消费者需…

ConcurrentHashMap(应对并发问题的工具类)

并发工具类 在JDK的并发包里提供了几个非常有用的并发容器和并发工具类。供我们在多线程开发中进行使用。 5.1 ConcurrentHashMap 5.1.1 概述以及基本使用 在集合类中HashMap是比较常用的集合对象&#xff0c;但是HashMap是线程不安全的(多线程环境下可能会存在问题)。为了…

Docker之overlay2的迁移

原因 docker默认将文件及其容器放置在了系统盘的挂载区内&#xff0c;如果长期使用会发现系统挂载区被overlay2挤爆了,因此在一开始我们将其迁移在大容量外挂磁盘上,就可以避免系统盘被挤爆,放心使用. 具体操作 # 停止容器 systemctl stop docker# 修改容器配置&#xff0c…

STM32CubeMX WS2812B灯驱动

一、WS2812B 数据发送速度可达800Kbps。 数据协议采用单线归零码的通讯方式&#xff0c;像素点在上电复位以后&#xff0c;DIN端接受从控制器传输过来的数据&#xff0c;首先送过来的24bit数据被第一个像素点提取后&#xff0c;送到像素点内部的数据锁存器&#xff0c;剩余的…