请大家动动小手,给我一个免费的 Star 吧~

这一章实现导入导出为JSON文件、另存为图片、上一步、下一步。

github源码

gitee源码

示例地址

导出为JSON文件

提取需要导出的内容

  getView() {
// 复制画布
const copy = this.render.stage.clone()
// 提取 main layer 备用
const main = copy.find('#main')[0] as Konva.Layer
// 暂时清空所有 layer
copy.removeChildren() // 提取节点
let nodes = main.getChildren((node) => {
return !this.render.ignore(node) && !this.render.ignoreDraw(node)
}) // 重新装载节点
const layer = new Konva.Layer()
layer.add(...nodes)
nodes = layer.getChildren() // 计算节点占用的区域
let minX = 0
let maxX = copy.width()
let minY = 0
let maxY = copy.height()
for (const node of nodes) {
const x = node.x()
const y = node.y()
const width = node.width()
const height = node.height() if (x < minX) {
minX = x
}
if (x + width > maxX) {
maxX = x + width
}
if (y < minY) {
minY = y
}
if (y + height > maxY) {
maxY = y + height
} if (node.attrs.nodeMousedownPos) {
// 修正正在选中的节点透明度
node.setAttrs({
opacity: copy.attrs.lastOpacity ?? 1
})
}
} // 重新装载 layer
copy.add(layer) // 节点占用的区域
copy.setAttrs({
x: -minX,
y: -minY,
scale: { x: 1, y: 1 },
width: maxX - minX,
height: maxY - minY
}) // 返回可视节点和 layer
return copy
}

1、首先复制一份画布

2、取出 main layer

3、筛选目标节点

4、计算目标节点占用区域

5、调整拷贝画布的位置和大小

导出 JSON

使用 stage 的 toJSON 即可。

  // 保存
save() {
const copy = this.getView() // 通过 stage api 导出 json
return copy.toJSON()
}

导入 JSON,恢复画布

相比导出,过程会比较复杂一些。

恢复节点结构

  // 恢复
async restore(json: string, silent = false) {
try {
// 清空选择
this.render.selectionTool.selectingClear() // 清空 main layer 节点
this.render.layer.removeChildren() // 加载 json,提取节点
const container = document.createElement('div')
const stage = Konva.Node.create(json, container)
const main = stage.getChildren()[0]
const nodes = main.getChildren() // 恢复节点图片素材
await this.restoreImage(nodes) // 往 main layer 插入新节点
this.render.layer.add(...nodes) // 上一步、下一步 无需更新 history 记录
if (!silent) {
// 更新历史
this.render.updateHistory()
}
} catch (e) {
console.error(e)
}
}

1、清空选择

2、清空 main layer 节点

3、创建临时 stage

4、通过 Konva.Node.create 恢复 JSON 定义的节点结构

5、恢复图片素材(关键)

恢复图片素材

  // 加载 image(用于导入)
loadImage(src: string) {
return new Promise<HTMLImageElement | null>((resolve) => {
const img = new Image()
img.onload = () => {
// 返回加载完成的图片 element
resolve(img)
}
img.onerror = () => {
resolve(null)
}
img.src = src
})
}
// 恢复图片(用于导入)
async restoreImage(nodes: Konva.Node[] = []) {
for (const node of nodes) {
if (node instanceof Konva.Group) {
// 递归
await this.restoreImage(node.getChildren())
} else if (node instanceof Konva.Image) {
// 处理图片
if (node.attrs.svgXML) {
// svg 素材
const blob = new Blob([node.attrs.svgXML], { type: 'image/svg+xml' })
// dataurl
const url = URL.createObjectURL(blob)
// 加载为图片 element
const image = await this.loadImage(url)
if (image) {
// 设置图片
node.image(image)
}
} else if (node.attrs.gif) {
// gif 素材
const imageNode = await this.render.assetTool.loadGif(node.attrs.gif)
if (imageNode) {
// 设置图片
node.image(imageNode.image())
}
} else if (node.attrs.src) {
// 其他图片素材
const image = await this.loadImage(node.attrs.src)
if (image) {
// 设置图片
node.image(image)
}
}
}
}
}

关于恢复 svg,关键在于拖入 svg 的时候,记录了完整的 svg xml 在属性 svgXML 中。

