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

相关文章

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

使用Python实现Word文档的自动化对比方案

《使用Python实现Word文档的自动化对比方案》我们经常需要比较两个Word文档的版本差异,无论是合同修订、论文修改还是代码文档更新,人工比对不仅效率低下,还容易遗漏关键改动,下面通过一个实际案例... 目录引言一、使用python-docx库解析文档结构二、使用difflib进行差异比对三、高级对比方

深入浅出Spring中的@Autowired自动注入的工作原理及实践应用

《深入浅出Spring中的@Autowired自动注入的工作原理及实践应用》在Spring框架的学习旅程中,@Autowired无疑是一个高频出现却又让初学者头疼的注解,它看似简单,却蕴含着Sprin... 目录深入浅出Spring中的@Autowired:自动注入的奥秘什么是依赖注入?@Autowired

Python自动化处理PDF文档的操作完整指南

《Python自动化处理PDF文档的操作完整指南》在办公自动化中,PDF文档处理是一项常见需求,本文将介绍如何使用Python实现PDF文档的自动化处理,感兴趣的小伙伴可以跟随小编一起学习一下... 目录使用pymupdf读写PDF文件基本概念安装pymupdf提取文本内容提取图像添加水印使用pdfplum

Python从Word文档中提取图片并生成PPT的操作代码

《Python从Word文档中提取图片并生成PPT的操作代码》在日常办公场景中,我们经常需要从Word文档中提取图片,并将这些图片整理到PowerPoint幻灯片中,手动完成这一任务既耗时又容易出错,... 目录引言背景与需求解决方案概述代码解析代码核心逻辑说明总结引言在日常办公场景中,我们经常需要从 W

PostgreSQL简介及实战应用

《PostgreSQL简介及实战应用》PostgreSQL是一种功能强大的开源关系型数据库管理系统,以其稳定性、高性能、扩展性和复杂查询能力在众多项目中得到广泛应用,本文将从基础概念讲起,逐步深入到高... 目录前言1. PostgreSQL基础1.1 PostgreSQL简介1.2 基础语法1.3 数据库

C#高效实现Word文档内容查找与替换的6种方法

《C#高效实现Word文档内容查找与替换的6种方法》在日常文档处理工作中,尤其是面对大型Word文档时,手动查找、替换文本往往既耗时又容易出错,本文整理了C#查找与替换Word内容的6种方法,大家可以... 目录环境准备方法一:查找文本并替换为新文本方法二:使用正则表达式查找并替换文本方法三:将文本替换为图

Python中的filter() 函数的工作原理及应用技巧

《Python中的filter()函数的工作原理及应用技巧》Python的filter()函数用于筛选序列元素,返回迭代器,适合函数式编程,相比列表推导式,内存更优,尤其适用于大数据集,结合lamb... 目录前言一、基本概念基本语法二、使用方式1. 使用 lambda 函数2. 使用普通函数3. 使用 N