前端canvas项目实战——在线图文编辑器(八):复制、删除、锁定、层叠顺序

目录

  • 前言
  • 一、效果展示
  • 二、实现步骤
    • 1. 复制
    • 2. 删除
    • 3. 锁定
    • 4. 层叠顺序
  • 三、实现过程中发现的bug
    • 1. clone方法不复制自定义属性
    • 2. 复制「锁定」状态的对象,得到的新对象也是「锁定」状态
  • 四、Show u the code
  • 后记

前言

上一篇博文中,我们细致的讲解了实现文字的加粗、斜体、下划线、删除线这些功能时,遇到的Bug以及优化点。

这篇博文是《前端canvas项目实战——在线图文编辑器》付费专栏系列博文的第八篇——复制、删除、锁定、层叠顺序,主要的内容有:

  1. 实现一组通用的功能按钮:复制、删除、锁定和层叠顺序,用户可以通过点击这些按钮来对画布中的对象进行:
  • 复制: 复制选中的对象,并将新对象添加到画布上。
  • 删除: 删除选中的对象。
  • 锁定: 使对象不可以被拖拽移动位置、不可以通过控制点来进行缩放、不可以旋转等。
  • 层叠顺序: 更改对象在z轴上的顺序,处于上层的对象会遮盖住下层的对象。

如有需要,你可以:

  • 点击这里,返回第一篇《前端canvas项目实战——在线图文编辑器(一)——左侧工具栏》
  • 点击这里,返回上一篇《前端canvas项目实战——在线图文编辑器(七):加粗、斜体、下划线、删除线(下)》

一、效果展示

  • 动手体验
    CodeSandbox会自动对代码进行编译,并提供地址以供体验代码效果
    由于CSDN的链接跳转有问题,会导致页面无法工作,请复制以下链接在浏览器打开:
    https://fjf3h6.csb.app/

  • 动态效果演示


二、实现步骤

1. 复制

「复制」是一个常用的功能。比如我们创建一个简历时,设置好了一个文本框的字体、字号、颜色等属性,此时如果想要再创建一个相同属性但文字不同的文本框,可以有以下两种实现方式:

  • 点击左侧工具栏生成一个默认的文本框,然后依次设置字体、字号、颜色等属性,然后修改文字。
  • 复制这个文本框,然后修改文字。

显然,「复制」是一个非常便捷的功能,省去了使用者很多重复的点击和操作。

前文中的动态图已经展示了这个按钮的功能,就不再做图示。以下是代码:

