c# - VSTO 字 2016 : Squiggly underline without affecting undo

标签 c# .net ms-word vsto

我正在开发一个实时语言分析工具,它需要在 Word 2016 中使用 VSTO 加载项突出显示单词以引起作者的注意,该加载项使用 C# 以 .NET4.6.1 编写。想想语法/拼写检查,它会在单词下方添加一条波浪线,告诉您该单词有语法或拼写错误。我正在为我自己定义的一些规则添加类似的功能。

我四处搜索添加波浪线,偶然发现了 Font.UnderlineFont.UnderlineColor。我将其设置在一个词的范围内,它似乎提供了我想要引起注意的视觉刺激。但是有一个问题。我添加的每个下划线或更改的下划线颜色都会向撤消堆栈添加一个撤消操作。

我不希望这种情况发生,或者我想要一种方法来从堆栈中弹出我刚刚在代码中执行的操作。目的是让用户能够使用 CTRL+Z 删除他更改的文本,而不影响我的语言分析结果。

我该怎么做?

最佳答案

我偶然发现了这个问题,它问的是完全相同的事情: Prevent actions to be added to the word undo redo OR Remove actions from the undo redo CommandBarComboBox

正如@Manu 所指出的,在 MSDN 上也有人提出了同样的问题。答案是:

The UndoClear method will empty the list. That's the best you can do.

这里也有类似的问题 Word VSTO override CTRL+Z / CTRL+Y 这建议 Microsoft.Office.Interop.Word.UndoRecord 路由或 AddIns 中的 MessageHooks。


经过更多研究后,我注意到此线程末尾的一个好主意:MSDN Draw my own squigglies on Word document您可以在其中跟踪您的操作,然后在撤消和重做操作中跳过它们。

这是执行“事务撤消/重做”的代码的绝佳示例 Can I create an undo transaction in Word or Excel? (VSTO) .除了一个大问题外,您可以在 VSTO 中执行相同的方法,正如 Dirk Vollmar 在他的回答中指出的那样:

I don't think that overwriting built-in Word commands is possible using VSTO alone, though

我已经使用键盘 Hook 事件覆盖了 VSTO 中的一些内置命令来拦截命令:How to perform .Onkey Event in an Excel Add-In created with Visual Studio 2010? 但是我不确定您是否可以重新创建功能区来拦截按钮命令。更具体地说,Undo 和 Redo 是 Ribbon UI 中的内置库,您无法使用内置的 Ribbon 库执行任何操作。在 2010 等版本中,撤消/重做按钮位于标题栏中 - 您不能 add/edit buttons on the title bar using VSTO :

enter image description here

因此,如果您担心捕获按钮命令(我认识的每个人都使用 Ctrl+Z & Y),您可以注入(inject) VBA 代码以访问 EditUndoEditRedo 事件,例如:

VB._VBComponent vbModule = VBProj.VBE.ActiveVBProject.VBComponents.Add(VB.vbext_ComponentType.vbext_ct_StdModule);
String functionText = "Public Sub EditUndo() \n";
functionText += "MsgBox \"Undo Happened\"\n";
functionText += "End Sub";
vbModule.CodeModule.AddFromString(functionText);

这种方法的主要问题是需要授予信任。


同一个 QA 中的另一个答案 Can I create an undo transaction in Word or Excel? (VSTO)由迈克·里根 (Mike Regan) 在德克 (Dirk) 三个月后回答。他使用了一个隐藏文档,并在需要时将其放置在真实文档中,以便将任意数量的 VSTO 操作一次撤消。

仍然没有解决阻止操作被记录在撤消历史记录中的问题。


我确实尝试过使用注册表项将 UndoHistory 限制为 0 并将其重置回 100(以便在添加操作时禁用历史记录),但它似乎仅适用于 Excel https://support.microsoft.com/en-gb/kb/211922

可能有一个未记录的注册键可以完全禁用撤消/重做历史记录,但它只会在启动时由 Word 读取。我认为包含数字的 UndoHistory 键会在每次撤消/重做之前被读取,但这种方法完全没有运气。


这不是一个容易解决的问题,有很大的局限性,所以可能更容易:

a) 接受您的拼写/语法检查器加载项包含在撤消/重做列表中(失败)。

b) 确定行/文本在屏幕上的位置,并显示突出问题的透明工具提示。这比看起来要难得多,也不理想,这里有两个很好的答案可以指导您使用此方法:Detecting text changes in Word 2016 from VSTO add-in或更简单的方法来检测此 Microsoft 电子邮件线程中的 XY 位置:https://groups.google.com/forum/#!topic/microsoft.public.word.vba.general/pKq4PsqD3cM

//position dialog relative to word insertion point (caret)
int left = 0;
int top = 0;
int width = 0;
int height = 0;
MSWord.Range r = Globals.ThisDocument.Application.Selection.Range;
MSWord.Window w = Globals.ThisDocument.ActiveWindow;

w.GetPoint(out left, out top, out width, out height, r);

frmPopUp newForm = new frmPopUp();
newForm.SetDesktopLocation( left + width + 2, top - newForm.Height + height );

c) 只通过事务性撤消/重做键盘捕获撤消/重做事件,并让用户使用按钮查看原子撤消/重做。删除撤消/重做按钮是非常不可靠的,这会导致大量我的撤消按钮不见了?支持案例。所以不要这样做,如果你很好奇,可以从“.qat”文件中完成快速访问工具栏自定义。引用:https://support.microsoft.com/en-us/kb/926805

d) 使用 Mouse Hook 并检测何时单击撤消/重做按钮。这是连接鼠标的代码,请注意这不会很好地与公司的反病毒产品一起使用,我不推荐它:https://blogs.msdn.microsoft.com/andreww/2009/02/24/message-hooks-in-add-ins/https://github.com/gmamaladze/globalmousekeyhook .

e) 尝试拦截原始窗口消息,例如 Excel CustomTaskPane with WebBrowser control - keyboard/focus issues根据我引用 BUG: Cant choose dates on a DatePicker that fall outside a floating VSTO Add-In 的评论,只是小心 Office 中的消息泵有时会表现出奇怪的行为。

关于c# - VSTO 字 2016 : Squiggly underline without affecting undo,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/40829373/

相关文章:

c# - 并行性和 Entity Framework

c# - 将 nhibernate 与 Unity3D 集成

c# - 有没有办法检查一个字符串是否不等于多个不同的字符串?

c# - Word 如何在高级搜索中找到匹配的词形?

ms-word - 如何禁用Word中的拼写检查弹出窗口?

VBA - 获取指定日期的前一天

c# - 使用 TextBox 或 RichTextBox 显示图像文件中的原始数据?

c# - 代码已编译并成功运行,但预期的输出是打印 "Sub"未打印。这段代码有什么错误?

c# - 似乎无法摆脱兼容性 View ?

.net - 依赖性检查