关于恢复 gif、其他图片,拖入的时候记录其 src 地址,就可以恢复到节点中。

上一步、下一步

其实就是需要记录历史记录

历史记录

  history: string[] = []
historyIndex = -1 updateHistory() {
this.history.splice(this.historyIndex + 1)
this.history.push(this.importExportTool.save())
this.historyIndex = this.history.length - 1
// 历史变化事件
this.config.on?.historyChange?.(_.clone(this.history), this.historyIndex)
}

1、从当前历史位置,舍弃后面的记录

2、从当前历史位置,覆盖最新的 JSON 记录

3、更新位置

4、暴露事件(用于外部判断历史状态,以此禁用、启用上一步、下一步)

更新历史记录

一切会产生变动的位置都执行 updateHistory,如拖入素材、移动节点、改变节点位置、改变节点大小、复制粘贴节点、删除节点、改变节点的层次。具体代码就不贴了,只是在影响的地方执行一句:

this.render.updateHistory()

上一步、下一步方法

  prevHistory() {
const record = this.history[this.historyIndex - 1]
if (record) {
this.importExportTool.restore(record, true)
this.historyIndex--
// 历史变化事件
this.config.on?.historyChange?.(_.clone(this.history), this.historyIndex)
}
} nextHistory() {
const record = this.history[this.historyIndex + 1]
if (record) {
this.importExportTool.restore(record, true)
this.historyIndex++
// 历史变化事件
this.config.on?.historyChange?.(_.clone(this.history), this.historyIndex)
}
}

另存为图片

  // 获取图片
getImage(pixelRatio = 1, bgColor?: string) {
// 获取可视节点和 layer
const copy = this.getView() // 背景层
const bgLayer = new Konva.Layer() // 背景矩形
const bg = new Konva.Rect({
listening: false
})
bg.setAttrs({
x: -copy.x(),
y: -copy.y(),
width: copy.width(),
height: copy.height(),
fill: bgColor
}) // 添加背景
bgLayer.add(bg) // 插入背景
const children = copy.getChildren()
copy.removeChildren()
copy.add(bgLayer)
copy.add(children[0], ...children.slice(1)) // 通过 stage api 导出图片
return copy.toDataURL({ pixelRatio })
}

主要关注有2点:

1、插入背景层

2、设置导出图片的尺寸

导出的时候,其实就是把当前矢量、非矢量素材统一输出为非矢量的图片,设置导出图片的尺寸越大,可以保留更多的矢量素材细节。

接下来,计划实现下面这些功能:

  • 实时预览窗
  • 对齐效果
  • 连接线
  • 等等。。。

是不是值得更多的 Star 呢?勾勾手指~

源码

gitee源码

示例地址

