vue + gojs 实现拖拽 流程图

news2025/1/13 10:06:47

一、流程图效果

最近一段时间在研究go.js,它是一款前端开发画流程图的一个插件,也是一个难点,要说为什么是难点,首先,它是依赖画布canvas知识开发。其次,要依赖于内部API开发需求,开发项目需求的时候就要花费大量的时间去熟悉go.js的API,然后才能进行开发。话不多说,我就先把我最近做的项目案例效果图展示一下:

看到效果图大家可能会想这个挺简单的,会想没什么难点,其实真正开发的时候才会知道的、才会领悟到。

二、为什么选go.js流程图插件去开发项目?

在项目开发一期的时候我用的不是go.js,而用的是一款轻便的流程插件jsplumb.js,它也集成了各种功能性API,但是在开发二期的时候它的内部功能已经满足不了需求了,所以我就开始在网上查找流程插件,看了很多插件,比如:G6,D3 等这些可视化流程插件都是不能满足需求。要说为什么不能满足需求,原因如下:

一、首先,看到效果图里的内置多点和其他模块单点连线问题,其他插件是无法这个满足需求的,可能我没有深入去了解其他的流程插件吧,但是go.js里内置点连线可以让开发者很快的理解代码逻辑,不用耗费大量的时间去想点与点的连线。

二、代码上的数据结构问题,其他插件里的API数据字段繁琐量多,不够清晰明了,而go.js里的数据结构就两个重要字段,一是所有模块的字段集合二是连线字段集合,根据需求可以随意加字段。

三、项目开发

(一)、首先直接使用go.js,画布中是有水印的

其实这个问题不大,替换一行代码就可以去除水印

引入go.js后,直接在编辑器中全局搜索7eba17a4ca3b1a8346,找到类似这样结构的代码

a.ir=b.W[Ra("7eba17a4ca3b1a8346")][Ra("78a118b7")](b.W,ok,4,4);

注:不同的版本代码不是完全相同的,可能是a.jv(属性名是会变的) =‘xxxxx’,将这行代码替换成

a.ir=function(){return true;}; //a.属性名 要保持一致

去除水印的效果

(二)、HTML

<--第一种-->

<template>
    <div id="wrap">
        <div id="chart-wrap">
            <div id="chart-palette"></div><-- 画布一 -->
            <div id="chart-diagram"></div><-- 画布二 -->
        </div>
    </div>
</template>

如图:

第二种

结合vue的拖拽组件vuedraggable 实现业务需求。

<template>
     <div id="chart-wrap">
                <div v-for="tab in tabLIst" :key="tab.id" class="tab">
              //拖动
           <vuedraggable @end.stop="end" @start.stop="move">
          
                                 <i :class="tab.icon" /> {{ tab.text }}
                                 <el-tooltip effect="dark" :content="tab.tooltip" placement="top">
                                             <i class="el-icon-question" />
                                 </el-tooltip>
                    </vuedraggable>
                  </div>
         <div id="chart-diagram"/> <--画布-->
      </div>
</template>  
  

如图:

  

(三)、画布的基本设置 。

