图表即代码是将图表以领域特定语言作为载体,围绕于不同的使用场景,转译生成二次产物 —— 如概念图、架构图、软件架构等。
对于造图形库这个库,我的想法由来已久。然而,直到最近,积压的需求越来越多的时候:
于是,在挖坑之前,我开始思索我要构建的是怎样一个图形库。值得庆幸的是:哪怕不存在上述的三个原因,我也打算造一个轮子。当然,之前的重点可能不是可用,现在必须要提供一个可用的轮子。
作为一个《万物即代码》(https://ascode.ink/)的先行者,对于图表来说,它的可版本化管理依旧是一个痛点。所以,在这个新的工具实现图表即代码,依旧是一个非常有意思的探索点。
图表即代码(Diagram as Code)是一个已经有一定基础的领域,在我与我的同事们一起构建开源应用 Ledge 的时候,我们已经大量地采用了这个思想。稍有不同的是,Ledge 的图表即代码偏向于是数据可视化,而我们即将要构建 Feakin 偏向于是概念/想法的可视化。但是,这并不影响,我们再次定义一下图表即代码。在先前的《文档代码化》 中,我们定义的文档代码化是:
文档代码化,将文档以类代码的领域特定语言的方式编写,并借鉴软件开发的方式(如源码管理、部署)进行管理。它可以借助于特定的工具进行编辑、预览、查看,又或者是通过专属的系统部署到服务器上。面向非技术人员的文档代码化的一种常见架构模式是:编辑-发布-开发分离』,
而对于图表即代码来说,它是可以相似的方式来定义的:
图表即代码即将图表以领域特定语言作为载体,围绕于不同的使用场景,转译生成二次产物 —— 如概念图、架构图、软件架构等。
这个定义从某种意义上是围绕于现有的工具而产生的,诸如于:
对于这样的系统,我想大家都知道如何去设计了。或者说,至少在心底是有个印象。
作为代码化的第一要素,它必须是采用 DSL (领域特定语言) 的设计,才能有效地进行代码化。值得注意的是不要畏惧 DSL:采用领域特定语言,并不意味着特别复杂的编译实现,哪怕是 JSON 格式描述,也可以适为一种 DSL。所以,诸如于 Graphviz 中设计的 DOT Language 就非常的简单:
digraph G {
a -> b
b -> c
}
简单的语法,可以生成非常有用的图形。只在我们需要一些额外的配置时,才需要去翻看对应的文档。而如果我们能提供更多的 samples,那么就能降低查看文档的成本,构建更好的开发体验。
从另外一个层面来说,图形的序列化结果,其实也算得是上一种领域特定语言。诸如于 .excalidraw
的 JSON 形式,.drawio
文件采用的编码后的 mxgraph 的 XML 格式,它们都是图形的一种类型的领域特定语言。
对于代码生成图形来说,用过 D3.js 或者是 Echart.js 的小伙伴,对于 Dagre、ForceLayout 等一系列的图形自动布局算法不陌生。在这里就不展开了 —— 主要是我也不是算法专家。
随后,布局的计算依赖于数据 + 模型,对于一个图表既代码的系统来说:
a
和 b
是节点,而 →
是指向关系。当然了,如果能提供一个抽象的算法接口,以接入更多的布局算法,那么就可以大大提高系统的灵活性。在这一点上 Cytoscape.js 就做得挺好的,提供了 ELK、CoSE、Cola、fCoSE 等算法的接入,底层的灵活性会带来更多的可扩展空间。
从现实的因素来考虑,并非所有的图表都应该用图表即代码的方式。人们采用图表即代码这种方式,也意味着:基于可视化的结果,进行后续的活动。诸如于:
也因此,与其说是图形即代码,不如说图形化只是中间的产物,作为沟通时的信息载体。在这点上,它与设计即代码颇为相似,DSL 充当的是图形的标准化输出。
与上述的内容相比,在代码与图形之间提供双向绑定显得非常有意思。代码化可以向程序员提供高效的输入方式,但是正如新手程序不习惯用 Terminal 一样,他们也需要图形化的方式。于是呢,如何在改变图形的同时,更新代码就变得非常有意思了。从结果上来说,图表工具在保存的时候,存储的是数据模型,而模型便是这个双向绑定的基础。如在使用 draw.io 这样的可视化工具时,当我们添加新的矩形、连接时,结果会更新到对应的数据模型中。
而图形化的编辑呢,存在一些额外的动作(action),如我们在撤销(undo)、重做(redo)的时候,要提供这种模型的重载。这些因素显然会带来一些额外的工作量。
于是乎,为了在 ArchGuard 和 Quake 中采用,我便在构思如何去设计这样一个图形工具,名为 Feakin —— 为了注册到 GitHub 的组织里:https://github.com/feakin/ 。随后,为了和 fxxk 做一个区分,Logo 是: f(k)
围绕于如何通过概念来构建系统 —— 即我们如何通过图形来传送想法,是 Feakin 的核心考虑因素。也因此如何支持层次化的思维表达,是 Feakin 的一个重点,也因此,诸如于 Echart.js、Ant G6 等面向数据的图形引擎,并不是 Feakin 的同类型产品。Feakin 的同类型工具是:Drawio、Excalidraw、Graphviz、Mermaid 这一类程序员工具箱里的工具。
起初,在设计 Feakin 的时候,我只想构建一个 Draw.io 的兼容引擎。于是,构建了一个 PoC:https://feakin.github.io/drawio-render-poc/ 。只是呢:
Draw.io 的其中一个可参考的点是 —— 内置了对其它图表库的支持,如 Mermaid、OrgChart 等等。
所以,如何构建一个图表的抽象模型,并提供它们的转换就非常有必要。
随后,在有了模型之后,我们就需要 Render。当前在 PoC 版本里,我们用的是同事建议的:react-konvas,一个基于 Canvas 的工具库。从功能上来说,它似乎能满足当前的需求。
Demo 地址:https://github.com/feakin/diagram-render
不过呢,我正在慢慢学习不同图表工具的设计,所以进展不是很理想。现在只支持基本的渲染 + 布局接口。
当然,还只是想想。
啊,说实话,其实就是只有一个想法 + PoC。
对于图形化底层引擎的开发,我算是个新手,如果你也有兴趣,欢迎来加入我们:https://github.com/feakin/ 。
最后,如何划定一个合理的边界,以让 Feakin 不臃肿就是一个值得深思的问题了。
围观我的Github Idea墙, 也许,你会遇到心仪的项目