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

news2025/1/13 10:27:33

目录

  • 前言
  • 一、效果展示
  • 二、实现步骤
    • 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点1
        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);
        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.coloradmin.cn/o/1573109.html

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

相关文章

高精度端到端在线校准环视相机和LIDAR(精度0.2度内!无需训练数据)

高精度端到端在线校准环视相机和LIDAR&#xff08;精度0.2度内&#xff01;无需训练数据&#xff09; 附赠自动驾驶学习资料和量产经验&#xff1a;链接 写在前面 在自动驾驶车辆的使用寿命内&#xff0c;传感器外参校准会因振动、温度和碰撞等环境因素而发生变化。即使是看似…

智过网:非安全专业能否报考注安?哪些专业可以报考?

近年来&#xff0c;随着社会对安全生产管理的日益重视&#xff0c;注册安全工程师&#xff08;简称注安&#xff09;这一职业逐渐受到广大从业人员的青睐。然而&#xff0c;对于许多非安全专业的朋友来说&#xff0c;他们可能会困惑&#xff1a;非安全专业是否可以报考注安&…

微软推出GPT-4 Turbo优先使用权:Copilot for Microsoft 365商业用户享受无限制对话及增强图像生成能力

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

thinkphp6入门(22)-- 如何下载文件

假设在public/uploads文件夹下有一个文件test.xlsx 在前端页面添加下载链接&#xff0c;用户点击该链接即可下载对应的文件。 <a href"xxxxxxx/downloadFile">下载文件</a> 2. 在后端控制器方法中&#xff0c;我们需要获取要下载的文件路径&#xff0…

Prometheus+grafana环境搭建Docker服务(docker+二进制两种方式安装)(八)

由于所有组件写一篇幅过长&#xff0c;所以每个组件分一篇方便查看&#xff0c;前七篇链接如下 Prometheusgrafana环境搭建方法及流程两种方式(docker和源码包)(一)-CSDN博客 Prometheusgrafana环境搭建rabbitmq(docker二进制两种方式安装)(二)-CSDN博客 Prometheusgrafana环…

软考115-上午题-【计算机网络】-HTML

一、真题 真题1&#xff1a; alink属性表示一个链接的当前激活状态的颜色&#xff0c;即用户正在点击或已经点击的链接的颜色。 vlink属性用于设定已访问过的超链接文本的显示颜色&#xff0c;即用户已经点击过并且已经访问过的链接的颜色。 真题2&#xff1a; <table bord…

智能感应门改造工程

今天记录一下物联网专业学的工程步骤及实施过程 智能感应门改造工程 1 规划设计1.1 项目设备清单1.2项目接线图 软件设计信号流 设备安装与调试工程函数 验收 1 规划设计 1.1 项目设备清单 1.2项目接线图 软件设计 信号流 设备安装与调试 工程函数 工程界面: using System; …

【C++算法竞赛 · 图论】图论基础

前言 图论基础 图的相关概念 图的定义 图的分类 按数量分类&#xff1a; 按边的类型分类&#xff1a; 边权 简单图 度 路径 连通 无向图 有向图 图的存储 方法概述 代码 复杂度 前言 图论&#xff08;Graph theory&#xff09;&#xff0c;是 OI 中的一样很大…

【Rust】生命周期

Rust 生命周期机制是与所有权机制同等重要的资源管理机制。 之所以引入这个概念主要是应对复杂类型系统中资源管理的问题。 引用是对待复杂类型时必不可少的机制&#xff0c;毕竟复杂类型的数据不能被处理器轻易地复制和计算。 但引用往往导致极其复杂的资源管理问题&#x…

C#探索之路基础夯实篇(4):UML类图中的六种关系详细说明

文章目录 UML类图中的关系前景1、关联关系&#xff08;Association&#xff09;&#xff1a;2、聚合关系&#xff08;Aggregation&#xff09;&#xff1a;3、组合关系&#xff08;Composition&#xff09;&#xff1a;4、泛化关系&#xff08;Generalization&#xff09;&…