this.diagram = $(go.Diagram, "chart-diagram",
                {
                    // 画布初始位置
                    initialContentAlignment: go.Spot.LeftSide, // 居中显示
                    "undoManager.isEnabled": true, // 支持 Ctrl-Z 和 Ctrl-Y 操作
                    // 初始坐标
                    // initialPosition: new go.Point(0, 0),
                    //allowSelect:false,  ///禁止选中
                    // "toolManager.hoverDelay": 100, //tooltip提示显示延时
                    // "toolManager.toolTipDuration": 10000, //tooltip持续显示时间
                    //  isReadOnly:true,//只读

                    //禁止水平拖动画布
                    //禁止水平滚动条
                    allowHorizontalScroll: false,
                    // 禁止垂直拖动画布
                    //禁止垂直滚动条
                    allowVerticalScroll: false,
                    allowZoom: true,//画布是否可以缩放
                    "grid.visible": false, //显示网格
                    // allowMove: true, //允许拖动
                    // allowDragOut:true,
                    allowDelete: true,//禁止删除节点
                    allowCopy: true,//禁止复制
                    // 禁止撤销和重做
                    // "undoManager.isEnabled": false,
                    // 画布比例
                    // scale:1.5,
                    // minScale:1.2,//画布最小比例
                    // maxScale:2.0,//画布最大比例
                    // 画布初始化动画时间
                    // "animationManager.duration": 600,
                    // 禁止画布初始化动画
                    "animationManager.isEnabled": false,
                    // autoScale:go.Diagram.Uniform,//自适应

                    // autoScale:go.Diagram.UniformToFill,//自适应
                    //    "draggingTool.dragsLink": false,//拖动线
                    // autoScale:go.Diagram.None,//默认值不自适应
                    // 画布边距padding
                    // padding:80或者new go.Margin(2, 0)或new go.Margin(1, 0, 0, 1)
                    // validCycle: go.Diagram.CycleDestinationTree,//只允许有一个父节点
                    //节点模块动画  S
                    // "animationManager.initialAnimationStyle":go.Animation.EaseOutExpo,
                    // "animationManager.initialAnimationStyle": go.Animation.EaseInOutQuad,
                    "animationManager.initialAnimationStyle": go.AnimationManager.None,
                    // "animationManager.initialAnimationStyle":go.AnimationManager.AnimateLocations,
                    //节点模块动画   D
                    // validCycle: go.Diagram.CycleNotUndirected,

                    // validCycle: go.Diagram.CycleNotDirected,
                    // validCycle: go.Diagram.CycleSourceTree,
                    //ismodelfied:true //禁止拖拽
                    // 禁止鼠标拖动区域选中
                    // "dragSelectingTool.isEnabled" : false,
                    //允许使用delete键删除模块
                    "commandHandler.deletesTree": true,
                    // "hasHorizontalScrollbar":false,//去除水平滚动条
                    // "hasVerticalScrollbar":false,//去除竖直滚动条
                    // "canStart":false,
                    // allowClipboard: true,
                    // "toolManager.mouseWheelBehavior": go.ToolManager.WheelZoom, //有鼠标滚轮事件放大和缩小,而不是向上和向下滚动
                    // layout: $(go.TreeLayout,
                    //     { angle: 90, layerSpacing: 80 }),
                }
            );

(三)、整体画布事件及节点的监听

                // 监听连线
                this.diagram.addDiagramListener("LinkDrawn", (e) => { 
                    console.log(e.subject.part); 
                });
                // 监听删除
                 this.diagram.addDiagramListener("SelectionDeleted", (e) => {
                     e.subject.each(function (n) {
                         console.log(n.data.key);
                     });
                  })
                // 修改节点
                this.diagram.addDiagramListener("TextEdited", (evt) => {
                    console.log(e.subject.part);
                });
                // 监听点击
                this.diagram.addDiagramListener("ObjectSingleClicked", (e) => {
            
                   //这是清除高亮的
                   // e.diagram.commit((d) => {
                    //      d.clearHighlighteds();
                     // }, "no highlighteds"); 
                });

                // // 移动事件
                this.diagram.addDiagramListener("SelectionMoved", (e) => {
                    console.log(e.diagram.lastInput.documentPoint);
                });

(四)、连线点封装函数

// 节点连接线

            makePort (name, align, spot, output, input) {
                var horizontal = align.equals(go.Spot.Top) || align.equals(go.Spot.Bottom)
                return $(go.Shape, {
                    fill: 'transparent', // 默认透明不现实
                    strokeWidth: 0, // 无边框
                    fromMaxLinks: 1, //
                    width: horizontal ? NaN : 10, // 垂直"port"则8像素宽
                    height: !horizontal ? NaN : 5, // 水平"port"则8像素
                    alignment: align, // 同其模块对齐
                    stretch: horizontal
                        ? go.GraphObject.Horizontal
                        : go.GraphObject.Vertical, // 自动同其模块一同伸缩
                    portId: name, // 声明ID
                    fromSpot: spot, // 声明连线头连出此"port"的位置
                    fromLinkable: output, // 布尔型,是否允许连线从此"port"连出
                    toLinkable: input, // 布尔型,是否允许连线从此"port"连出
                    toSpot: spot, // 声明连线尾连入此"port"的位置
                    cursor: 'pointer', // 鼠标由指针改为手指,表示此处可点击生成连线
                    mouseEnter: function (e, port) {
                        // 鼠标移到"port"位置后,高亮
                        if (!e.diagram.isReadOnly) port.fill = 'rgba(255,0,255,0.3)'
                    },
                    mouseLeave: function (e, port) {
                        // 鼠标移出"port"位置后,透明
                        port.fill = 'transparent'
                    }
                })
            }

