sourcegit/src/Views/CommandLogContentPresenter.cs
2025-04-17 18:21:55 +08:00

152 lines
5 KiB
C#

using System;
using System.Collections.Generic;
using Avalonia;
using Avalonia.Controls.Primitives;
using Avalonia.Interactivity;
using Avalonia.Media;
using AvaloniaEdit;
using AvaloniaEdit.Document;
using AvaloniaEdit.Editing;
using AvaloniaEdit.Rendering;
using AvaloniaEdit.TextMate;
namespace SourceGit.Views
{
public class CommandLogContentPresenter : TextEditor
{
public class LineStyleTransformer : DocumentColorizingTransformer
{
protected override void ColorizeLine(DocumentLine line)
{
var content = CurrentContext.Document.GetText(line);
if (content.StartsWith("$ git ", StringComparison.Ordinal))
{
ChangeLinePart(line.Offset, line.Offset + 1, v =>
{
v.TextRunProperties.SetForegroundBrush(Brushes.Orange);
});
ChangeLinePart(line.Offset + 2, line.EndOffset, v =>
{
var old = v.TextRunProperties.Typeface;
v.TextRunProperties.SetTypeface(new Typeface(old.FontFamily, old.Style, FontWeight.Bold));
});
}
else if (content.StartsWith("remote: ", StringComparison.Ordinal))
{
ChangeLinePart(line.Offset, line.Offset + 7, v =>
{
v.TextRunProperties.SetForegroundBrush(Brushes.SeaGreen);
});
}
else
{
foreach (var err in _errors)
{
var idx = content.IndexOf(err, StringComparison.Ordinal);
if (idx >= 0)
{
ChangeLinePart(line.Offset + idx, line.Offset + err.Length + 1, v =>
{
v.TextRunProperties.SetForegroundBrush(Brushes.Red);
});
}
}
}
}
private readonly List<string> _errors = ["! [rejected]", "! [remote rejected]"];
}
public static readonly StyledProperty<ViewModels.CommandLog> LogProperty =
AvaloniaProperty.Register<CommandLogContentPresenter, ViewModels.CommandLog>(nameof(Log));
public ViewModels.CommandLog Log
{
get => GetValue(LogProperty);
set => SetValue(LogProperty, value);
}
public static readonly StyledProperty<string> PureTextProperty =
AvaloniaProperty.Register<CommandLogContentPresenter, string>(nameof(PureText));
public string PureText
{
get => GetValue(PureTextProperty);
set => SetValue(PureTextProperty, value);
}
protected override Type StyleKeyOverride => typeof(TextEditor);
public CommandLogContentPresenter() : base(new TextArea(), new TextDocument())
{
IsReadOnly = true;
ShowLineNumbers = false;
WordWrap = false;
HorizontalScrollBarVisibility = ScrollBarVisibility.Auto;
VerticalScrollBarVisibility = ScrollBarVisibility.Auto;
TextArea.TextView.Margin = new Thickness(4, 0);
TextArea.TextView.Options.EnableHyperlinks = false;
TextArea.TextView.Options.EnableEmailHyperlinks = false;
}
protected override void OnLoaded(RoutedEventArgs e)
{
base.OnLoaded(e);
if (_textMate == null)
{
_textMate = Models.TextMateHelper.CreateForEditor(this);
Models.TextMateHelper.SetGrammarByFileName(_textMate, "Log.log");
TextArea.TextView.LineTransformers.Add(new LineStyleTransformer());
}
}
protected override void OnUnloaded(RoutedEventArgs e)
{
base.OnUnloaded(e);
if (_textMate != null)
{
_textMate.Dispose();
_textMate = null;
}
GC.Collect();
}
protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change)
{
base.OnPropertyChanged(change);
if (change.Property == LogProperty)
{
if (change.NewValue is ViewModels.CommandLog log)
{
Text = log.Content;
log.Register(OnLogLineReceived);
}
else
{
Text = string.Empty;
}
}
else if (change.Property == PureTextProperty)
{
if (!string.IsNullOrEmpty(PureText))
Text = PureText;
}
}
private void OnLogLineReceived(string newline)
{
AppendText("\n");
AppendText(newline);
}
private TextMate.Installation _textMate = null;
}
}