dotnet 文档应用的撤销重做设计

2024-02-19 20:30

本文主要是介绍dotnet 文档应用的撤销重做设计,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文档应用是指如 Word 或 PPT 等的提供给用户进行内容创作的工具,而撤销重做其实也被称为撤销恢复功能。本文来告诉大家撤销重做这个模块的设计路线,从简单的复杂

大部分的应用软件都可以采用敏捷开发,不断进行迭代。应用的每个小功能都在不断迭代中,成为模块或者某个团队产品。撤销重做功能也是可以从一个小功能,迭代成为一个文档的核心模块

在软件开始开发的时候,很少会有人能了解这个软件产品的未来,如果此时就给很多小功能模块投入大量的资源,那大部分都会是浪费的。本文记录的功能的迭代也仅仅只是在我当前团队里面,跟随产品逐步修改的,不一定适合你当前的团队

本文以下的撤销重做和撤销恢复说的是相同的功能。但是本质上这个词是我当前团队用错的,如在Word里面的重做,也就是标题上左上角的按钮,其实指的是当前的输入再次输入,而恢复只有在用户点击撤销之后,才会看到恢复按钮

默认在 WPF 或 UWP 等应用的文本框或者富文本框里面都有自带的撤销恢复机制,只要自己不作,那么就依靠这个功能就能玩很长时间了

当更多的需要自己开始自定义各个控件的时候,此时就很难用上框架提供的撤销重做,需要自己定义一套撤销重做机制。从需求层面上讲,撤销就是撤回到上一个步骤,而重做或者说恢复其实就是在恢复撤销的步骤。可以看到越在后面添加的操作,在撤销的时候越快进行撤销。而越早撤销的操作,在重做的时候就越早重做。刚好,这就是数据结构的栈的定义,先进入的数据后拿出,后进入的数据先拿出

撤销重做的数据结构层面使用栈是最合适的,在使用了 栈 之后,撤销重做模块就有了一个概念叫 撤销重做栈

在软件开发里面,很多开发的开始是在定义数据结构或者说在设计类,但按照本文的编写方法,如果一开始就来开设计类,我预计将会十分无聊

在定义好了 撤销重做栈 之后,咱将会遇到一个问题,那就是这个 撤销重做栈 的代码应该如何写?在 dotnet 里面 Stack 可以表示栈这个数据结构,而 Stack 推荐是使用泛形的 Stack<T> 那问题是这个 T 应该是什么

假定咱的文档应用可以输入的内容,不单是文本,加入咱还可以输入图片,而图片还可以被拖动修改坐标,图片可以被修改颜色等,可以看到咱的应用可以输入的内容有很多不同的业务定义。因此咱需要有一个足够通用类型用来定义撤销重做操作

最基础的撤销重做操作其实只有两个动作,一个是就是被撤销,另一个就是被重做恢复,可以定义的类型如下

	interface IOperation{void Undo();void Redo();}

为什么我上面会推荐定义为接口而不是抽象类?原因是在 C# 里面是单继承的,如果是抽象的类,将会让某些业务的代码不好编写。有些业务的代码已经需要继承某个类了,而如果此时这个类需要插入到 撤销重做栈 将会发现不能再继承一个抽象类。另外,从撤销重做的业务上,也不需要使用抽象类,只需要有撤销和重做两个方法就可以

在应用程序可以根据业务定义多个撤销重做栈的内容,例如说做一个和 PPT 差很多的软件,有编辑和播放两个不同的界面,这两个界面的撤销重做相互独立,那么在这两个模式里面就可以定义两个不同的 撤销重做栈 对象

在撤销重做栈这个类型里面,最简单的版本是只有两个 Stack 一个是撤销另一个是恢复。在用户输入操作的时候,将操作放入到撤销的栈。在用户重做恢复时,从撤销的栈弹出操作,放入到重做恢复的栈里

随着业务的迭代,其实纯撤销重做栈会有一些通用的撤销恢复的功能还需要额外开发

提供当前合入多个不同的业务的操作做一个的业务,例如我有图片编辑模块,这个模块的编辑每一步默认都会作为一个操作加入到撤销重做栈,而我还有另一个是文本编辑模块,每一个文本编辑步骤就是一个操作。在我进入特殊的模式,例如是插入某个复杂元素,如公式,允许在公式里面编辑文本和图片。此时在插入公式过程中,编辑文本和图片每一步都可以撤销,而在插入公式完成之后,撤销的是整个公式。也就是说允许在底层撤销重做提供的一个功能,这个功能相当于一个开关,在打开的时候,插入的所有操作在开关关闭的时候将会合并成为一个操作的功能。在撤销重做模块提供这个功能可以方便业务端解耦,而这个功能就需要定义的操作类型里面有组合操作才能实现

咱上文说的操作,是指继承了 IOperation 接口的类。基本上每一个操作都是独立的,单独的。而组合操作是特殊的,在组合操作里面将会包含其他的多个操作,将会在撤销恢复时按照顺序进行撤销恢复

在实现了撤销重做的功能,每个业务都需要有 IOperation 来表示业务的用户输入,而刚好如果有漫游同步的功能,这个功能也是需要用户的输入。如果一开始在软件开发就判断有漫游同步的功能,那么将漫游同步和撤销重做同步是一个很好的设计

每一个用户的输入都可以被抽象为一个 IOperation 同时也是一个同步的对象,刚好两个功能可以作为一个功能实现,开发的工作量可以更少。如果有这样的需求,那么对于 IOperation 的设计上,就需要开发者设置为基于数据,不能基于对象的动作