(五)、节点连线的高亮函数。

isHighlightedFun(link) {
                return $$(go.Shape,
                      "RoundedRectangle", {
                          fill: "rgba(217,236,255,.2)",
                          stroke: "rgba(39,154,242,.1)",
                          strokeWidth: 1,
                      },
                      new go.Binding("stroke", "isHighlighted", (h) => {
                            return h ? "rgba(39,154,242,1)" : "rgba(39,154,242,.8)"; 
                      }).ofObject(),
                      new go.Binding("strokeWidth", "isHighlighted", (h) => {
                          return h ? 2.5 : 1.3;
                      }).ofObject(),
                )
    }

如图:

(六)、 节点连接线。(方法封装)

makePort(name, align, spot, output, input) {
                var horizontal = align.equals(go.Spot.Top) || align.equals(go.Spot.Bottom);
                return $$(go.Shape,
                      {
                          fill: "transparent",  // 默认透明
                          strokeWidth: 0,  // 无边框
                          fromMaxLinks: 1,  //最大连接数
                          width: horizontal ? NaN : 10,  // 垂直"port"则8像素宽
                          height: !horizontal ? NaN : 2,  // 水平"port"则8像素
                          alignment: align,  // 同其模块对齐
                          stretch: (horizontal ? go.GraphObject.Horizontal : go.GraphObject.Vertical),//自动同其模块一同伸缩
                          portId: name,  // 声明ID
                          fromSpot: spot,  // 声明连线头连出此"port"的位置
                          fromLinkable: output,  // 布尔型,是否允许连线从此"port"连出
                          toLinkable: input,  // 布尔型,是否允许连线从此"port"连出
                          toSpot: spot,  // 声明连线尾连入此"port"的位置
                          cursor: "pointer",  // 鼠标由指针改为手指,表示此处可点击生成连线
                          mouseEnter: (e, port) => {  //鼠标移到"port"位置后,高亮
                              if (!e.diagram.isReadOnly) port.fill = "rgba(255,0,255,0.3)";
                          },
                          mouseLeave: (e, port) => {// 鼠标移出"port"位置后,透明
                              port.fill = "transparent";
                          }
                      });
            }

注意:要在this.diagram.nodeTemplateMap.add 增加模块里调用。

示例:

this.diagram.nodeTemplateMap.add(type,
                      $$(go.Node, "Auto",
               this.makePort("T", go.Spot.Top, go.Spot.TopCenter, false, true)
))

