Canvas简历编辑器-Monorepo+Rspack工程实践

Canvas简历编辑器-Monorepo+Rspack工程实践

在之前我们围绕Canvas聊了很多代码设计层面的东西,在这里我们聊一下工程实践。在之前的文中我也提到过,因为是本着学习的态度以及对技术的好奇心来做的,所以除了一些工具类的库例如 ArcoDesignResizeObserveJest 等包之外,关于 数据结构packages/delta、插件化packages/plugin、核心引擎packages/core 等都是手动实现的,所以在这里除了学习了Canvas之外,实际上还做了一些项目工程化的实践。

  • 在线编辑: https://windrunnermax.github.io/CanvasEditor
  • 开源地址: https://github.com/WindrunnerMax/CanvasEditor

关于Canvas简历编辑器项目的相关文章:

  • 掘金老给我推Canvas,我也学习Canvas做了个简历编辑器
  • Canvas图形编辑器-数据结构与History(undo/redo)
  • Canvas图形编辑器-我的剪贴板里究竟有什么数据
  • Canvas简历编辑器-图形绘制与状态管理(轻量级DOM)
  • Canvas简历编辑器-Monorepo+Rspack工程实践
  • Canvas简历编辑器-层级渲染与事件管理能力设计
  • Canvas简历编辑器-选中绘制与拖拽多选交互方案

Pnpm+Monorepo

我们先来聊聊为什么要用monorepo,先举一个我之前踩过的坑作为例子,在之前我的富文本编辑器项目 DocEditor 就是完全写在了独立的单个src目录中,在项目本身的运行过程中是没什么问题的,但是当时我想将编辑器独立出来作为NPM包用,打包的过程是借助了Rollup也没什么问题,问题就出在了引用方上。当时我在简历编辑器中引入文档编辑器的NPM包时,发现有一个模块被错误的TreeShaking了,现在都还能在编辑器中看到这部分兼容。

