using System; using System.Collections.Generic; using Avalonia; using Avalonia.Controls; using Avalonia.Controls.Primitives; using Avalonia.Input; using Avalonia.Interactivity; using Avalonia.VisualTree; namespace SourceGit.Views { public class ChangeTreeNodeToggleButton : ToggleButton { protected override Type StyleKeyOverride => typeof(ToggleButton); protected override void OnPointerPressed(PointerPressedEventArgs e) { if (e.GetCurrentPoint(this).Properties.IsLeftButtonPressed && DataContext is ViewModels.ChangeTreeNode { IsFolder: true } node) { var tree = this.FindAncestorOfType(); tree?.ToggleNodeIsExpanded(node); } e.Handled = true; } } public class ChangeCollectionContainer : ListBox { protected override Type StyleKeyOverride => typeof(ListBox); protected override void OnKeyDown(KeyEventArgs e) { if (SelectedItems is [ViewModels.ChangeTreeNode { IsFolder: true } node] && e.KeyModifiers == KeyModifiers.None) { if ((node.IsExpanded && e.Key == Key.Left) || (!node.IsExpanded && e.Key == Key.Right)) { this.FindAncestorOfType()?.ToggleNodeIsExpanded(node); e.Handled = true; } } if (!e.Handled && e.Key != Key.Space) base.OnKeyDown(e); } } public partial class ChangeCollectionView : UserControl { public static readonly StyledProperty IsUnstagedChangeProperty = AvaloniaProperty.Register(nameof(IsUnstagedChange)); public bool IsUnstagedChange { get => GetValue(IsUnstagedChangeProperty); set => SetValue(IsUnstagedChangeProperty, value); } public static readonly StyledProperty SelectionModeProperty = AvaloniaProperty.Register(nameof(SelectionMode)); public SelectionMode SelectionMode { get => GetValue(SelectionModeProperty); set => SetValue(SelectionModeProperty, value); } public static readonly StyledProperty ViewModeProperty = AvaloniaProperty.Register(nameof(ViewMode), Models.ChangeViewMode.Tree); public Models.ChangeViewMode ViewMode { get => GetValue(ViewModeProperty); set => SetValue(ViewModeProperty, value); } public static readonly StyledProperty> ChangesProperty = AvaloniaProperty.Register>(nameof(Changes)); public List Changes { get => GetValue(ChangesProperty); set => SetValue(ChangesProperty, value); } public static readonly StyledProperty> SelectedChangesProperty = AvaloniaProperty.Register>(nameof(SelectedChanges)); public List SelectedChanges { get => GetValue(SelectedChangesProperty); set => SetValue(SelectedChangesProperty, value); } public static readonly RoutedEvent ChangeDoubleTappedEvent = RoutedEvent.Register(nameof(ChangeDoubleTapped), RoutingStrategies.Tunnel | RoutingStrategies.Bubble); public event EventHandler ChangeDoubleTapped { add { AddHandler(ChangeDoubleTappedEvent, value); } remove { RemoveHandler(ChangeDoubleTappedEvent, value); } } public ChangeCollectionView() { InitializeComponent(); } public void ToggleNodeIsExpanded(ViewModels.ChangeTreeNode node) { if (Content is ViewModels.ChangeCollectionAsTree tree) { node.IsExpanded = !node.IsExpanded; var depth = node.Depth; var idx = tree.Rows.IndexOf(node); if (idx == -1) return; if (node.IsExpanded) { var subrows = new List(); MakeTreeRows(subrows, node.Children); tree.Rows.InsertRange(idx + 1, subrows); } else { var removeCount = 0; for (int i = idx + 1; i < tree.Rows.Count; i++) { var row = tree.Rows[i]; if (row.Depth <= depth) break; removeCount++; } tree.Rows.RemoveRange(idx + 1, removeCount); } } } public Models.Change GetNextChangeWithoutSelection() { var selected = SelectedChanges; var changes = Changes; if (selected == null || selected.Count == 0) return changes.Count > 0 ? changes[0] : null; if (selected.Count == changes.Count) return null; var set = new HashSet(); foreach (var c in selected) set.Add(c.Path); if (Content is ViewModels.ChangeCollectionAsTree tree) { var lastUnselected = -1; for (int i = tree.Rows.Count - 1; i >= 0; i--) { var row = tree.Rows[i]; if (!row.IsFolder) { if (set.Contains(row.FullPath)) { if (lastUnselected == -1) continue; break; } lastUnselected = i; } } if (lastUnselected != -1) return tree.Rows[lastUnselected].Change; } else { var lastUnselected = -1; for (int i = changes.Count - 1; i >= 0; i--) { if (set.Contains(changes[i].Path)) { if (lastUnselected == -1) continue; break; } lastUnselected = i; } if (lastUnselected != -1) return changes[lastUnselected]; } return null; } protected override void OnPropertyChanged(AvaloniaPropertyChangedEventArgs change) { base.OnPropertyChanged(change); if (change.Property == ViewModeProperty) UpdateDataSource(false); else if (change.Property == ChangesProperty) UpdateDataSource(true); else if (change.Property == SelectedChangesProperty) UpdateSelection(); } private void OnRowDoubleTapped(object sender, TappedEventArgs e) { var grid = sender as Grid; if (grid?.DataContext is ViewModels.ChangeTreeNode node) { if (node.IsFolder) { var posX = e.GetPosition(this).X; if (posX < node.Depth * 16 + 16) return; ToggleNodeIsExpanded(node); } else { RaiseEvent(new RoutedEventArgs(ChangeDoubleTappedEvent)); } } else if (grid?.DataContext is Models.Change) { RaiseEvent(new RoutedEventArgs(ChangeDoubleTappedEvent)); } } private void OnRowSelectionChanged(object sender, SelectionChangedEventArgs _) { if (_disableSelectionChangingEvent) return; _disableSelectionChangingEvent = true; var selected = new List(); if (sender is ListBox { SelectedItems: { } selectedItems }) { foreach (var item in selectedItems) { if (item is Models.Change c) selected.Add(c); else if (item is ViewModels.ChangeTreeNode node) CollectChangesInNode(selected, node); } } var old = SelectedChanges ?? []; if (old.Count != selected.Count) { SetCurrentValue(SelectedChangesProperty, selected); } else { bool allEquals = true; foreach (var c in old) { if (!selected.Contains(c)) { allEquals = false; break; } } if (!allEquals) SetCurrentValue(SelectedChangesProperty, selected); } _disableSelectionChangingEvent = false; } private void MakeTreeRows(List rows, List nodes) { foreach (var node in nodes) { rows.Add(node); if (!node.IsExpanded || !node.IsFolder) continue; MakeTreeRows(rows, node.Children); } } private void UpdateDataSource(bool disableEvents) { _disableSelectionChangingEvent = disableEvents; var changes = Changes; if (changes == null || changes.Count == 0) { Content = null; _disableSelectionChangingEvent = false; return; } var selected = SelectedChanges ?? []; if (ViewMode == Models.ChangeViewMode.Tree) { HashSet oldFolded = new HashSet(); if (Content is ViewModels.ChangeCollectionAsTree oldTree) { foreach (var row in oldTree.Rows) { if (row.IsFolder && !row.IsExpanded) oldFolded.Add(row.FullPath); } } var tree = new ViewModels.ChangeCollectionAsTree(); tree.Tree = ViewModels.ChangeTreeNode.Build(changes, oldFolded); var rows = new List(); MakeTreeRows(rows, tree.Tree); tree.Rows.AddRange(rows); if (selected.Count > 0) { var sets = new HashSet(); foreach (var c in selected) sets.Add(c); var nodes = new List(); foreach (var row in tree.Rows) { if (row.Change != null && sets.Contains(row.Change)) nodes.Add(row); } tree.SelectedRows.AddRange(nodes); } Content = tree; } else if (ViewMode == Models.ChangeViewMode.Grid) { var grid = new ViewModels.ChangeCollectionAsGrid(); grid.Changes.AddRange(changes); if (selected.Count > 0) grid.SelectedChanges.AddRange(selected); Content = grid; } else { var list = new ViewModels.ChangeCollectionAsList(); list.Changes.AddRange(changes); if (selected.Count > 0) list.SelectedChanges.AddRange(selected); Content = list; } _disableSelectionChangingEvent = false; } private void UpdateSelection() { if (_disableSelectionChangingEvent) return; _disableSelectionChangingEvent = true; var selected = SelectedChanges ?? []; if (Content is ViewModels.ChangeCollectionAsTree tree) { tree.SelectedRows.Clear(); if (selected.Count > 0) { var sets = new HashSet(); foreach (var c in selected) sets.Add(c); var nodes = new List(); foreach (var row in tree.Rows) { if (row.Change != null && sets.Contains(row.Change)) nodes.Add(row); } tree.SelectedRows.AddRange(nodes); } } else if (Content is ViewModels.ChangeCollectionAsGrid grid) { grid.SelectedChanges.Clear(); if (selected.Count > 0) grid.SelectedChanges.AddRange(selected); } else if (Content is ViewModels.ChangeCollectionAsList list) { list.SelectedChanges.Clear(); if (selected.Count > 0) list.SelectedChanges.AddRange(selected); } _disableSelectionChangingEvent = false; } private void CollectChangesInNode(List outs, ViewModels.ChangeTreeNode node) { if (node.IsFolder) { foreach (var child in node.Children) CollectChangesInNode(outs, child); } else if (!outs.Contains(node.Change)) { outs.Add(node.Change); } } private bool _disableSelectionChangingEvent = false; } }