npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

svg-flow-editor-mvp

v1.1.2

Published

svg flow editor 是一款自研流程图编辑器,提供了一系列流程图交互、编辑所必需的功能,计划支持前端研发自定义开发各种逻辑编排场景,如流程图、ER 图、BPMN 流程等。项目使用typescript与svg、canvas等技术进行搭建,脱离vue、react等框架的限制,使得用户更快、更轻松融合到自己的项目中,意在工作审批配置、机器人逻辑编排、无代码平台流程配置都有较好的应用

Downloads

40

Readme

Svg-Flow-Editor-MVP

项目简介

​ svg flow editor 是一款自研流程图编辑器,提供了一系列流程图交互、编辑所必需的功能,计划支持前端研发自定义开发各种逻辑编排场景,如流程图、ER 图、BPMN 流程等。项目使用typescript与svg、canvas等技术进行搭建,脱离vue、react等框架的限制,使得用户更快、更轻松融合到自己的项目中,意在工作审批配置、机器人逻辑编排、无代码平台流程配置都有较好的应用。

开源说明

本项目采用 Apache 2.0 开源策略

功能点

  • 流程图基本元件
    • 矩形 rect
    • 圆形 circle
    • 椭圆 ellipse
    • 菱形 diamond
    • 三角形 triangle
    • 五角星 star
    • 箭头 arrow
    • 表格 table
    • 统计图 echarts
    • 自定义图片 image
    • 自定义图标 icon
  • 元件基本操作
    • 创建、删除、移动、旋转、定位、属性修改、框选、多选(ctrl)、层级处理、文本显示等
  • 工具类
    • 图片导出
  • 直角折线
  • 全局 API
  • Command API
  • 右键菜单(内部、自定义)
  • 快捷键(内部、自定义)
  • event Bus 、 listener 事件监听机制
  • Canvas 实现背景网格、圆点、水印、辅助线
  • catalog、operation、footer、echart、websocket协同 插件化

待开发 - TODO

  • 拓展元件类型
    • 线段 line
    • HTML html
  • history 历史记录管理器(redo、undo、history)
  • tool工具类: 一键美化、组合/取消组合、锁定/取消锁定、
  • network协同: 支持用户光标、用户操作、聊天通信
  • 可视区渲染优化算法
  • uni-Flow : 项目重构-使用 canvas 实现绘制(可考虑使用 konva.js 作为底层技术支持)、考虑性能与渲染

使用方式

// 引入
import { SFEditor } from 'svg-flow-editor'

// 创建 editor 编辑器,需要传递编辑器 container,支持 css selector
const editor = new SFEditor(".flow-box");

// 创建矩形 Rect
const rect = editor.Rect(100, 70)

// 设置位置
rect.position(100, 100)

// SFEditor 所有的类、方法均支持链式调用,上诉代码等价于
new SFEditor(".flow-box").Rect(100, 70).position(100, 100)

// 示例:执行更换主题API
editor.command.setTheme('colorful_theme2')

