这很困难,因为大写可以改变显示字符和文档之间的映射(视觉列与文档偏移量)。
例如,单个字符“ß”(德语升号 s)仅以小写字母形式存在,并且在调用时会转换为两个字符的字符串“SS”string.ToUpper()
。
编辑此文本很棘手:我们不能允许用户仅替换其中一个“S”,因为底层文档仅包含“ß”。
一个简单的解决方案是使用char.ToUpper()
方法,在原始字符和大写字符之间强制执行一对一映射。这将使“ß”等字母保持不变。
在 AvalonEdit 4.2 中,仅允许对已生成的 VisualLineElements 进行两种转换:
- 更改文本运行属性,例如字体大小、文字颜色等
- 将 VisualLineElement 一分为二 - 这是内部使用的
ChangeLinePart()
以便可以更改文本部分的属性。
这意味着无法在着色器中进行文本替换,您需要使用VisualLineElementGenerator
.
/// <summary>
/// Makes all text after a colon (until the end of line) upper-case.
/// </summary>
public class UppercaseGenerator : VisualLineElementGenerator
{
public override int GetFirstInterestedOffset(int startOffset)
{
TextDocument document = CurrentContext.Document;
int endOffset = CurrentContext.VisualLine.LastDocumentLine.EndOffset;
for (int i = startOffset; i < endOffset; i++) {
char c = document.GetCharAt(i);
if (c == ':')
return i + 1;
}
return -1;
}
public override VisualLineElement ConstructElement(int offset)
{
DocumentLine line = CurrentContext.Document.GetLineByOffset(offset);
return new UppercaseText(CurrentContext.VisualLine, line.EndOffset - offset);
}
/// <summary>
/// Displays a portion of the document text, but upper-cased.
/// </summary>
class UppercaseText : VisualLineText
{
public UppercaseText(VisualLine parentVisualLine, int length) : base(parentVisualLine, length)
{
}
protected override VisualLineText CreateInstance(int length)
{
return new UppercaseText(ParentVisualLine, length);
}
public override TextRun CreateTextRun(int startVisualColumn, ITextRunConstructionContext context)
{
if (context == null)
throw new ArgumentNullException("context");
int relativeOffset = startVisualColumn - VisualColumn;
StringSegment text = context.GetText(context.VisualLine.FirstDocumentLine.Offset + RelativeTextOffset + relativeOffset, DocumentLength - relativeOffset);
char[] uppercase = new char[text.Count];
for (int i = 0; i < text.Count; i++) {
uppercase[i] = char.ToUpper(text.Text[text.Offset + i]);
}
return new TextCharacters(uppercase, 0, uppercase.Length, this.TextRunProperties);
}
}
}
在 AvalonEdit 4.3.0.8868 中,我添加了该方法VisualLine.ReplaceElement()
。这可以用来替换默认的VisualText
元素与UppercaseText
线路变压器(着色器)内的元素。
请注意,还可以实现对“ß”显示为“SS”的支持。为此,您必须实现自己的副本VisualLineText
而不是仅仅覆盖现有的。然后您可以使用与文档长度不同的视觉长度。这GetRelativeOffset
and GetVisualColumns
方法将用于提供文档和视觉坐标之间的映射。
您还可以使用另一种选择:小型大写字母。
// in the colorizer:
ChangeLinePart(start, end, e => e.TextRunProperties.SetTypographyProperties(new CapsTypography()));
// helper class
class CapsTypography : DefaultTextRunTypographyProperties
{
public override FontCapitals Capitals {
get { return FontCapitals.SmallCaps; }
}
}
但是,仅当使用支持小型大写字母的 OpenType 字体时,WPF 才会呈现小型大写字母。在我的测试中,Cambria
可以使用小型大写字母,而大多数其他字体则不能。
另外,SetTypographyProperties
方法和DefaultTextRunTypographyProperties
类需要 AvalonEdit 4.3。