using System; using System.Collections.Generic; using Avalonia; using Avalonia.Controls; using Avalonia.Controls.Models.TreeDataGrid; using Avalonia.Controls.Templates; using Avalonia.Interactivity; namespace SourceGit.Views { public partial class ChangeCollectionView : UserControl { public static readonly StyledProperty IsWorkingCopyChangeProperty = AvaloniaProperty.Register(nameof(IsWorkingCopyChange), false); public bool IsWorkingCopyChange { get => GetValue(IsWorkingCopyChangeProperty); set => SetValue(IsWorkingCopyChangeProperty, value); } public static readonly StyledProperty SingleSelectProperty = AvaloniaProperty.Register(nameof(SingleSelect), true); public bool SingleSelect { get => GetValue(SingleSelectProperty); set => SetValue(SingleSelectProperty, 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), null); public List Changes { get => GetValue(ChangesProperty); set => SetValue(ChangesProperty, value); } public static readonly StyledProperty> SelectedChangesProperty = AvaloniaProperty.Register>(nameof(SelectedChanges), null); 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); } } static ChangeCollectionView() { ViewModeProperty.Changed.AddClassHandler((c, e) => c.UpdateSource()); ChangesProperty.Changed.AddClassHandler((c, e) => c.UpdateSource()); SelectedChangesProperty.Changed.AddClassHandler((c, e) => c.UpdateSelected()); } public ChangeCollectionView() { InitializeComponent(); } private void UpdateSource() { if (Content is TreeDataGrid tree && tree.Source is IDisposable disposable) disposable.Dispose(); Content = null; var changes = Changes; if (changes == null || changes.Count == 0) return; var viewMode = ViewMode; if (viewMode == Models.ChangeViewMode.Tree) { var filetree = Models.FileTreeNode.Build(changes, true); var template = this.FindResource("TreeModeTemplate") as IDataTemplate; var source = new HierarchicalTreeDataGridSource(filetree) { Columns = { new HierarchicalExpanderColumn( new TemplateColumn(null, template, null, GridLength.Auto), x => x.Children, x => x.Children.Count > 0, x => x.IsExpanded) } }; var selection = new Models.TreeDataGridSelectionModel(source, x => x.Children); selection.SingleSelect = SingleSelect; selection.RowDoubleTapped += (_, e) => RaiseEvent(new RoutedEventArgs(ChangeDoubleTappedEvent)); selection.SelectionChanged += (s, _) => { if (!_isSelecting && s is Models.TreeDataGridSelectionModel model) { var selected = new List(); foreach (var c in model.SelectedItems) CollectChangesInNode(selected, c); TrySetSelected(selected); } }; source.Selection = selection; CreateTreeDataGrid(source); } else if (viewMode == Models.ChangeViewMode.List) { var template = this.FindResource("ListModeTemplate") as IDataTemplate; var source = new FlatTreeDataGridSource(changes) { Columns = { new TemplateColumn(null, template, null, GridLength.Auto) } }; var selection = new Models.TreeDataGridSelectionModel(source, null); selection.SingleSelect = SingleSelect; selection.RowDoubleTapped += (_, e) => RaiseEvent(new RoutedEventArgs(ChangeDoubleTappedEvent)); selection.SelectionChanged += (s, _) => { if (!_isSelecting && s is Models.TreeDataGridSelectionModel model) { var selected = new List(); foreach (var c in model.SelectedItems) selected.Add(c); TrySetSelected(selected); } }; source.Selection = selection; CreateTreeDataGrid(source); } else { var template = this.FindResource("GridModeTemplate") as IDataTemplate; var source = new FlatTreeDataGridSource(changes) { Columns = { new TemplateColumn(null, template, null, GridLength.Auto) }, }; var selection = new Models.TreeDataGridSelectionModel(source, null); selection.SingleSelect = SingleSelect; selection.RowDoubleTapped += (_, e) => RaiseEvent(new RoutedEventArgs(ChangeDoubleTappedEvent)); selection.SelectionChanged += (s, _) => { if (!_isSelecting && s is Models.TreeDataGridSelectionModel model) { var selected = new List(); foreach (var c in model.SelectedItems) selected.Add(c); TrySetSelected(selected); } }; source.Selection = selection; CreateTreeDataGrid(source); } } private void UpdateSelected() { if (_isSelecting || Content == null) return; var tree = Content as TreeDataGrid; if (tree == null) return; _isSelecting = true; var selected = SelectedChanges; if (tree.Source.Selection is Models.TreeDataGridSelectionModel changeSelection) { if (selected == null || selected.Count == 0) changeSelection.Clear(); else changeSelection.Select(selected); } else if (tree.Source.Selection is Models.TreeDataGridSelectionModel treeSelection) { if (selected == null || selected.Count == 0) { treeSelection.Clear(); _isSelecting = false; return; } var set = new HashSet(); foreach (var c in selected) set.Add(c); var nodes = new List(); foreach (var node in tree.Source.Items) CollectSelectedNodeByChange(nodes, node as Models.FileTreeNode, set); if (nodes.Count == 0) treeSelection.Clear(); else treeSelection.Select(nodes); } _isSelecting = false; } private void CreateTreeDataGrid(ITreeDataGridSource source) { Content = new TreeDataGrid() { AutoDragDropRows = false, ShowColumnHeaders = false, CanUserResizeColumns = false, CanUserSortColumns = false, Source = source, }; } private void CollectChangesInNode(List outs, Models.FileTreeNode node) { if (node.IsFolder) { foreach (var child in node.Children) CollectChangesInNode(outs, child); } else { var change = node.Backend as Models.Change; if (change != null && !outs.Contains(change)) outs.Add(change); } } private void CollectSelectedNodeByChange(List outs, Models.FileTreeNode node, HashSet selected) { if (node == null) return; if (node.IsFolder) { foreach (var child in node.Children) CollectSelectedNodeByChange(outs, child, selected); } else if (node.Backend != null && selected.Contains(node.Backend)) { outs.Add(node); } } private void TrySetSelected(List changes) { var old = SelectedChanges; if (old == null && changes.Count == 0) return; if (old != null && old.Count == changes.Count) { bool allEquals = true; foreach (var c in old) { if (!changes.Contains(c)) { allEquals = false; break; } } if (allEquals) return; } _isSelecting = true; SetCurrentValue(SelectedChangesProperty, changes); _isSelecting = false; } private bool _isSelecting = false; } }