四、当前画布的监听事件名称包括

  • "AnimationStarting" :动画渲染前事件,加载图表的动画即将开始;

  • "AnimationFinished" :动画渲染完事件,加载图表刚刚完成的动画;

  • "BackgroundSingleClicked": 背景单击事件, 单击图表背景;

  • "BackgroundDoubleClicked" :背景双击事件, 双击图表背景;

  • "BackgroundContextClicked" :背景右键事件, 右键单击图表背景;

  • "ChangingSelection": 改变选择前事件, 一个操作即将改变Diagram.selection图表选择集合,

  • "ChangedSelection" :改变选择后事件, 一个操作已经改变?Diagram.selection图表选择集合,

  • "ClipboardChanged" :剪切板改变事件, 零部件已被CommandHandler.copySelection复制到剪贴板上;

  • "ClipboardPasted" :剪切板粘贴事件,零部件已由CommandHandler.pasteSelection从剪贴板复制到图表中;

  • "DocumentBoundsChanged": 文档范围改变事件, 图表中各零部件的面积,?Diagram.documentBounds,已经改变了;

  • "ExternalObjectsDropped" :(节点或线等)零部件拖放生成事件, 零部件已经通过拖拽从图的外部复制到图中;

  • "GainedFocus" :获得键盘焦点事件, 该图获得了键盘焦点,例如在调用Diagram.focus之后.

  • "LayoutCompleted" :布局完成事件, 整个图表布局刚刚更新;

  • "LinkDrawn": 线创建事件,用户刚刚使用LinkingTool创建了一个新链接;?

  • "LinkRelinked": 线重新连接事件, 用户刚刚通过RelinkingTool?或?DraggingTool重新连接了现有线;

  • "LinkReshaped": 线路径改变事件, 用户刚刚通过LinkReshapingTool调整了线的路径;

  • "LostFocus" :图表失去焦点事件,这个图表失去了键盘焦点,

  • "Modified" :图表改变事件,?Diagram.isModified?属性已被设置为一个新值——用于将窗口标记为自上次保存以来已被修改;

  • "ObjectSingleClicked": 对象单击事件, 单击了图形对象(节点和线等);

  • "ObjectDoubleClicked":双击了图形对象(节点和线等);

  • "ObjectContextClicked":右键单击了图形对象(节点和线等);

  • "PartCreated" :Part创建事件,用户通过?ClickCreatingTool插入新的零部件;

  • "PartResized" :Part大小改变事件, 用户通过ResizingTool 调整工具改变了一个图形对象的大小;

  • "PartRotated" :Part旋转事件, 用户通过RotatingTool 旋转工具改变了一个图形对象的角度 ;

  • "SelectionMoved" :拖动事件, 用户通过DraggingTool 拖动工具移动了选定的部分;

  • "SelectionCopied" :复制事件,户通过DraggingTool 拖动工具复制了选定的部分;

  • "SelectionDeleted" :删除后事件,用户通过?CommandHandler.deleteSelection?已经删除了选定的部分;

  • "SelectionDeleting" :删除前事件,用户通过?CommandHandler.deleteSelection即将删除选定的部分;

  • "SelectionGrouped" :选择创建分组事件, 通过CommandHandler.groupSelection已经从选择的零部件中创建了一个新的组 ;

  • "SelectionUngrouped":用户已删除选定的组,但通过CommandHandler.ungroupSelection保留其成员;

  • "SubGraphCollapsed" :子图折叠事件, 用户通过CommandHandler.collapseSubGraph将选定的组折叠;

  • "SubGraphExpanded" :子图展开事件, 用户通过CommandHandler.expandSubGraph将选定的组展开;

  • "TextEdited" :文本块修改事件,用户通过文本编辑工具改变了文本块的字符串值;

  • "TreeCollapsed": 树折叠事件, 用户通过CommandHandler.collapseTree折叠所选节点的子树;

  • "TreeExpanded" :树展开事件, 用户通过CommandHandler.expandTree展开了所选节点的子树;

  • "ViewportBoundsChanged":视窗范围改变事件,图表中可见的区域,?Diagram.viewportBounds,发生了改变;

五、迭代篇

(一)、项目的迭代。

从数据结构到ui风格的整体重构,有了新的变化。

整体流程 如图一:

模块流程侧边设置图二:

伸缩当前灰色意图如图三:

为什么收起意图会有多条连线,是因为当前模块下的意图和部分模块有关系线,所以在收起时要知道当前模块下有几个意图和其他模块有关系。

(二)、整体用的js方法

1、画布里的功能js:

2、工具条js 方法:

(三)、功能分析

1、工具条中的功能方法

(1)、查看

注: 是通过gojs 实例中的makeSVG方法打开一个新窗口生成的一个svg

printDiagram() {
    var svgWindow = window.open();
if (!svgWindow) return;
var printSize = new go.Size(3000, 2000);
var bnds = this.diagram.documentBounds;
var x = bnds.x;
var y = bnds.y;
while (y < bnds.bottom) {
        while (x < bnds.right) {
            var svg = this.diagram.makeSVG({scale: 1.0, position: new go.Point(x, y), size: printSize});
svgWindow.document.body.appendChild(svg);
x += printSize.width;
}
        x = bnds.x;
y += printSize.height;
}
}

(2)、居中

这里将调用CommandHandler.scrollToPart,先还原画布初始状态后执行动画以将视口滚动到节点所在的位置。

注:如果节点靠近边缘,除非您增加Diagram.scrollMargin,否则无法滚动视口,以使该节点更靠近视口的中心。

centerDiagram() {
            this.resetCanvas();
            this.diagram.commandHandler.scrollToPart(this.diagram.findNodeForKey(this.initModuleId));
},
 // 还原
        resetCanvas() {
            this.diagram.scale = 1;
            this.$refs.set.zoom = 100;
        },

尽请等待周六周日继续完善。