// 示例:通过 listener 监听事件
editor.listener.graphLoaded = ()=> { // your code... }

// 取消事件监听
editor.listener.graphLoaded = null

// 示例:通过 event Bus 监听事件
editor.eventBus.on('graphLoaded',()=> { // your code... })

// 取消事件监听
editor.eventBus.off('graphLoaded')

// 示例:注册快捷键
editor.register.shortcutList = [
  {
    key: "s",
    ctrl: true,
    callback: () => {
      console.log("点击了Ctrl S");
    },
  },
];

// 示例:注册右键菜单
editor.register.contextMenuList = [
  {
    title: "测试右键菜单",
    callback: () => {
      console.log("右键菜单点击事件");
    },
  },
];

// 示例:调用全局API
editor.global.destroy()

// 示例:加载元件库插件
editor.plugin("pluginName")

// 示例:使用 Echart
const echart = editor.plugin("echart");
// 定义echart 数据
var data = ["Mon", "Tue", "Wed", "Thu", "Fri", "Sat", "Sun"];
var option = {
    xAxis: { type: "category", data, },
 	// ... other options
  };

// 初始化统计图
const line = echart?.init(option);

// 监听统计图事件
line.event.on('click', params => {
  // your code...
})

架构设计

架构设计

​ 项目对外暴露流程图操作对象 SFEditor,身上有用户操作的对象属性及方法,例如: svg 构造器(Rect、Circle、Ellipse)、command api操作、event事件中心以及全局api,通过暴露操作对象,实现对内部的数据访问、对象操作等。在核心模块中,需要考虑用户的使用习惯,封装完整的工具类,实现流程图的基本操作、拓展功能。底层依赖了svg对项目元件库的基础元件进行创作,同时使用了canvas对背景网格、水印等进行绘制,使用html进行页面布局,并且提供了typescript的全类型支持。 在API设计的设计上,采取了Command CommandAdapt 两个类实现,Command中不进行用户方法的直接处理,增加adapt类进行方法中转,防止用户通过API直接操作核心类。Command调用 adapt 的实例方法,在adapt 中获取draw、svg 等核心类进行用户的响应。

Graph 实例属性方法

graph.getElement()

  • 方法说明:获取元件SVG结构;
  • 返回值:SVGRectElement|SVGEllipseElement|HTMLDivElement;

graph.getID()

  • 方法说明:获取元件的ID属性;
  • 返回值:string;

graph.setID(nodeID: string)

  • 方法说明:设置元件的ID属性;
  • 返回值:graph实例;

graph.getX()

  • 方法说明:获取元件的x坐标(是graphBox的left属性);
  • 返回值:number;

graph.setX(x: number)

  • 方法说明:设置元件的x坐标;
  • 返回值:graph实例;

graph.getY()

  • 方法说明:获取元件的y坐标(是graphBox的top属性);
  • 返回值:number;

graph.setY(y: number)

  • 方法说明:设置元件的y坐标;
  • 返回值:graph实例;

graph.getWidth()

  • 方法说明:获取元件的宽度;
  • 返回值:number;

graph.setWidth(w: number)

  • 方法说明:设置元件的宽度;
  • 返回值:graph实例;

graph.getHeight()

  • 方法说明:获取元件的高度;
  • 返回值:number;

graph.setHeight(h: number)

  • 方法说明:设置元件的高度;
  • 返回值:graph实例;

graph.getStroke()

  • 方法说明:获取元件的边框;
  • 返回值:string;

graph.setStroke(stroke: string)

  • 方法说明:设置元件的边框;
  • 返回值:graph实例;

graph.getFill()

  • 方法说明:获取元件的填充样式;
  • 返回值:string;

graph.setFill(fill: string)

  • 方法说明:设置元件的填充;
  • 返回值:graph实例;

graph.getRotate()

  • 方法说明:获取元件的旋转角度;
  • 返回值:number;

graph.setRotate(rotate: number)

  • 方法说明:设置元件旋转角度;
  • 返回值:graph实例;

graph.getText()

  • 方法说明:获取元件的文本内容;
  • 返回值:string;

graph.setText(text: string)

  • 方法说明:设置元件的文本内容;
  • 返回值:graph实例;

graph.position(x: number, y: number)

  • 方法说明:复合属性,内部调用 setX()、setY() 实现;
  • 返回值:graph实例;

graph.getOption()

  • 方法说明:Echart 特有属性,获取统计图的配置信息;
  • 返回值:统计图配置信息;

graph.setOption(option:object)

  • 方法说明:Echart 特有属性,设置统计图配置信息;
  • 返回值:graph示例;

Command APIS

executeBackground(payload?: IBackground)

  • 方法说明:设置Canvas相关属性-网格、圆点、水印;
  • 返回值:void;
  • 参数说明:
    • origin boolean [可选] 是否显示圆点
    • originColor string [可选] 圆点的颜色
    • gridline boolean [可选] 是否显示网格线
    • gridlineColor string [可选] 网格线颜色
    • watermark boolean [可选] 是否显示水印
    • watermarkColor string [可选] 水印颜色
    • watermarkText string [可选] 水印显示文本
  • 用法示例:
1. 如果没有传任何参数,则表示关闭网格、关闭水印、关闭圆点
editor.command.executeBackground()

2. 只显示网格 
editor.command.executeBackground({gridline:true})

3. 只显示圆点 
editor.command.executeBackground({origin:true})

4. 只显示网格 
editor.command.executeBackground({watermark:true})


5. 内部实现原理:
 // 网格背景与小圆点背景互斥-原因是圆点的坐标直接取得网格的交点,导致网格显示后,圆点显示不明显
if (gridline) canvas.gridLine(gridlineColor);
else if (origin) canvas.origin(originColor);
// 水印则是独立存在
if (watermark) canvas.waterMark(watermarkText, watermarkColor);

executeAddGraph(payload: node)

  • 方法说明:设置Canvas相关属性-网格、圆点、水印;
  • 返回值:添加的元件实例 IGraph;
  • 参数说明:
    • type "rect" | "circle" | "ellipse" [必传] 添加元件类型
    • width number [必传] 元件的宽度
    • height number [必传] 元件的高度
    • nodeID string [可选] 元件的ID
    • x number [可选] 元件的x坐标
    • y number [可选] 元件的y坐标
    • rotate number [可选] 元件的旋转角度
    • stroke string [可选] 元件的边框颜色
    • fill string [可选] 元件的填充颜色
    • text string [可选] 元件文本
  • 用法示例:
1. 必传参数
editor.command.executeAddGraph({type:"rect",width:100,height:100})

2. 可选参数示例
editor.command.executeAddGraph({type:"rect",width:100,height:100,text:"demo"})

3. 圆形、椭圆参数说明
	** 圆形底层使用椭圆 Ellipse 实现,在宽高的设计上,使用短长轴设计:
    **  3.1. 想要半径50的圆,传入的宽高是半径的一半
    **  3.2. 想要椭圆类似,实际处理: setWidth(width*2) setHeight(height*2)

executeDeleteGraph()

  • 方法说明:删除元件;
  • 返回值:无;
  • 参数说明:无

setTheme(theme: string | IThemeOpt)

  • 方法说明:设置主题;
  • 返回值:无;
  • 参数说明:系统默认主题名称或者自定义的主题对象
  • 用法示例:
// 系统默认主题名称
editor.command.setTheme('colorful_theme1')

// 自定义主题
editor.command.setTheme({
  background?: string; // 背景颜色
  stroke?: string; // 元件边框颜色
  fill?: string; // 元件填充颜色
  text?: string; // 文本颜色
  line?: string; // 线条颜色
  auxiliaryLine?: string; // 辅助线颜色
})

executeUpdateGraph(payload: IUpdateGraph)

  • 方法说明:更新元件信息-用于 dialog 中修改元件样式;
  • 返回值:void;
  • 参数说明:更新元件基本参数
  • 用法示例:
// demo
editor.command.executeUpdateGraph({fill:'red'})

// 参数说明
nodeID?: string[]; // 需要更新的元件 ID 支持数组
stroke?: string; // 需要修改的边框颜色
fill?: string; // 需要修改的填充颜色
strokeWidth?: number; // 需要修改的边框宽度
radius?: number; // 需要修改的圆角尺寸
dasharray?: string; // 需要修改的虚线配置 为字符串,类似 '5,5',为svg 虚线配置

executeTop(nodeID:string)

  • 方法说明:置于顶层;
  • 返回值:void;
  • 参数说明:指定某个元件的ID

executeBottom(nodeID:string)

  • 方法说明:置于底层;
  • 返回值:void;
  • 参数说明:指定某个元件的ID

executeHoldUp(nodeID:string)

  • 方法说明:上移一层;
  • 返回值:void;
  • 参数说明:指定某个元件的ID

executePutDown(nodeID:string)

  • 方法说明:下移一层;
  • 返回值:void;
  • 参数说明:指定某个元件的ID

executeFullScreen()

  • 方法说明:进入全屏;
  • 返回值:void;

executeExitFullScreen()

  • 方法说明:退出全屏;
  • 返回值:void;

executePageScaleRecovery()

  • 方法说明:重置页面缩放;
  • 返回值:void;

executePageScaleMinus()

  • 方法说明:缩小页面;
  • 返回值:void;

executePageScaleAdd()

  • 方法说明:放大页面;
  • 返回值:void;

setPageScale(scale: number)

  • 方法说明:缩放页面至指定倍率;
  • 参数说明: scale 支持 0.4 至 2 之间(40% - 200%缩放比)
  • 返回值:void;

executeSearchReplace(keyword?: string)

  • 方法说明:打开搜索替换框;
  • 参数说明: keyword 默认搜索文本 [可选参数]
  • 返回值:void;
  • 用法实例:
// 唤起搜索替换框
editor.command.executeSearchReplace()

// 唤起搜索替换框,并传入默认搜索关键字
editor.command.executeSearchReplace('123')

executeSearchPre()

  • 方法说明:搜索上一处;
  • 参数说明: 无
  • 返回值:void;

executeSearchNext()

  • 方法说明:搜索上一处;
  • 参数说明: 无
  • 返回值:void;

executeReplace(newWord: string)

  • 方法说明:替换当前;
  • 参数说明: 替换的新值
  • 返回值:void;

executeReplaceAll(newWord: string)

  • 方法说明:替换全部;
  • 参数说明: 替换的新值
  • 返回值:void;

executeUpdateText(nodeID: string[], key: textType, color?: string)

  • 方法说明:进行文本样式调整;
  • 参数说明:
    • nodeID 元件 ID 集合
    • key 支持修改的属性:"bold" | "italic" | "underline" | "textcolor"
    • color: 需要修改的新值(仅当 key='textcolor'时,color为必传,仅支持16进制颜色值)
  • 返回值:void;
  • 用法示例:
editor.command.executeUpdateText(['yF48bz3Cptl2egTy1BAv5'],'bold')
editor.command.executeUpdateText(['yF48bz3Cptl2egTy1BAv5'],'italic')
editor.command.executeUpdateText(['yF48bz3Cptl2egTy1BAv5'],'bold')
editor.command.executeUpdateText(['yF48bz3Cptl2egTy1BAv5'],'textcolor','#ccc')

setPageSize(w: number, h: number)

  • 方法说明:设置页面尺寸;
  • 参数说明: w - 页面宽度 h - 页面高度
  • 返回值:void;

executeScreenShot(filetype?:string)

  • 方法说明:进行图片下载(截图);
  • 参数说明:保存的文件类型 - 目前支持 png 、jpg 格式
  • 返回值:void;

事件监听(listener)

loaded

  • 方法说明:编辑器加载完成;
  • 用法示例
editor.listener.loaded = () => {
    // your code...
};

graphResized

  • 方法说明:元件重置大小;
  • 回调参数说明:
    • nodeID string
    • **width ** number
    • **height **number
    • **x **number
    • **y **number
  • 用法示例
editor.listener.graphResized = (payload) => {
    // your code...
};

destroyed

  • 方法说明:编辑器销毁完成;
  • 用法示例
editor.listener.destroyed = () => {
    // your code...
};

graphNumberChanged

  • 方法说明:元件数量变化回调;
  • 回调参数说明:
    • number目前编辑器上的元件数量
  • 用法示例
editor.listener.graphNumberChanged = (number) => {
    // your code...
};

pageScale

  • 方法说明:页面缩放;
  • 回调参数说明:
    • **scale **缩放比 0.4 - 2之间
  • 用法示例
editor.listener.pageScale = (scale) => {
    // your code...
};

事件监听(eventBus)

loaded

  • 方法说明:编辑器加载完成;
  • 用法示例
editor.eventBus.on('loaded',()=>{
    // your code
})

graphResized

  • 方法说明:元件重置大小;
  • 回调参数说明:
    • **nodeID **string
    • **width **number
    • **height **number
    • **x **number
    • **y **number
  • 用法示例
editor.eventBus.on('graphResized',(payload)=>{
    // your code...
});

destroyed

  • 方法说明:编辑器销毁完成;
  • 用法示例
editor.eventBus.on('destroyed',()=>{
    // your code...
});

graphNumberChanged

  • 方法说明:元件数量变化回调;
  • 回调参数说明:
    • **number **目前编辑器上的元件数量
  • 用法示例
editor.eventBus.on('graphNumberChanged',(number)=>{
    // your code...
});

pageScale

  • 方法说明:页面缩放;
  • 回调参数说明:
    • **scale **缩放比 0.4 - 2之间
  • 用法示例
editor.eventBus.on('pageScale',(scale)=>{
    // your code...
});

内部快捷键

  • Ctrl + 左键 多选
  • Ctrl + C 复制
  • Ctrl + V 粘贴
  • Ctrl + X 剪切
  • Ctrl + A 全选
  • backspace、delete 进行删除
  • 上下左右 移动
  • Ctrl + Z 撤销 undo
  • Ctrl + Y 重做 redo
  • Ctrl + P 打印
  • Ctrl + S 保存
  • Ctrl + Shift + S 进行另存为
  • Alt + Shift + F 快捷美化
  • ...

自定义快捷键

editor.register.shortcutList=[
    {
      key: KeyMap;
      ctrl?: boolean;
      meta?: boolean;
      mod?: boolean; // windows:ctrl || mac:command
      shift?: boolean;
      alt?: boolean;
      isGlobal?: boolean;
      callback?: (command: Command) => any;
      disable?: boolean;
    }
  ]

内部右键菜单

自定义右键菜单

全局API

editor.destroy()

  • 方法说明:销毁编辑器

插件的使用

const editor = new SFEditor('.box')

// 使用 footer 插件 - 无返回值
editor.plugin('footer')

// 使用元件库插件 - 无返回值
editor.plugin('catalog')

// 使用顶部菜单栏插件 - 无返回值
editor.plugin('operation')

// 使用统计图插件-自定义统计图
const echart = editor.plugin('echart') // 只有注册统计图插件,才可通过外部访问、创建统计图,并且只有 echart 会返回实例对象

// 这个配置信息是官网的样例
const option = {
  xAxis: {
    type: 'category',
    data: ['Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat', 'Sun']
  },
  yAxis: {
    type: 'value'
  },
  series: [
    {
      data: [820, 932, 901, 934, 1290, 1330, 1320],
      type: 'line',
      smooth: true
    }
  ]
};

// 初始化统计图
const line = echart.init(option)

// 监听事件回调
line.event.on('click',params => {
    // your code...
})

// 外部更新了数据后,执行 setOption 进行重绘
option.xAxis.data[0] = '快乐星期一'
line.setOption(option)

// 不同的插件之间无任何关联,可以任选插件进行注册使用

版本更替说明

  • 1.0.1

    • 实现元件基本绘制
    • 实现形变锚点、连接锚点、旋转事件
    • 实现 contenteditable文本输入
    • 实现右键菜单、canvas 拖拽辅助线、层级处理优化等
    • 完成项目 footer、operation、catalog 插件
    • 优化自定义事件、自定义快捷键相关BUG
  • 1.0.2

    • 优化 worker 打包后路径异常问题
  • 1.0.6

    • 优化静态资源打包后请求 404 相关问题
    • 优化项目打包后入口文件
  • 1.0.7

    • 实现 svg image 图片创建与加载
    • 实现自定义图片上传、icon 图标
    • 实现 统计图
    • 实现直角折线算法、关键点获取算法、A*算法
  • 1.0.13

    • 重构文本底层实现,摈弃 svg text 的实现方式
    • 实现搜索替换、dialog公共组件
    • 优化打包后 rotate cursor 异常问题
    • 实现 websocket 初步协同、服务端 node demo
    • 实现 Yjs 关键冲突、合并方法重写
    • 优化 dialog 事件响应机制,拓展dialog应用场景
    • 优化相干BUG
  • 1.0.15

    • 重写层级相关API,便于实现协同
    • 修复拖拽经过元素时位置异常问题
    • 实现GTable表格,优化 table 样式
    • 实现相关 Command API
    • 优化 nextTick 实现方式