AvalonEdit:可以突出显示这种语法吗?

时间:2014-04-20 18:23:20

标签: c# regex syntax-highlighting avalonedit

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

这是我坚持的事情。

有变量,它们附有 $ 符号( $ var1 $ )。我用这个规则强调它们:

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

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

换句话说:某些变量可以有&#34;区域&#34;,它从变量前的第一个 {开始,到第一个}结束在变量之后。多个变量不能位于一个区域中,因此在{ $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正则表达式解决此问题(对于{和},当然使用此方法时,不可能跳过包含{ $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>

一些问题:

  • 虽然multiline中的trueBegin正则表达式,但End { $var$ 不适用于多行。所以它并不匹配:

    .
  • 如果没有结束括号,则跨度将占用所有内容,直到文档结束。这就是我添加{{1}}规则的原因。

2 个答案:

答案 0 :(得分:3)

对于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());

答案 1 :(得分:1)

尝试用以下代码替换您的开始模式:

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