如有需求或者疑问加QQ:二79三64七52 讨论。 先写到这,下周写整体迭代更新(二)看下一篇, 如有不足 可以私信单聊共享技术经验。。。。。。。。。。。。。

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

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

相关文章

js逆向爬取某音乐网站某歌手的歌曲

js逆向爬取某音乐网站某歌手的歌曲一、分析网站1、案例介绍2、寻找列表页Ajax入口&#xff08;1&#xff09;页面展示图。&#xff08;2&#xff09;寻找部分歌曲信息Ajax的token。&#xff08;3&#xff09;寻找歌曲链接&#xff08;4&#xff09;获取歌曲名称和id信息3、寻找…

vue-plugin-hiprint vue hiprint vue使用hiprint打印控件VUE HiPrint HiPrint简单使用

vue-plugin-hiprint vue hiprint vue使用hiprint打印控件VUE HiPrint HiPrint简单使用安装相关依赖安装 vue-plugin-hiprintJQuery安装 打印客户端引入依赖打印 html 内容 - 简单使用根据模版打印 - 简单使用以下内容 和上面demo 没关系 &#xff01;&#xff01;&#xff01;&…

使用videjs+vue2+elementui自定义播放器控件

一、安装项目所需依赖 videojs依赖&#xff1a; npm install --save-dev video.js elementui依赖&#xff08;这个图方便就不按需引入了&#xff09;&#xff1a; npm i element-ui -S 二、main.js修改 增加以下几行&#xff1a; import videojs from video.js import e…

成功解决:下载的谷歌浏览器,打开却是“2345浏览器”,方法亲测有效

今天打开谷歌浏览器使用&#xff0c;浏览器界面显示的2345浏览器&#xff0c;难道谷歌把2345收购了&#xff1f;应该不能&#xff0c;上网查找问题原因才发现&#xff0c;原来的谷歌首页是被劫持了。&#xff08;如果迫切解决问题&#xff0c;直接拉到底找方法&#xff09; 试了…

前端必备的谷歌浏览器JSON可视化插件:JSON-Handle

功能简介: 日常开发过程中,对接后台返回的数据接口时,数据格式常常是各种json格式字符串,在netWork里面查看十分不便,需要在网上找一个json格式化的工具再查看,然后再取数据字段,然后绑定到页面上,十分不便,推荐这么一款前端开发的浏览器插件工具给大家使用。 返回数…

【React-Hooks进阶】useState回调函数的参数 / useEffect发送网络请求/ useRef / useContext

前言 博主主页&#x1f449;&#x1f3fb;蜡笔雏田学代码 专栏链接&#x1f449;&#x1f3fb;React专栏 上篇文章初步学习了Hooks的基础知识 今天来深入学习Hooks的一些扩展知识 感兴趣的小伙伴一起来看看吧~&#x1f91e; 文章目录useState -回调函数的参数使用场景语法语法规…

el-switch

目录 在element ui中el-switch开关组件具有先改变开关值再传值的特点。&#xff08;先改后传&#xff09; 1、触发change事件时 2、绑定disabled属性写三元表达式时 3、解决办法 在element ui中el-switch开关组件具有先改变开关值再传值的特点。&#xff08;先改后传&#xf…

cron表达式 详解

corn表达式是&#xff1a;由若干数字、空格、符号按一定的规则&#xff0c;组成的一组字符串&#xff0c;从而表达时间的信息。 好像和正则表达式有点类似哈&#xff0c;都是一个字符串表示一些信息。Cron 表达式生成器&#xff1a; https://www.smart-tools.cn/cron简介Cron 表…

基于Java+MySQL 实现(Web)日程管理系统【100010222】

基于Java的日程管理系统开发 摘要 日程管理在日常生活中是十分普通的一件事情&#xff0c;人们无论在生活中还是工作中都会有大大小小、各种各样的事情安排&#xff0c;如果仅仅靠纸张或者自己记录这些事情&#xff0c;往往会遗忘。针对这样的痛点&#xff0c;本文提供了日程…

HTML小图标的使用(无需下载图标源码)

我们在浏览一个网页中&#xff0c;会遇到很多有趣的小图标&#xff0c;这些小图标与访问者形成了友好的互动&#xff0c;所以我们在开发中都会适当插入一些生动有趣的图标来吸引访问者。插入图标的网站有很多&#xff0c;我这里以阿里巴巴图标库&#xff08;iconfont-阿里巴巴矢…