module: {rules: [{// 对`doc-editor-light`的`TreeShaking`有点问题test: /doc-editor-light\/dist\/tslib.*\.js/,sideEffects: true,},]
}

这个问题导致了我在dev模式下没有什么问题,但是在build之后这部分代码被错误地移除掉了,导致编辑器的wrapper节点出现了问题,列表等元素不能正确添加。当然实际上这不能说明独立包项目不好,只能说整个管理的时候可能并不是那么简单,尤其是打包为NPM包的时候需要注意各个入口问题。那么现在引用我的富文本编辑器包已经变成了4个独立的包分别引用,各司其职,就没再出现过这个问题。

说起来打包的问题,我还踩过一个坑,不知道大家是不是见到过ReactInvalid hook call这个经典报错。之前我将其独立拆包的时候之后,发现会报这个错,但是我在package.json中是标注的peerDependencies "react": ">=16",按理说这里会直接应用安装该包的React,不可能出现版本不一致的问题,至于Rules of Hooks肯定也不可能,因为我之前是好好的,拆完包才出的问题。最后发现是我在rollup中没把peerDependencies这部分解析,导致jsx-runtime被打进了包里,虽然React的版本都是17.0.2但是实际上是运行了两个独立词法作用域的React Hooks,这才导致了这个问题。

Invalid hook call. Hooks can only be called inside of the body of a function component. This could happen for one of the following reasons:1.  You might have mismatching versions of React and the renderer (such as React DOM)
2.  You might be breaking the Rules of Hooks
3.  You might have more than one copy of React in the same app See for tips about how to debug and fix this problem.

接着回到项目本身,当前项目已经抽离出来独立的RspackMonoTemplate,平时开发也会基于这个模版创建仓库。当前简历编辑器项目的结构tree -L 2 -I node_modules --dirsfirst如下:

CanvasEditor
│── packages
│   ├── core
│   ├── delta
│   ├── plugin
│   ├── react
│   └── utils
├── package.json
├── pnpm-lock.yaml
├── pnpm-workspace.yaml
└── tsconfig.json
  • packages/core: 编辑器核心引擎模块,对于 剪贴板操作、事件管理、状态管理、History模块、Canvas操作、选区操作 等等都封装在这里,相当于实现了基本的Canvas引擎能力。
  • packages/delta: 数据结构模块,设计了基准数据结构,实现了DeltaSet数据结构以及原子化的Op操作,主要用于描述整个编辑器的数据结构以及操作,实现了invert等能力,对于实现History模块有很大的意义。
  • packages/plugin: 插件模块,在packages/delta的基础上设计了插件化的能力,主要为了实现编辑器的功能模块化,例如TextImageRect等插件都是在这里实现的。
  • packages/react: React模块,主要是为了通过实现编辑器的视图层,在这里有比较重要的一点,我们的核心模块是视图框架无关的,如果有必要的话同样可以使用VueAngular等框架来实现视图层。
  • packages/utils: 工具模块,主要是一些工具函数的封装,例如FixedNumberPalette等等,这些工具函数在整个编辑器中都有使用,是作为基础包在整个workspace中引用的。
  • package.json: 整个workspacepackage.json,在这里配置了一些项目的信息,EsLintStyleLint相关的配置也都在这里实现。
  • pnpm-lock.yaml: pnpm的锁文件,用于锁定整个workspace的依赖版本。
  • pnpm-workspace.yaml: pnpmworkspace配置文件,用于配置monorepo的能力。
  • tsconfig.json: 整个workspacetsconfig配置文件,用于配置整个workspaceTypeScript编译配置,在这里是作为基准配置以提供给项目中的模块引用。

pnpm自身是非常优秀的包管理器,通过硬链接和符号链接来节省磁盘空间,每个版本的包只需要存储一次,最重要的是pnpm创建了一个非扁平化的node_modules结构,从而确保依赖与声明严格匹配,严格控制了依赖提升,能够避免依赖升级的意外问题,这提高了项目的一致性和可预测性。

而说回到monorepopnpm不光是非常优秀的包管理器,其还提供了一个开箱即用的monorepo能力。在pnpm中存在一个pnpm-workspace.yaml文件,这个文件是用来配置workspack的,而pnpmworkspace就可以作为monorepo的能力,而我们的配置也非常简单,我们认为在packages目录下的所有目录都作为子项目。

packages:- 'packages/*'

通过monorepo我们可以很方便的管理所有子项目,特别是对于需要发Npm包的项目,将子模块拆分是个不错的选择,特别如果能够做到视图层框架无关的话就显得更加有意义。此外,monorepo对于整个项目的管理也有很多益处,例如在打包整个应用的时候,我们不需要对每个子项目发新的包之后才能打包,而是可以直接将编译过程放在workspace层面,这样就可以保证整个项目的一致性,简化了构建过程和持续集成流程,让所有项目可以共享构建脚本和工具配置。此外所有项目和模块共用同一个版本控制系统,便于进行统一的版本管理和变更跟踪,而且还有助于同步更新这些项目间的依赖关系。

TS+Rspack最佳实践

说了这么多使用pnpm + monorepo管理项目带来的好处,我们再来聊聊我对TSRspack应用于Monorepo的最佳实践,不知道大家是不是遇到过这样的两个问题:

  • 子项目的TS声明更改后不能实时生效,必须要编译一次子项目才可以,而子项目编译的过程中如果将dist等产物包删除,那么在vsc或者其他编辑器中就会报TS找不到引用声明的错误,这个时候就必须要用命令重新Reload TypeScript Project来去掉报错。而如果不将产物包删除的话,就会出现一些隐性的问题,例如原来某个文件命名为a.tsx,此时因为一些原因需要将其移动到同名的a目录并且重新命名为index.tsx,那么执行了这一顿操作之后,发现如果更改此时的index.tsx代码不会更新,必须要重启应用的webpack等编译器才行,因为其还是引用了原来的文件,产生类似的问题虽然不复杂但是排查起来还是需要时间的。
  • 更改子项目的TS代码必须要重新编译子项目,因为项目是monorepo管理的,在package.json中会有workspace引用,而workspace实际上是在node_modules被引用的,所以虽然是子项目但是仍然需要遵循node_modules的规则才可以,那么其通常需要被编译为js才可以被执行,所以每次修改代码都必须要全量执行一遍很是麻烦,当然通常我们可以通过-w命令来观察变动,但是毕竟多了一道步骤,且如果是存在alias的项目可能仅仅使用tsc来编译还不够。此外在monorepo中我们通常会有很多子项目,如果每个子项目都需要这样的话,特别在这种编译时全量编译而不是增量编译的情况下,那么整个项目的编译时间就会变得非常长。

那么在这里我们先来看第一个问题,子项目的TS声明更改后不能实时生效,因为我们也提到了monorepo子项目实际上是通过node_modules来管理和引用的,所以其在默认情况下依然需要遵循node_modules的规则,即packages.jsontypes字段指向的TS声明文件,那么我们有没有什么办法可以修改这个行为呢,当然是有的,我们在整个项目的根tsconfig.json配置path就可以完美解决这个问题。当我们配置好如下的内容之后,通过按住Ctrl加鼠标左键点击的时候,就可以跳转到子项目的根目录声明了。此外这里有个要关注的点是,在项目中不建议配置"baseUrl": ".",在这里会有一些奇奇怪怪的路径引用问题,所以在简历编辑器项目中除了要打包Npmtsconfig.build.json之外,都是直接使用相对路径配置的。

{"compilerOptions": {"...": "...","paths": {"sketching-core": ["./packages/core/src"],"sketching-delta": ["./packages/delta/src"],"sketching-plugin": ["./packages/plugin/src"],"sketching-utils": ["./packages/utils/src"],},},"include": ["packages/*/src"]
}

那么解决了项目的TS声明问题之后,我们再来看编译的问题,这里的问题看起来会复杂一些,因为TS声明就单纯只是类型声明而已,不会影响到项目本身代码的编译,编译类型检查除外。那么在Rspack中应该配置才能让我们的代码直接指向子项目,而不是必须要走node_modules这套规则,实际上这里也很简单,只需要配置resolve.alias就可以了,这样当我们直接修改TS代码时,也能让编辑器立即响应增量编译。

{
// ....resolve: {alias: {"@": path.resolve(__dirname, "./src"),"sketching-core": path.resolve(__dirname, "../core/src"),"sketching-delta": path.resolve(__dirname, "../delta/src"),"sketching-plugin": path.resolve(__dirname, "../plugin/src"),"sketching-utils": path.resolve(__dirname, "../utils/src"),},},
// ....
}

实际上对于Rspack而言其帮我们做了很多事,比如即使是node_modulesTS文件也会编译,而对于一些通过CRA创建的webpack项目来说,这个配置就麻烦一些,当然我们同样也可以借助customize-cra来完成这件事,此外我们还要关闭一些类似于ModuleScopePlugin的插件才可以,下面是富文本编辑器项目 DocEditor 的配置。

const src = path.resolve(__dirname, "src");
const index = path.resolve(__dirname, "src/index.tsx");
const core = path.resolve(__dirname, "../core/src");
const delta = path.resolve(__dirname, "../delta/src");
const plugin = path.resolve(__dirname, "../plugin/src");
const utils = path.resolve(__dirname, "../utils/src");module.exports = {paths: function (paths) {paths.appSrc = src;paths.appIndexJs = index;return paths;},webpack: override(...[// ...addWebpackResolve({alias: {"doc-editor-core": core,"doc-editor-delta": delta,"doc-editor-plugin": plugin,"doc-editor-utils": utils,},}),babelInclude([src, core, delta, plugin, utils]),// ...configWebpackPlugins(),].filter(Boolean)),
};

此外,简历编辑器是纯前端的项目,这样的项目有个很大的优势是可以直接使用静态资源就可以运行,而如果我们借助GitHub Action就可以通过Git Pages在仓库中直接部署,并且可以直接通过GitHub Pages访问,这样在仓库中就能呈现一个完整的DEMO

// .github/workflows/deploy.yml
name: deploy gh-pageson:push:branches:- masterjobs:build-and-deploy:runs-on: ubuntu-lateststeps:- name: checkoutuses: actions/checkout@v2with:fetch-depth: 0persist-credentials: false- name: install node-v16uses: actions/setup-node@v3with:node-version: '16.16.0'- name: install dependenciesrun: |node -vnpm install -g pnpmpnpm config set registry https://registry.npmjs.org/pnpm install --registry=https://registry.npmjs.org/- name: build projectrun: |npm run build:react- name: deploy projectuses: JamesIves/github-pages-deploy-action@releases/v3with:GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}BRANCH: gh-pagesFOLDER: packages/react/build

最后

在这里我们聊了为什么要用Monorepo以及简单聊了一下pnpm workspace的优势,然后解决了在子项目开发中会遇到的TS编译、项目编译的两个实际问题,分别在MonorepoRspackWebpack项目中相关的部分实践了一下,最后还简单聊了一下利用GitHub Action直接在Git Pages部署在线DEMO。那么再往后边的文章中,我们就需要聊一聊如何实现 层级渲染与事件管理 的能力设计。

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

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

相关文章

【路径规划】绘制算术和几何布朗运动- 绘制布朗桥、2D 和 3D 布朗运动- 绘制一些随机路径

摘要 本文演示了如何生成和绘制布朗运动、几何布朗运动和布朗桥的随机路径。这些随机路径广泛应用于金融、物理和工程领域,用于模拟随机过程。实验结果包括了多条随机路径的示例,展示了不同类型的布朗运动的特征。 理论 1. 布朗运动 (Brownian Motion…

构建高效房屋租赁系统:Spring Boot应用

1 绪论 1.1 研究背景 中国的科技的不断进步,计算机发展也慢慢的越来越成熟,人们对计算机也是越来越更加的依赖,科研、教育慢慢用于计算机进行管理。从第一台计算机的产生,到现在计算机已经发展到我们无法想象。给我们的生活改变很…

如何在NXP源码基础上适配ELF 1开发板的UART功能

UART即通用异步收发器,是一种支持全双工串行通信协议的接口。在i.MX6ULL处理器平台上,该处理器原生支持多达8路的UART接口,提供了丰富的串行通信能力。 针对ELF 1开发板,实际引出了4路UART接口供开发者使用,具体包括U…

Node-RED-L2-Node-RED在Linux系统启动时自动运行

Node-RED在Linux系统启动时自动运行 目的步骤1创建服务文件:2重新加载服务:3启用服务:4启动Node-RED服务:5检查服务状态:6其他说明7如果没启动正确的Node-RED执行路径:确保使用绝对路径: 检查用…

Flutter 约束布局

配置插件依赖 设置组件大小 通过属性 childConstraints 实现 分别设置 约束布局一 和 约束布局二 大大小为:160 和 200 点击查看代码文件 class SummaryPageState extends State<SummaryPage1> {ConstraintId constraintId_1 = ConstraintId(ConstraintId_1);Constrain…

易航网址导航系统V2.45完美去授权版

简介 易航网址导航系统V2.45完美去授权版 界面

Spring(看这一篇就够了)

Spring 概述 Spring 是最受欢迎的企业级 Java 应用程序开发框架&#xff0c;数以百万的来自世界各地的开发人员使用 Spring 框架来创建性能好、易于测试、可重用的代码。 Spring 框架是一个开源的 Java 平台&#xff0c;它最初是由 Rod Johnson 编写的&#xff0c;并且…

SFUD库移植

1.源码 GitHub - armink/SFUD: An using JEDECs SFDP standard serial (SPI) flash universal driver library | 一款使用 JEDEC SFDP 标准的串行 (SPI) Flash 通用驱动库 2.介绍 这个通用驱动库,实际就是帮你封装好了读写spiflash的函数, 我们只需要对接以下底层,就可以轻松…

【个人笔记】线程和线程池的状态以及转换方式

线程和线程池的状态是不一样的&#xff01;&#xff01; 线程有 6 种状态&#xff0c;查看Thread的State枚举类&#xff1a; NEW&#xff1a;创建后没启动的线程就处于这种状态RUNNABLE&#xff1a;正在java虚拟机中执行的线程就处于这种状态BLOCKED&#xff1a;受阻塞并等待…

Observability:构建下一代托管接入服务

作者&#xff1a;来自 Elastic Vishal Raj, Marc Lopez Rubio 随着无服务器&#xff08;serverless&#xff09;的引入&#xff0c;向 Elastic Cloud 发送可观察性数据变得越来越容易。你可以在 Elastic Cloud Serverless 中创建一个可观察性无服务器项目&#xff0c;并将可观察…

【Java】虚拟机(JVM)内存模型全解析

目录 一、运行时数据区域划分 版本的差异&#xff1a; 二、程序计数器 程序计数器主要作用 三、Java虚拟机 1. 虚拟机运行原理 2. 活动栈被弹出的方式 3. 虚拟机栈可能产生的错误 4. 虚拟机栈的大小 四、本地方法栈 五、堆 1. 堆区的组成&#xff1a;新生代老生代 …

Ubuntu磁盘不足扩容

1.问题 Ubuntu磁盘不足扩容 2.解决方法 安装一下 sudo apt-get install gpartedsudo gparted

Mysql梳理6——order by排序

目录 6 order by排序 6.1 排序数据 6.2 单列排序 6.3 多行排列 6 order by排序 6.1 排序数据 使用ORDER BY字句排序 ASC&#xff08;ascend&#xff09;:升序DESC(descend):降序 ORDER BY子句在SELECT语句的结尾 6.2 单列排序 如果没有使用排序操作&#xff0c;默认…

C语言课程设计题目一:职工信息管理系统设计

文章目录 题目一&#xff1a;职工信息管理系统设计代码块employeeManagement.hemployeeManage.ctest.c 调试验证录入信息&#xff0c;并浏览验证职工号唯一保存职工信息&#xff0c;加载职工信息按职工号进行查询根据id删除职工修改职工信息 题目一&#xff1a;职工信息管理系统…

下水道内缺陷识别检测数据集 yolo数据集 共2300张

下水道内缺陷识别检测数据集 yolo数据集 共2300张 下水道内部缺陷识别数据集&#xff08;Sewer Interior Defect Recognition Dataset, SIDRD&#xff09; 摘要 SIDRD 是一个专门针对下水道内部缺陷识别的数据集&#xff0c;旨在为城市基础设施维护和管理提供一个标准化的训练…

VmWare安装虚拟机保姆级教程(centos7,虚拟机网络设置,虚拟机桌面显示)

VMWare下载&#xff1a; 下载 VMware Workstation Pro - VMware Customer Connect 安装包&#xff1a;&#xff08;16的版本&#xff09;免费&#xff01;&#xff08;一个赞就行&#xff09; 一直点下一步即可&#xff0c;注意修改一下安装位置就好 二、安装虚拟机 安装虚…

论文复现:考虑电网交互的风电、光伏与电池互补调度运行(MATLAB-Yalmip-Cplex全代码)

论文复现:考虑电网交互的风电、光伏与电池储能互补调度运行(MATLAB-Yalmip-Cplex全代码) 针对风电、光伏与电化学储能电站互补运行的问题,已有大量通过启发式算法寻优的案例,但工程上更注重实用性和普适性。Yalmip工具箱则是一种基于MATLAB平台的优化软件工具箱,被广泛应用…

[uni-app]小兔鲜-02项目首页

轮播图 轮播图组件需要在首页和分类页使用, 封装成通用组件 准备轮播图组件 <script setup lang"ts"> import type { BannerItem } from /types/home import { ref } from vue // 父组件的数据 defineProps<{list: BannerItem[] }>()// 高亮下标 const…

【React】Ant Design 5.x版本drawer抽屉黑边问题

环境 antd: ^5.14.1react: ^18 问题情况 <Drawer open{open} closable{false} mask{false} width{680}getContainer{props.getContainer || undefined}><p>Some contents...</p><p>Some contents...</p><p>Some contents...</p> …

时序数据库 TDengine 的入门体验和操作记录

时序数据库 TDengine 的学习和使用经验 什么是 TDengine &#xff1f;什么是时序数据 &#xff1f;使用RPM安装包部署默认的网络端口 TDengine 使用TDengine 命令行&#xff08;CLI&#xff09;taosBenchmark服务器内存需求删库跑路测试 使用体验文档纠错 什么是 TDengine &…