另外,即使没有漫游同步的功能,其实文档保存也可以复用撤销重做提供的功能。在文档保存的时候,很多文档软件都有自动保存的功能,如 VS 软件。在文档内容很多,保存一次需要大量的时候时,就需要用到增量的功能,那么如何实现增量?如果文档可以划分为不同的元素,根据元素是否加入撤销重做就可以了解元素或页面等有没进行编辑,从而决定是否保存

我搭建了自己的博客 https://blog.lindexi.com/ 欢迎大家访问,里面有很多新的博客。只有在我看到博客写成熟之后才会放在csdn或博客园,但是一旦发布了就不再更新

如果在博客看到有任何不懂的,欢迎交流,我搭建了 dotnet 职业技术学院 欢迎大家加入

如有不方便在博客评论的问题,可以加我 QQ 2844808902 交流

知识共享许可协议
本作品采用知识共享署名-非商业性使用-相同方式共享 4.0 国际许可协议进行许可。欢迎转载、使用、重新发布,但务必保留文章署名林德熙(包含链接:http://blog.csdn.net/lindexi_gd ),不得用于商业目的,基于本文修改后的作品务必以相同的许可发布。如有任何疑问,请与我联系。

这篇关于dotnet 文档应用的撤销重做设计的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/725861

相关文章

使用EasyPoi快速导出Word文档功能的实现步骤

《使用EasyPoi快速导出Word文档功能的实现步骤》EasyPoi是一个基于ApachePOI的开源Java工具库,旨在简化Excel和Word文档的操作,本文将详细介绍如何使用EasyPoi快速... 目录一、准备工作1、引入依赖二、准备好一个word模版文件三、编写导出方法的工具类四、在Export

利用Python操作Word文档页码的实际应用

《利用Python操作Word文档页码的实际应用》在撰写长篇文档时,经常需要将文档分成多个节,每个节都需要单独的页码,下面:本文主要介绍利用Python操作Word文档页码的相关资料,文中通过代码... 目录需求:文档详情:要求:该程序的功能是:总结需求:一次性处理24个文档的页码。文档详情:1、每个

C++读写word文档(.docx)DuckX库的使用详解

《C++读写word文档(.docx)DuckX库的使用详解》DuckX是C++库,用于创建/编辑.docx文件,支持读取文档、添加段落/片段、编辑表格,解决中文乱码需更改编码方案,进阶功能含文本替换... 目录一、基本用法1. 读取文档3. 添加段落4. 添加片段3. 编辑表格二、进阶用法1. 文本替换2

Java中的分布式系统开发基于 Zookeeper 与 Dubbo 的应用案例解析

《Java中的分布式系统开发基于Zookeeper与Dubbo的应用案例解析》本文将通过实际案例,带你走进基于Zookeeper与Dubbo的分布式系统开发,本文通过实例代码给大家介绍的非常详... 目录Java 中的分布式系统开发基于 Zookeeper 与 Dubbo 的应用案例一、分布式系统中的挑战二

Java 缓存框架 Caffeine 应用场景解析

《Java缓存框架Caffeine应用场景解析》文章介绍Caffeine作为高性能Java本地缓存框架,基于W-TinyLFU算法,支持异步加载、灵活过期策略、内存安全机制及统计监控,重点解析其... 目录一、Caffeine 简介1. 框架概述1.1 Caffeine的核心优势二、Caffeine 基础2

使用Node.js和PostgreSQL构建数据库应用

《使用Node.js和PostgreSQL构建数据库应用》PostgreSQL是一个功能强大的开源关系型数据库,而Node.js是构建高效网络应用的理想平台,结合这两个技术,我们可以创建出色的数据驱动... 目录初始化项目与安装依赖建立数据库连接执行CRUD操作查询数据插入数据更新数据删除数据完整示例与最佳

Python实现自动化删除Word文档超链接的实用技巧

《Python实现自动化删除Word文档超链接的实用技巧》在日常工作中,我们经常需要处理各种Word文档,本文将深入探讨如何利用Python,特别是借助一个功能强大的库,高效移除Word文档中的超链接... 目录为什么需要移除Word文档超链接准备工作:环境搭建与库安装核心实现:使用python移除超链接的

C#实现一键批量合并PDF文档

《C#实现一键批量合并PDF文档》这篇文章主要为大家详细介绍了如何使用C#实现一键批量合并PDF文档功能,文中的示例代码简洁易懂,感兴趣的小伙伴可以跟随小编一起学习一下... 目录前言效果展示功能实现1、添加文件2、文件分组(书签)3、定义页码范围4、自定义显示5、定义页面尺寸6、PDF批量合并7、其他方法

PHP应用中处理限流和API节流的最佳实践

《PHP应用中处理限流和API节流的最佳实践》限流和API节流对于确保Web应用程序的可靠性、安全性和可扩展性至关重要,本文将详细介绍PHP应用中处理限流和API节流的最佳实践,下面就来和小编一起学习... 目录限流的重要性在 php 中实施限流的最佳实践使用集中式存储进行状态管理(如 Redis)采用滑动

Java实现在Word文档中添加文本水印和图片水印的操作指南

《Java实现在Word文档中添加文本水印和图片水印的操作指南》在当今数字时代,文档的自动化处理与安全防护变得尤为重要,无论是为了保护版权、推广品牌,还是为了在文档中加入特定的标识,为Word文档添加... 目录引言Spire.Doc for Java:高效Word文档处理的利器代码实战:使用Java为Wo