import store from "../modules/store";const cloneActiveObjects = () => {const {canvas} = store.getState();const activeObject = canvas.getActiveObject();const handleCloneObject = (newObject) => {// Bug点1for (const key in activeObject) {if (activeObject.hasOwnProperty(key) && !newObject.hasOwnProperty(key) && typeof activeObject[key] !== "function") {newObject.set(key, activeObject[key]);}}// Bug点2lockUnlockObject(newObject, false);canvas.add(newObject);newObject.set({left: activeObject.left + 25, top: activeObject.top + 25});canvas.setActiveObject(newObject);canvas.renderAll();};activeObject.clone((newObject) => handleCloneObject(newObject));
};

以上是复制按钮的点击事件处理方法,代码逻辑比较清晰,以下做简要的说明:

  • 获取当前选中的对象: 从中央数据仓库取得,因此使这个方法不需要入参。
  • 复制选中的对象: 通过fabric.Object原生的clone方法复制对象。
  • 复制对象之后的动作: 在回调方法handleCloneObject中,我们:
    • 首先将「原对象」中的所有属性设置给「新对象」,这里是一个Bug点,下文中会详细讲解。
    • 然后解锁「新对象」。无论「原对象」是否锁定状态,新复制出来的对象都应该是非锁定的。
    • 将「新对象」添加到画布中,并移动到「原对象」右下角一定距离(这里设置为25像素)。
    • 将「新对象」设置为画布中当前选中的对象。

2. 删除

「删除」即从画布中移除当前选中的对象,代码如下:

const deleteActiveObjects = () => {const {canvas} = store.getState();const activeObject = canvas.getActiveObject();canvas.remove(activeObject);canvas.discardActiveObject();canvas.renderAll();
};

以上是点击「删除」按钮后的事件处理方法,代码逻辑分为3个部分:

  • 获取当前选中的对象: 从中央数据仓库取得,因此使这个方法不需要入参。
  • 移除选中的对象: 通过fabric.Canvas原生的remove方法移除对象。
  • 画布丢弃当前选中对象: 调用canvas.discardActiveObject()方法,使canvas将当前选中的对象置为空,即表示当前画布中没有选中的对象

3. 锁定

「锁定」是一个逻辑上的功能,我们首先要定义,当用户点击这个按钮时,我们应该锁住哪些操作?

起初,我定义一个「被锁定的对象」是:一个除了「解锁」之外不可以进行任何编辑操作的对象。但在实现过程中,发现这样的定义不合理,且实现起来十分复杂。原因如下:

  • 不合理性: 一般意义上,「锁定」功能只是锁住一个对象的位移、缩放等操作。如果用户拿到一个别人设计好的精美的简历模板,想要通过替换其中的文字来快速制作自己的简历,那TA需要进行的操作有:

    • 逐个解锁TextboxImage等对象;
    • 修改各个对象的文本、图片等内容;
    • 锁定这些对象,避免误操作使其发生位移或缩放,影响简历的美观。

    可见这样的定义会使用户徒增「加锁」和「解锁」的操作,增加操作的复杂性。

  • 实现中的困难: 根据这种定义,当对象被锁定时,需要逐个「屏蔽」用户可以对其进行的操作,难免有遗漏,且如果有新增的操作能力,也需要同步添加「屏蔽」的能力。

基于这样的实践经验和思考,我们将「被锁定的对象」定义为:一个不能移动、不能被缩放的对象。

下面我们来实现它:

// 部分控制点可见
const _fewControlsVisible = {tl: false,tr: false,ml: true,mr: true,mt: false,mb: false,bl: false,br: false
};// 全部控制点可见
const _allControlsVisible = {tl: true,tr: true,ml: true,mr: true,mt: true,mb: true,bl: true,br: true
};// 对象的控制点可见情况
const objectControlsVisibility = {object: _allControlsVisible,rect: _allControlsVisible,circle: _allControlsVisible,activeSelection: _allControlsVisible,line: _fewControlsVisible,textbox: _fewControlsVisible,group: _fewControlsVisible
};const lockUnlockObject = (object, locked) => {object.set({lockMovementX: locked,lockMovementY: locked,lockRotation: locked,lockScalingX: locked,lockScalingY: locked,lockSkewingX: locked,lockSkewingY: locked,lockScalingFlip: locked,locked});// 根据锁定状态设置选择框的3个「自定义」控制点隐藏或显示object.setControlsVisibility({lock: locked,mtr: !locked,del: !locked});// 根据锁定状态设置选择框的8个「基础」控制点的隐藏或显示let controlsVisibility = objectControlsVisibility[object.type] || objectControlsVisibility["object"];let {tl, tr, ml, mr, mt, mb, bl, br} = controlsVisibility;object.setControlsVisibility({tl: !locked && tl,tr: !locked && tr,ml: !locked && ml,mr: !locked && mr,mt: !locked && mt,mb: !locked && mb,bl: !locked && bl,br: !locked && br});
};const lockUnlockActiveObjects = () => {const {canvas} = store.getState();const activeObject = canvas.getActiveObject();const locked = !(activeSelection.locked || false);// 设置选中的对象的锁定状态lockUnlockObject(activeObject, locked);canvas.renderAll();store.dispatch(Actions.updateActiveObjectProperty("locked", locked));
};

以上是「锁定」按钮的点击事件处理方法,代码比较多,但是结构是清晰简洁的,以下逐段进行介绍:

  • objectControlsVisibility字典: 定义了fabric.js种不同的对象类型,其选择框显示和隐藏的控制点设置,其中Line线条Textbox文本框只显示mlmr两个控制点,其他的对象都显示全部的控制点。
    具体效果如下图所示:
  • lockUnlockObject方法: 「锁定/解锁」一个对象,需要经过以下3个步骤:

    • 设置对象属性: 通过设置对象的锁定相关的属性值为truefalse,使对象可以/不可以移动、缩放、旋转、扭曲
    • 显示/隐藏3个自定义控制点: 根据对象的locked属性设置旋转、删除、锁定等3个自定义控制点隐藏或者显示。
      • lockedfalse时,显示旋转和删除,隐藏锁定;
      • lockedtrue时,隐藏旋转和删除,显示锁定。
    • 显示/隐藏8个基础控制点: 根据对象的locked属性和上述的objectControlsVisibility字典设置8个基础控制点隐藏或者显示。
      • lockedfalse时,仅显示当前对象类型可以显示的控制点,隐藏其他控制点;
      • lockedtrue时,隐藏所有8个基础控制点。

    具体效果如下图所示:

  • lockUnlockActiveObjects方法: 这个方法中获取了画布中当前选中的对象,然后调用了上述的lockUnlockObject方法来 「加锁/解锁」 这个对象。

4. 层叠顺序

「层叠顺序」也称为z-index。即除了二维画布的xy两个坐标轴外,想象有一条从屏幕里穿出,垂直于屏幕的坐标轴,称作「z轴」。

当用户在画布中创建了多个对象时,位置相近的对象间可能会互相遮挡。处在上层的对象会遮住处在下层的对象的部分或全部区域。

在画布中,默认「后创建的对象」在z轴上高于「先创建的对象」。一般情况下,我们不会一开始就想好所有对象的创建顺序,然后依次创建它们。所以需要灵活得调整对象之间的层叠顺序

那么我们来实现它:

	...const zIndexProps = {className: "none",tip: "层叠顺序",menu: {items: [{key: "toTop",icon: <VerticalLeftOutlined style={{transform: "rotate(-90deg)"}}/>,label: "移至顶层"}, {key: "up",icon: <UpOutlined/>,label: "向上一层"}, {key: "down",icon: <DownOutlined/>,label: "向下一层"}, {key: "toBottom",icon: <VerticalRightOutlined style={{transform: "rotate(-90deg)"}}/>,label: "移至底层"}],onClick: adjustActiveObjectZIndex}};return (...<SwitchValueButton {...zIndexProps}><BlockOutlined className="property-operation-img"/></SwitchValueButton>...);...const adjustActiveObjectZIndex = (selectedItem) => {const {canvas} = store.getState();const activeObject = canvas.getActiveObject();if (activeObject) {if (selectedItem?.key === "toTop") {canvas.bringToFront(activeObject);} else if (selectedItem?.key === "up") {canvas.bringForward(activeObject);} else if (selectedItem?.key === "down")  {canvas.sendBackwards(activeObject);} else {canvas.sendToBack(activeObject);}canvas.renderAll();}};

代码逻辑很清晰,下面我们分为两个部分来说明:

  • 视图部分: 这里和其他的按钮略有不同,点击后会弹出一个下拉菜单。 我们传入了一个菜单项列表menu,最后的onClick: adjustActiveObjectZIndex表示,当菜单项被点击时,响应的逻辑由adjustActiveObjectZIndex方法处理。
  • 逻辑部分: adjustActiveObjectZIndex方法的实现也很简洁,根据用户点击的操作项的key来执行不同的操作
    • toTop: 置于顶层,调用canvasbringToFront方法
    • up: 向上一层,调用canvasbringForward方法
    • down: 向下一层,调用canvassendBackwards方法
    • toBottom: 置于底层,调用canvassendToBack方法

三、实现过程中发现的bug

还记得前文中的handleCloneObject方法吗?这个方法在我们实现复制功能时,在新对象复制完成的回调方法中:

	...const handleCloneObject = (newObject) => {// Bug点1:clone方法不复制自定义属性for (const key in activeObject) {if (activeObject.hasOwnProperty(key) &&!newObject.hasOwnProperty(key) &&typeof activeObject[key] !== "function") {newObject.set(key, activeObject[key]);}}// Bug点2:复制「锁定」状态的对象,得到的新对象也是「锁定」状态lockUnlockObject(newObject, false);...};

这段代码包含了两个问题及其解决方案:

1. clone方法不复制自定义属性

在实现的过程中,我们对部分对象的属性进行了扩充。例如:

  • fabric.Line线条对象的startPointTypeendPointType: 为了实现线条的两个端点,我们为它加上了这两个额外的属性。fabric.js原生的clone方法只会将默认的属性复制到新对象中,这些我们后添加上去的属性则不处理。
  • fabric.Object所有对象的locked是否锁定属性: 同理,fabric.js原生的clone方法也不会把这个属性自动复制给「新对象」。

因此,if判断条件的意思就是如果一个属性满足 「旧对象」有,「新对象」没有,且不是function,就把这个属性赋值给「新对象」。

2. 复制「锁定」状态的对象,得到的新对象也是「锁定」状态

在「复制」的代码中,我们用以下方法限制了「新对象」的位置在「旧对象」右边25像素,下边25像素:

    newObject.set({left: activeObject.left + 25, top: activeObject.top + 25});

一般情况下,用户会在复制出「新对象」后把它拖动到自己想要的位置。但如果「旧对象」是「锁定」状态,我们就需要在复制完成后,调用lockUnlockObject方法对「新对象」进行「解锁」。


四、Show u the code

按照惯例,本节的完整代码我也托管在了CodeSandbox中,点击前往,查看完整代码


后记

这篇博文中,我们实现一组通用的功能按钮:复制、删除、锁定和层叠顺序。虽然是几个不算复杂的功能,但也有很多细节方面的问题值得考量。

有了这些按钮,会使用户在使用我们的编辑器时更加快捷、稳定得完成自己的需要。

如有需要,你可以:

  • 点击这里,返回第一篇《前端canvas项目实战——在线图文编辑器(一)——左侧工具栏》
  • 点击这里,返回上一篇《前端canvas项目实战——在线图文编辑器(七):加粗、斜体、下划线、删除线(下)》

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

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

相关文章

Oracle 使用维进行查询重写

Oracle 使用维进行查询重写 conn / as sysdba alter user sh account unlock identified by sh; conn sh/sh query_rewrite_integrity TRUSTED --物化视图的定义 select query from user_mviews where MVIEW_NAMECAL_MONTH_SALES_MV;CREATE MATERIALIZED VIEW cal_month_s…

VPDN(L2TP、PPTP)

1、虚拟专用拨号网络 远程接入VPN&#xff0c;客户端可以是PC机 技术&#xff1a;L2TP、PPTP 术语&#xff1a;LAC&#xff1a;L2TP的访问集中器 --- 提供用户的接入 LNS&#xff1a;L2TP的网络服务器 --- 提供L2TP服务的服务器 2、技术 1&#xff09;PPTP 点对点隧道…

DFS(排列数字、飞机降落、选数、自然数的拆分)

注&#xff1a;1.首先要知道退出条件 2.还原现场 典型&#xff1a;全排列 题目1&#xff1a; 代码&#xff1a; #include<bits/stdc.h> using namespace std; int a[1005],p[1005],v[1005]; int n; void dfs(int x) {//此次dfs结束条件,即搜到底 if(xn1){for(int i1;i&…

C语言自定义类型变量——枚举(enum)

一.枚举的定义和声明 字面意思&#xff0c;枚举就是一一列举&#xff0c;把可能的取值一一列举&#xff0c;在我们现实生活中有许多可以列举的事物&#xff0c;例如&#xff1a;一周七天&#xff0c;一年四季&#xff0c;性别&#xff0c;月份&#xff0c;三原色等等。当我们需…

栈的详解和例题(力扣有效括号)

感谢各位大佬的光临&#xff0c;希望和大家一起进步&#xff0c;望得到你的三连&#xff0c;互三支持&#xff0c;一起进步 个人主页&#xff1a;LaNzikinh-CSDN博客 收入专栏:初阶数据结构_LaNzikinh篮子的博客-CSDN博客 文章目录 前言一.什么是栈二.栈的实现三.例题&#xff…

golang设计模式图解——命令模式

设计模式 GoF提出的设计模式有23个&#xff0c;包括&#xff1a; &#xff08;1&#xff09;创建型(Creational)模式&#xff1a;如何创建对象&#xff1b; &#xff08;2&#xff09;结构型(Structural )模式&#xff1a;如何实现类或对象的组合&#xff1b; &#xff08;3&a…

leetcode热题100.跳跃游戏2

Problem: 45. 跳跃游戏 II 文章目录 题目思路复杂度Code 题目 给定一个长度为 n 的 0 索引整数数组 nums。初始位置为 nums[0]。 每个元素 nums[i] 表示从索引 i 向前跳转的最大长度。换句话说&#xff0c;如果你在 nums[i] 处&#xff0c;你可以跳转到任意 nums[i j] 处: …

webpack-前置知识

前置知识-node的内置模块path path模块用于对路径和文件进行处理&#xff0c; 从路径中获取信息 dirname: 获取文件的父文件夹。 basename:获取文件名。 extname: 获取文件拓展名。 const path require("path")const fileName "C://test/a/b/c.txt"//.t…

安全左移是什么,如何为网络安全建设及运营带来更多可能性

长久以来&#xff0c;网络安全技术产品和市场需求都聚焦于在“右侧”防护&#xff0c;即在各种系统、业务已经投入使用的网络环境外围或边界&#xff0c;检测进出的流量、行为等是不是存在风险&#xff0c;并对其进行管控或调整。 然而事实上&#xff0c;安全风险不仅是“跑”…

如何保护大模型API安全

大模型的崛起正在改变着我们对机器学习和人工智能的理解&#xff0c;它们不仅提供了令人惊叹的预测和分析能力&#xff0c;还在各行各业的应用中发挥着重要作用。通过提供 API&#xff0c;用户无需了解底层实现细节&#xff0c;使大型模型能够更好地与用户和应用程序进行交互&a…

电商技术揭秘七:搜索引擎中的SEO关键词策略与内容优化技术

文章目录 引言一、关键词策略1.1 关键词研究与选择1. 确定目标受众2. 使用关键词研究工具3. 分析搜索量和竞争程度4. 考虑长尾关键词5. 关键词的商业意图6. 创建关键词列表7. 持续监控和调整 1.2 关键词布局与密度1. 关键词自然分布2. 标题标签的使用3. 首次段落的重要性4. 关键…

【opencv】示例-asift.cpp 对两张图片之间进行仿射特征比对

#include <opencv2/core.hpp> // 包含OpenCV核心功能的头文件 #include <opencv2/imgproc.hpp> // 包含OpenCV图像处理功能的头文件 #include <opencv2/features2d.hpp> // 包含OpenCV特征检测相关功能的头文件 #include <opencv2/highgui.hpp> // 包含…

sqlmap(五)

一、进行文件读写操作 1.1 前提条件 高权限 目录有读写权限 secure_file_priv " " 1.2 测试目标 第一步&#xff1a;用抓包的方式获取请求测试站点的数据包 可以使用Burpsuite 第二步&#xff1a;将抓到的数据包&#xff0c;保存到sqlmap目录下的a.txt 第三步&am…

从FasterTransformer源码解读开始了解大模型(1.1)一个decoder-only的模型长啥样

从FasterTransformer源码解读开始了解大模型&#xff08;1.1&#xff09;一个decoder-only的模型长啥样 写在前面的话 对于一个没有接触过LLM的初学者来说&#xff0c;如果想要了解一个大模型的推理框架&#xff0c;首先应该知道大模型整个的工作原理是怎样的&#xff0c;知道…

了解自动化机器学习 AutoML

&#x1f349; CSDN 叶庭云&#xff1a;https://yetingyun.blog.csdn.net/ 自动化机器学习&#xff08;AutoML&#xff09;概述 自动化机器学习&#xff08;AutoML&#xff09;旨在自动化机器学习模型的开发流程&#xff0c;通过简化或去除需要专业知识的复杂步骤&#xff0c;…

CSS面试题常用知识总结day03

大家好我是没钱的君子下流坯&#xff0c;用自己的话解释自己的知识 前端行业下坡路&#xff0c;甚至可说前端已死&#xff0c;我还想在前段行业在干下去&#xff0c;所以从新开始储备自己的知识。 从CSS——>Javascript——>VUE2——>Vuex、VueRouter、webpack——>…

Stale Diffusion、Drag Your Noise、PhysReaction、CityGaussian

本文首发于公众号&#xff1a;机器感知 Stale Diffusion、Drag Your Noise、PhysReaction、CityGaussian Drag Your Noise: Interactive Point-based Editing via Diffusion Semantic Propagation Point-based interactive editing serves as an essential tool to compleme…

Nuxt 3 项目中配置 Tailwind CSS

官方文档&#xff1a;https://www.tailwindcss.cn/docs/guides/nuxtjs#standard 安装 Tailwind CSS 及其相关依赖 执行如下命令&#xff0c;在 Nuxt 项目中安装 Tailwind CSS 及其相关依赖 npm install -D tailwindcss postcss autoprefixerpnpm install -D tailwindcss post…

【cpp】快速排序优化

标题&#xff1a;【cpp】快速排序 水墨不写bug 正文开始&#xff1a; 快速排序的局限性&#xff1a; 虽然快速排序是一种高效的排序算法&#xff0c;但也存在一些局限性&#xff1a; 最坏情况下的时间复杂度&#xff1a;如果选择的基准元素不合适&#xff0c;或者数组中存在大…

Netty 3 - 组件和设计

这里将回顾我们之前章节讲到过的主要概念和组件。 1 Channel 、EventLoop和ChannelFuture Channel —— Socket;EventLoop —— 控制流、多线程处理、并发;ChannelFuture —— 异步通知。 1.1 Channel 接口 基本的I/O操作&#xff08;bind()、connect()、read()和write()&a…