Div标签里放img和span标签实现垂直水平居中

正常默认布局 代码实现&#xff1a; <div style"width: 400px; height:400px; background-color:blueviolet"><img style"width: 80px; height: 80px;" src"./picture.png"><span style"color:white;">我是span标…

Vue3 使用MD5加密(清晰明了)

概述 最近在想做个cloud项目,gitee上找了个模板项目&#xff0c;前端使用到vue3 typeScript、Element Plus、Vue Router、Pinia、Axios、i18n、Vite等技术&#xff0c;最近使用到vue3 MD5加密&#xff0c;顺便学习一下&#xff0c;在此总结一下&#xff0c;若有不足之处&…

《EDA前端软件开发工程师面试指南》

2023届EDA领域校招总结&#xff0c;完结撒花&#xff01;&#xff01;&#xff01; 目录 前言 一、EDA公司介绍 二、项目面试 1.自我介绍 2.项目深入 3.专业经验 4.成果和技能 5.对面试官有什么问题 三、C面试 1、高频考点 2、其他知识点 3、算法题 四、逻辑综合面…

【前端代码实例】使用HTML5+CSS3+JavaScript制作一个响应式的后台管理系统~带侧边导航栏仪表盘功能

bilibili在线视频演示地址: 【前端代码实例】使用HTML5+CSS3+JavaScript制作一个响应式的后台管理系统~带侧边导航栏仪表盘功能 效果图: 完整代码: <!DOCTYPE html> <html lang="en"> <head><meta charset="UTF-8"><meta…

前端已死?后端已亡?弯弯绕绕,几分真几分假

前段时间&#xff0c;我在掘金分享了一篇GPT-4 性能文章&#xff0c;也许是过于强大带来的威胁性&#xff0c;引来评论区的排队哀嚎&#xff08;如下图&#xff09;&#xff0c;所以“前端已死&#xff0c;后端已亡”这个概念真的成立吗&#xff1f;本文着重探讨前端。 前端和后…

Seurat | 强烈建议收藏的单细胞分析标准流程(基础质控与过滤)(一)

1. 写在前面 作为现在最火的scRNAseq分析包&#xff0c;Seurat当之无愧。&#x1f618; 本期开始我们介绍一下Seurat包的用法&#xff0c;先从基础质控和过滤开始吧。&#x1f973; 2.用到的包 rm(list ls()) library(Seurat) library(tidyverse) library(SingleR) library…

钉钉消息防撤回功能研究与实现-可查看历史消息[文件/图文/管理员/链接 撤回拦截]

研究背景 由于在某个大学进行上课的时候,遇到的某个老师,总是习惯发过的消息,到第二天的时候撤回,我们用聊天工具的其中一个原因,不就是因为可以随时去查看发过的消息吗&#xff0c;&#xff0c;而这位老师的操作,也让包括我在内的很多人感到痛不欲生。 想一想,当自己想要去看下…

Error in mounted hook: TypeError: Cannot read properties of undefined (reading isHiddenDay ) found

Error in mounted hook: TypeError: Cannot read properties of undefined (reading isHiddenDay ) found 无法读取未定义的属性‘isHiddenDay’. 在vue中使用fullcalendar在mounted钩子中渲染报错 背景 我在一个小demo中实现还是好好的&#xff0c;并且用的依赖都是6.x版本的…

初入了解——什么是VUE

个人简介&#xff1a;云计算网络运维专业人员&#xff0c;了解运维知识&#xff0c;掌握TCP/IP协议&#xff0c;每天分享网络运维知识与技能。座右铭&#xff1a;海不辞水&#xff0c;故能成其大&#xff1b;山不辞石&#xff0c;故能成其高。个人主页&#xff1a;小李会科技的…

vue3+ts项目里如何使用状态管理pinia以及数据持久化

我们都知道在vue2项目里搭配状态管理vuex3XX使用&#xff0c;效果极好的。 虽然在vue3项目里&#xff0c;vuex4XX仍能发挥余热&#xff0c;但由于缺乏对于ts的支持&#xff0c;使得类型推断陷入僵局。 所以在vue3ts的项目里&#xff0c;vuex渐被舍弃&#xff0c;pinia取而代之…