PTA C 1050 螺旋矩阵(思路与优化)

本题要求将给定的 N 个正整数按非递增的顺序&#xff0c;填入“螺旋矩阵”。所谓“螺旋矩阵”&#xff0c;是指从左上角第 1 个格子开始&#xff0c;按顺时针螺旋方向填充。要求矩阵的规模为 m 行 n 列&#xff0c;满足条件&#xff1a;mn 等于 N&#xff1b;m≥n&#xff1b;且…

gitea简单介绍

Gitea是一个轻量级的开源自托管Git服务&#xff0c;提供了类似GitHub的功能和界面。它是一个简单、易于安装和使用的Git代码托管解决方案&#xff0c;适用于个人、小型团队和企业。 Gitea的主要特点如下&#xff1a; 自托管&#xff1a;Gitea允许在自己的服务器上搭建和管理…

zdpreact_antdesginpro 继续优化Ant Design开发的后台管理系统

登录后台管理系统 首先&#xff0c;将项目跑起来&#xff1a; 浏览器访问&#xff1a;http://localhost:8000/user/login 通过上次的优化&#xff0c;我们已经能够使用自己的账号密码进行登录了&#xff1a; 底部优化 登录后台以后&#xff0c;目前的底部是长这样的&…

《QT实用小工具·十六》IP地址输入框控件

1、概述 源码放在文章末尾 该项目为IP地址输入框控件&#xff0c;主要包含如下功能&#xff1a; 可设置IP地址&#xff0c;自动填入框。 可清空IP地址。 支持按下小圆点自动切换。 支持退格键自动切换。 支持IP地址过滤。 可设置背景色、边框颜色、边框圆角角度。 下面…

计算机网络 实验指导 实验12

路由信息协议&#xff08;RIP&#xff09;实验 1.实验拓扑图 名称接口IP地址网关Switch AF0/1192.168.1.1/24F0/2172.1.1.1/24Switch BF0/1192.168.1.2/24F0/2172.2.2.1/24PC1172.1.1.2/24172.1.1.1PC2172.1.1.3/24172.1.1.1PC3172.2.2.2/24172.2.2.1PC4172.2.2.3/24172.2.2.1…

纯小白蓝桥杯备赛笔记--DAY9(动态规划)

文章目录 一、动态规划基础&#xff08;1&#xff09;线性DP简介步骤例题数字三角形--1536破损的楼梯-3367安全序列-3423 &#xff08;2&#xff09;二维DP简介例题摆花--389选数异或--3711数字三角形--505 &#xff08;3&#xff09;最长公共子序列LCSLCS算法模型最长公共子序…

【Kafka】Kafka安装、配置、使用

【Kafka】安装Kafka 1. 安装Kafka2. Kafka使用2.0 集群分发脚本xsync(重要)2.0.1 scp命令2.0.2 rsync远程同步工具2.0.3 写一个集群分发脚本xsync (Shell 脚本) 2.1 Zookeeper安装2.2 对Kafka进行分发2.2.1 执行同步脚本2.2.2 三台云主机配置Kafka环境变量 1. 安装Kafka Kafka…

ARM架构麒麟操作系统安装配置Mariadb数据库

、安装配置JDK (1)检查机器是否已安装JDK 执行 java -version命令查看机器是否安装JDK,一般麒麟操作系统默认安装openjdk 1.8。 (2)安装指定版本JDK 如果麒麟操作系统默认安装的openjdk 1.8不符合需求的话,可以卸载机器安装的openjdk 1.8并按需安装所需的openjdk版本…

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…

租用阿里云4核16G服务器优惠价格多少钱?

阿里云4核16G服务器优惠价格26.52元1个月、79.56元3个月、149.00元半年&#xff0c;配置为阿里云服务器ECS经济型e实例ecs.e-c1m4.xlarge&#xff0c;4核16G、按固定带宽 10Mbs、100GB ESSD Entry系统盘&#xff0c;活动链接 aliyunfuwuqi.com/go/aliyun 活动链接打开如下图&a…