using System; using System.Collections.Generic; using System.Text.RegularExpressions; using Avalonia; using Avalonia.Collections; using Avalonia.Controls; using Avalonia.Controls.Documents; using Avalonia.Input; using Avalonia.VisualTree; namespace SourceGit.Views { public partial class CommitMessagePresenter : SelectableTextBlock { [GeneratedRegex(@"\b([0-9a-fA-F]{8,40})\b")] private static partial Regex REG_SHA_FORMAT(); public static readonly StyledProperty MessageProperty = AvaloniaProperty.Register(nameof(Message)); public string Message { get => GetValue(MessageProperty); set => SetValue(MessageProperty, value); } public static readonly StyledProperty> IssueTrackerRulesProperty = AvaloniaProperty.Register>(nameof(IssueTrackerRules)); public AvaloniaList IssueTrackerRules { get => GetValue(IssueTrackerRulesProperty); set => SetValue(IssueTrackerRulesProperty, value); } protected override Type StyleKeyOverride => typeof(SelectableTextBlock); protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) { base.OnPropertyChanged(change); if (change.Property == MessageProperty || change.Property == IssueTrackerRulesProperty) { Inlines!.Clear(); _matches = null; ClearHoveredIssueLink(); var message = Message; if (string.IsNullOrEmpty(message)) return; var matches = new List(); if (IssueTrackerRules is { Count: > 0 } rules) { foreach (var rule in rules) rule.Matches(matches, message); } var shas = REG_SHA_FORMAT().Matches(message); for (int i = 0; i < shas.Count; i++) { var sha = shas[i]; if (!sha.Success) continue; var start = sha.Index; var len = sha.Length; var intersect = false; foreach (var match in matches) { if (match.Intersect(start, len)) { intersect = true; break; } } if (!intersect) matches.Add(new Models.Hyperlink(start, len, sha.Groups[1].Value, true)); } if (matches.Count == 0) { Inlines.Add(new Run(message)); return; } matches.Sort((l, r) => l.Start - r.Start); _matches = matches; int pos = 0; foreach (var match in matches) { if (match.Start > pos) Inlines.Add(new Run(message.Substring(pos, match.Start - pos))); var link = new Run(message.Substring(match.Start, match.Length)); link.Classes.Add(match.IsCommitSHA ? "commit_link" : "issue_link"); Inlines.Add(link); pos = match.Start + match.Length; } if (pos < message.Length) Inlines.Add(new Run(message.Substring(pos))); } } protected override void OnPointerMoved(PointerEventArgs e) { base.OnPointerMoved(e); if (e.Pointer.Captured == null && _matches != null) { var point = e.GetPosition(this) - new Point(Padding.Left, Padding.Top); var x = Math.Min(Math.Max(point.X, 0), Math.Max(TextLayout.WidthIncludingTrailingWhitespace, 0)); var y = Math.Min(Math.Max(point.Y, 0), Math.Max(TextLayout.Height, 0)); point = new Point(x, y); var pos = TextLayout.HitTestPoint(point).TextPosition; foreach (var match in _matches) { if (!match.Intersect(pos, 1)) continue; if (match == _lastHover) return; SetCurrentValue(CursorProperty, Cursor.Parse("Hand")); _lastHover = match; if (!_lastHover.IsCommitSHA) { ToolTip.SetTip(this, match.Link); ToolTip.SetIsOpen(this, true); } return; } ClearHoveredIssueLink(); } } protected override void OnPointerPressed(PointerPressedEventArgs e) { if (_lastHover != null) { e.Pointer.Capture(null); if (_lastHover.IsCommitSHA) { var parentView = this.FindAncestorOfType(); if (parentView is { DataContext: ViewModels.CommitDetail detail }) detail.NavigateTo(_lastHover.Link); } else { Native.OS.OpenBrowser(_lastHover.Link); } e.Handled = true; return; } base.OnPointerPressed(e); } protected override void OnPointerExited(PointerEventArgs e) { base.OnPointerExited(e); ClearHoveredIssueLink(); } private void ClearHoveredIssueLink() { if (_lastHover != null) { ToolTip.SetTip(this, null); SetCurrentValue(CursorProperty, Cursor.Parse("IBeam")); _lastHover = null; } } private List _matches = null; private Models.Hyperlink _lastHover = null; } }