本质上,我的问题是,如果用户编辑了内容,当我尝试以编程方式将 2 个以上的链接应用于 ITextDocument 时,我会收到 AccessViolationException。我基于 Windows Phone (8.1) 空白应用程序模板整理了一个简单的演示应用程序。
我添加到主页:
<StackPanel Margin="19,0,0,0">
<Button
Content="Apply Links"
Click="Button_Click"
/>
<RichEditBox
x:Name="RtfBox"
Height="300"
Loaded="RtfBox_Loaded"
Margin="0,0,19,0"
TextWrapping="Wrap"
/>
</StackPanel>
对于我添加的同一页面的代码隐藏(不包括使用语句):
private void RtfBox_Loaded(object sender, RoutedEventArgs e)
{
//RtfBox.Document.SetText(TextSetOptions.None, "Links to demo, example, test. More links to demo, demo, example, test and test.");
}
private void Button_Click(object sender, RoutedEventArgs e)
{
var pages = new Dictionary<Guid, string> { { Guid.NewGuid(), "demo" }, { Guid.NewGuid(), "example" }, { Guid.NewGuid(), "test" } };
// NOTE: Avoid performance implications of many small updates
RtfBox.Document.BatchDisplayUpdates();
ITextRange range;
foreach (var page in pages)
{
var link = string.Format(""richtea.demo://pages/{0}"", page.Key);
var skip = 0;
while ((range = RtfBox.Document.GetRange(skip, TextConstants.MaxUnitCount)).FindText(page.Value, TextConstants.MaxUnitCount, FindOptions.None) != 0)
{
if (range.Link == "")
{
// TODO: Stop this throw exceptions
System.Diagnostics.Debug.WriteLine("Setting text at position {0} to link: '{1}'.", range.StartPosition, link);
range.Link = link;
}
skip = range.EndPosition;
}
}
RtfBox.Document.ApplyDisplayUpdates();
}
如果您启动并键入类似"演示页面的链接"并单击按钮,它将正确变为链接。您可以继续放置相同的文本并单击按钮,它将继续工作。
但是,如果您输入三个或更多(出于某种原因对我来说总是 3 个或更多)单词演示、示例或测试(我的关键字)并点击按钮,它会在设置range.Link = link
AccessViolationException
出错。值得注意的是,如果您在调试时检查范围。链接属性实际上已设置。
更有趣的是,如果您取消注释RtfBox_Loaded
的内容,并运行该应用程序并立即单击该按钮,它会很好地处理它。所以它似乎与在 RichEditBox 上设置的选择有关?我尝试在应用链接之前禁用控件,但这对我没有帮助。
使我更难诊断此处问题的其他一些事情包括:
- 如果我逐行调试,它似乎更频繁地工作,所以也可能与时间有关
- 我似乎不能在 UI 线程上使用 ITextDocument(COM 对象无法强制转换),所以虽然看起来异步可能是一种更好的方法,但我在这里还没有成功。
同样为了记录,我尝试在 mass 上进行所有更新而不是在用户键入它们时进行更新的原因是我不想在重命名或删除笔记时处理清理,而且我真的不希望这些链接在编辑时或保存,但我可以接受后者。
这个解决方案由 Eric Fleck 发布在 MSDN 论坛上,对我有用:
RtfBox.Document.Selection.StartPosition = RtfBox.Document.Selection.EndPosition = range.StartPosition;
range.Link = link;
RtfBox.Document.Selection.StartPosition = RtfBox.Document.Selection.EndPosition = range.EndPosition;
似乎围绕设置的每个链接进行操作很重要,因为除非我弄错了,否则我在更新所有链接之前尝试过这个,但没有帮助。
我还没有使用将选择恢复到其原始位置的功能,但我将来可能想要,所以我做了这个小实用程序类。另外,这样我就可以将这些地方包装在 using() 块中以获得一些语法糖。
public static class ITextDocumentExtensions
{
public static IDisposable SuppressSelection(this ITextDocument document)
{
var start = document.Selection.StartPosition;
var end = document.Selection.EndPosition;
var disposable = new ActionDisposable(() => document.Selection.SetRange(start, end));
document.Selection.SetRange(0, 0);
return disposable;
}
private sealed class ActionDisposable : IDisposable
{
private readonly Action dispose;
public ActionDisposable(Action dispose)
{
this.dispose = dispose;
}
public void Dispose()
{
dispose();
}
}
}
这允许我写
using (RtfBox.Document.SuppressSelection())
{
range.Link = link;
}