前端使用 Konva 实现可视化设计器(7)- 导入导出、上一步、下一步的更多相关文章

  1. 惊闻企业Web应用生成平台 活字格 V4.0 免费了,不单可视化设计器免费,服务器也免费!

    官网消息: 针对活字格开发者,新版本完全免费!您可下载活字格 Web 应用生成平台 V4.0 Updated 1,方便的创建各类 Web 应用系统,任意部署,永不过期. 我之前学习过活字格,也曾经向用 ...

  2. ActiveReports 9 新功能:可视化查询设计器(VQD)介绍

    在最新发布的ActiveReports 9报表控件中添加了多项新功能,以帮助你在更短的时间里创建外观绚丽.功能强大的报表系统,本文将重点介绍可视化数据查询设计器,无需手动编写任何SQL语句,主要内容如 ...

  3. (原创)【B4A】一步一步入门02:可视化界面设计器、控件的使用

    一.前言 上篇 (原创)[B4A]一步一步入门01:简介.开发环境搭建.HelloWorld 中我们创建了默认的项目,现在我们来看一下B4A项目的构成,以及如何所见即所得的设计界面,并添加和使用自带的 ...

  4. Windows Phone 十二、设计器同步

    在设计阶段为页面添加数据源 Blend或者VS的可视化设计器会跑我们的代码,然后来显示出来,当我们Build之后,设计器会进入页面的构造函数,调用InitializeComponent();方法来将U ...

  5. WinForms项目升级.Net Core 3.0之后,没有WinForm设计器?

    目录 .NET Conf 2019 Window Forms 设计器 .NET Conf 2019 2019 9.23-9.25召开了 .NET Conf 2019 大会,大会宣布了 .Net Cor ...

  6. C#基础系列:开发自己的窗体设计器(PropertyGrid显示中文属性名)

    既然是一个窗体设计器,那就应该能够设置控件的属性,设置属性最好的当然是PropertyGrid了,我们仅仅需要使用一个PropertyGrid.SelectedObject = Control就可以搞 ...

  7. 通过用 .NET 生成自定义窗体设计器来定制应用程序

    通过用 .NET 生成自定义窗体设计器来定制应用程序 https://www.microsoft.com/china/MSDN/library/netFramework/netframework/Cu ...

  8. Qt编写自定义控件属性设计器

    以前做.NET开发中,.NET直接就集成了属性设计器,VS不愧是宇宙第一IDE,你能够想到的都给你封装好了,用起来不要太爽!因为项目需要自从全面转Qt开发已经6年有余,在工业控制领域,有一些应用场景需 ...

  9. VS2015 android 设计器不能可视化问题解决。

    近期安装了VS2015,体验了一下android 的开发,按模板创建执行了个,试下效果非常不错.也能够可视化设计.但昨天再次打开或创建一个android程序后,设计界面直接不能显示,显示错误:(可能是 ...

  10. 可视化流程设计——流程设计器演示(基于Silverlight)

    上一篇文章<通用流程设计>对鄙人写的通用流程做了一定的介绍,并奉上了相关源码.但一个好的流程设计必少不了流程设计器的支持,本文将针对<通用流程设计>中的流程的设计器做一个简单的 ...

随机推荐

  1. 记录--如何在H5中实现OCR拍照识别身份证功能

    这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助 业务背景 由于当前项目中需要实现身份证拍照识别的功能,如果是小程序可以使用微信提供的 ocr-navigator 插件实现,但是在企业微信 ...

  2. 记录--a标签跳转新地址无法访问,但手动输入新地址可以访问

    这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助 问题描述 最近遇到一个有意思的问题,项目中有一个地方,点击需要跳转到一个新的域名地址 笔者使用a标签做跳转,跳是跳过去了,可是跳过去以后, ...

  3. FreeRTOS教程10 低功耗

    1.准备材料 正点原子stm32f407探索者开发板V2.4 STM32CubeMX软件(Version 6.10.0) Keil µVision5 IDE(MDK-Arm) 野火DAP仿真器 XCO ...

  4. [Spring]aop的配置与使用

    [版权声明]未经博主同意,谢绝转载!(请尊重原创,博主保留追究权) https://blog.csdn.net/m0_69908381/article/details/129907717 出自[进步* ...

  5. WPF实现树形表格控件(TreeListView)

    前言 本文将探讨如何利用WPF框架实现树形表格控件,该控件不仅能够有效地展示复杂的层级数据,还能够提供丰富的个性化定制选项.我们将介绍如何使用WPF提供的控件.模板.布局.数据绑定等技术来构建这样一个 ...

  6. #原根,BSGS,扩欧,矩阵乘法#CF1106F Lunar New Year and a Recursive Sequence

    题目 已知数列 \(f\) 满足 \(f_{1\sim k-1}=1\) 且 \(f_n=m\), 并且知道 \(f_i=(\prod_{j=1}^kf_{i-j}b_j)\bmod{99824435 ...

  7. #01-Trie,Cayley定理#51nod 1601 完全图的最小生成树计数

    题目 分析 考虑建出一棵Trie,然后最小生成树就是0的部分到1的部分连一条边, 这个可以用区间短的一方查询另一棵trie,这样时间复杂度为 \(O(n\log^2{mx})\) 方案数注意相同的 \ ...

  8. 【Kotlin】List、Set、Map简介

    1 List ​ Java 的 List.Set.Map 介绍见 → Java容器及其常用方法汇总. 1.1 创建 List 1.1.1 emptyList var list = emptyList& ...

  9. 如何跑各种check

    如何进行 Fastcheck? 首先,导入环境变量: export CODE_BASE=/data/openGauss-server export BINARYLIBS=/data/openGauss ...

  10. 深入理解DES算法:原理、实现与应用

    title: 深入理解DES算法:原理.实现与应用 date: 2024/4/14 21:30:21 updated: 2024/4/14 21:30:21 tags: DES加密 对称加密 分组密码 ...