AvalonEdit:是否可以突出显示此语法

本文关键字:显示 语法 是否 AvalonEdit | 更新日期: 2023-09-27 17:59:17

我有一种简单的"语言"(类似于模板语言和BBcode等简单标记语言——基本上它只是带有一些变量、标记和类似功能的普通文本),我想强调它的语法。

这是我一直纠结的事情。

存在变量,它们用$符号($var1$

<RuleSet name="VariableSet">
    <Rule color="Variable">
        '$'S+?'$
    </Rule>
</RuleSet>

变量周围的某些区域可以用{}字符包围。

换句话说:有些变量可以有它的"区域",它从变量前的第一个{开始,到变量后的第一个

}结束。多个变量不能在一个区域中,因此在像{ $var1$ $var2$ }这样没有任何区域的情况下,{}将被视为普通字符并被忽略。它不像C风格语言中的函数和局部作用域那样是作用域。

这里有一个例子:

[b]lorem ipsum[/b] $var0$ dolor sit amet, consectetur adipisicing elit, sed do 
eiusmod tempor incididunt ut labore et dolore magna aliqua. 
{ // <— not nighlighted
{ // <— highlighted
ut enim ad minim veniam, quis nostrud 
$var1$
exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat.
} // <— highlighted
} // <— not nighlighted
// all brackets below also should be not highlighted
duis aute { $50, 25$} irure { dolor } excepteur $var2$ sint occaecat cupidatat non
proident, sunt in culpa qui mollit anim id { est $var2$ laborum.
{ $var3$ $var4$ }

首先,我尝试使用两个Rule regexp来解决这个问题(对于{和},当然,使用这种方法不可能跳过带有未闭合括号的情况,如{ $var$$var$ },但这不是什么大问题)。然而,我发现Rule只在一行中工作。

然后我尝试了Span,如下所示:

<Span color="VariableAreaDelimiter" multiline="true">
    <Begin>
        '{(?!'{.*?'$'S+?'$)(?=.*?'$'S+?'$)
    </Begin>
    <End>
        '}
    </End>
    <RuleSet>
        <Import ruleSet="VariableSet"/>
        <Rule foreground="Black" fontWeight="normal">
            .
        </Rule>
    </RuleSet>
</Span>

一些问题:

  • 虽然multilinetrue,但BeginEnd中的regexp不适用于多行。所以它与此不匹配:

    {
    $var$
    
  • 若并没有右括号,则跨度会占用所有内容,直到文档结束。这就是我添加.规则的原因。

AvalonEdit:是否可以突出显示此语法

使用AvalonEdit的高亮显示引擎根本不可能做到这一点。

引擎是基于行的,不能执行任何多行匹配。将信息从一行传递到下一行的唯一方法是打开跨度——高亮显示引擎维护的唯一状态是当前打开的跨度的堆栈。

高亮显示引擎是这样设计的,以允许增量更新(这对大文件的性能至关重要)。如果更改一行中的文本,则仅更新该单行。如果此更新导致行末尾的跨度堆栈发生更改,则以下行也会更新(但前提是它们位于文本区域的可见部分,否则它们的更新会延迟到用户向下滚动)。

一个可能的解决方案是实现自己的IVisualLineTransformer,而不是使用语法高亮显示引擎。下面是一个示例实现,它突出显示了单词"AvalonEdit"的所有出现:

public class ColorizeAvalonEdit : DocumentColorizingTransformer
{
    protected override void ColorizeLine(DocumentLine line)
    {
        int lineStartOffset = line.Offset;
        string text = CurrentContext.Document.GetText(line);
        int start = 0;
        int index;
        while ((index = text.IndexOf("AvalonEdit", start, StringComparison.Ordinal)) >= 0) {
            base.ChangeLinePart(
                lineStartOffset + index, // startOffset
                lineStartOffset + index + 10, // endOffset
                (VisualLineElement element) => {
                    // This lambda gets called once for every VisualLineElement
                    // between the specified offsets.
                    Typeface tf = element.TextRunProperties.Typeface;
                    // Replace the typeface with a modified version of
                    // the same typeface
                    element.TextRunProperties.SetTypeface(new Typeface(
                        tf.FontFamily,
                        FontStyles.Italic,
                        FontWeights.Bold,
                        tf.Stretch
                    ));
                });
            start = index + 1; // search for next occurrence
        }
    }
}
// Usage:
textEditor.TextArea.TextView.LineTransformers.Add(new ColorizeAvalonEdit());

尝试将您的开始模式替换为:

'{(?=[^{}$]*'$[^{}$'s]+'$[^{}$]*'})