mirror of
https://github.com/sourcegit-scm/sourcegit.git
synced 2025-01-02 22:27:21 -08:00
389 lines
14 KiB
C#
389 lines
14 KiB
C#
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 class ChangeTreeNode
|
|
{
|
|
public string FullPath { get; set; } = string.Empty;
|
|
public bool IsFolder { get; set; } = false;
|
|
public bool IsExpanded { get; set; } = false;
|
|
public Models.Change Change { get; set; } = null;
|
|
public List<ChangeTreeNode> Children { get; set; } = new List<ChangeTreeNode>();
|
|
|
|
public static List<ChangeTreeNode> Build(IList<Models.Change> changes, bool expanded)
|
|
{
|
|
var nodes = new List<ChangeTreeNode>();
|
|
var folders = new Dictionary<string, ChangeTreeNode>();
|
|
|
|
foreach (var c in changes)
|
|
{
|
|
var sepIdx = c.Path.IndexOf('/', StringComparison.Ordinal);
|
|
if (sepIdx == -1)
|
|
{
|
|
nodes.Add(new ChangeTreeNode()
|
|
{
|
|
FullPath = c.Path,
|
|
Change = c,
|
|
IsFolder = false,
|
|
IsExpanded = false
|
|
});
|
|
}
|
|
else
|
|
{
|
|
ChangeTreeNode lastFolder = null;
|
|
var start = 0;
|
|
|
|
while (sepIdx != -1)
|
|
{
|
|
var folder = c.Path.Substring(0, sepIdx);
|
|
if (folders.TryGetValue(folder, out var value))
|
|
{
|
|
lastFolder = value;
|
|
}
|
|
else if (lastFolder == null)
|
|
{
|
|
lastFolder = new ChangeTreeNode()
|
|
{
|
|
FullPath = folder,
|
|
IsFolder = true,
|
|
IsExpanded = expanded
|
|
};
|
|
folders.Add(folder, lastFolder);
|
|
InsertFolder(nodes, lastFolder);
|
|
}
|
|
else
|
|
{
|
|
var cur = new ChangeTreeNode()
|
|
{
|
|
FullPath = folder,
|
|
IsFolder = true,
|
|
IsExpanded = expanded
|
|
};
|
|
folders.Add(folder, cur);
|
|
InsertFolder(lastFolder.Children, cur);
|
|
lastFolder = cur;
|
|
}
|
|
|
|
start = sepIdx + 1;
|
|
sepIdx = c.Path.IndexOf('/', start);
|
|
}
|
|
|
|
lastFolder.Children.Add(new ChangeTreeNode()
|
|
{
|
|
FullPath = c.Path,
|
|
Change = c,
|
|
IsFolder = false,
|
|
IsExpanded = false
|
|
});
|
|
}
|
|
}
|
|
|
|
folders.Clear();
|
|
return nodes;
|
|
}
|
|
|
|
private static void InsertFolder(List<ChangeTreeNode> collection, ChangeTreeNode subFolder)
|
|
{
|
|
for (int i = 0; i < collection.Count; i++)
|
|
{
|
|
if (!collection[i].IsFolder)
|
|
{
|
|
collection.Insert(i, subFolder);
|
|
return;
|
|
}
|
|
}
|
|
|
|
collection.Add(subFolder);
|
|
}
|
|
}
|
|
|
|
public partial class ChangeCollectionView : UserControl
|
|
{
|
|
public static readonly StyledProperty<bool> IsWorkingCopyChangeProperty =
|
|
AvaloniaProperty.Register<ChangeCollectionView, bool>(nameof(IsWorkingCopyChange), false);
|
|
|
|
public bool IsWorkingCopyChange
|
|
{
|
|
get => GetValue(IsWorkingCopyChangeProperty);
|
|
set => SetValue(IsWorkingCopyChangeProperty, value);
|
|
}
|
|
|
|
public static readonly StyledProperty<bool> SingleSelectProperty =
|
|
AvaloniaProperty.Register<ChangeCollectionView, bool>(nameof(SingleSelect), true);
|
|
|
|
public bool SingleSelect
|
|
{
|
|
get => GetValue(SingleSelectProperty);
|
|
set => SetValue(SingleSelectProperty, value);
|
|
}
|
|
|
|
public static readonly StyledProperty<Models.ChangeViewMode> ViewModeProperty =
|
|
AvaloniaProperty.Register<ChangeCollectionView, Models.ChangeViewMode>(nameof(ViewMode), Models.ChangeViewMode.Tree);
|
|
|
|
public Models.ChangeViewMode ViewMode
|
|
{
|
|
get => GetValue(ViewModeProperty);
|
|
set => SetValue(ViewModeProperty, value);
|
|
}
|
|
|
|
public static readonly StyledProperty<List<Models.Change>> ChangesProperty =
|
|
AvaloniaProperty.Register<ChangeCollectionView, List<Models.Change>>(nameof(Changes), null);
|
|
|
|
public List<Models.Change> Changes
|
|
{
|
|
get => GetValue(ChangesProperty);
|
|
set => SetValue(ChangesProperty, value);
|
|
}
|
|
|
|
public static readonly StyledProperty<List<Models.Change>> SelectedChangesProperty =
|
|
AvaloniaProperty.Register<ChangeCollectionView, List<Models.Change>>(nameof(SelectedChanges), null);
|
|
|
|
public List<Models.Change> SelectedChanges
|
|
{
|
|
get => GetValue(SelectedChangesProperty);
|
|
set => SetValue(SelectedChangesProperty, value);
|
|
}
|
|
|
|
public static readonly RoutedEvent<RoutedEventArgs> ChangeDoubleTappedEvent =
|
|
RoutedEvent.Register<ChangeCollectionView, RoutedEventArgs>(nameof(ChangeDoubleTapped), RoutingStrategies.Tunnel | RoutingStrategies.Bubble);
|
|
|
|
public event EventHandler<RoutedEventArgs> ChangeDoubleTapped
|
|
{
|
|
add { AddHandler(ChangeDoubleTappedEvent, value); }
|
|
remove { RemoveHandler(ChangeDoubleTappedEvent, value); }
|
|
}
|
|
|
|
static ChangeCollectionView()
|
|
{
|
|
ViewModeProperty.Changed.AddClassHandler<ChangeCollectionView>((c, e) => c.UpdateSource());
|
|
ChangesProperty.Changed.AddClassHandler<ChangeCollectionView>((c, e) => c.UpdateSource());
|
|
SelectedChangesProperty.Changed.AddClassHandler<ChangeCollectionView>((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 = ChangeTreeNode.Build(changes, true);
|
|
var template = this.FindResource("TreeModeTemplate") as IDataTemplate;
|
|
var source = new HierarchicalTreeDataGridSource<ChangeTreeNode>(filetree)
|
|
{
|
|
Columns =
|
|
{
|
|
new HierarchicalExpanderColumn<ChangeTreeNode>(
|
|
new TemplateColumn<ChangeTreeNode>(null, template, null, GridLength.Auto),
|
|
x => x.Children,
|
|
x => x.Children.Count > 0,
|
|
x => x.IsExpanded)
|
|
}
|
|
};
|
|
|
|
var selection = new Models.TreeDataGridSelectionModel<ChangeTreeNode>(source, x => x.Children);
|
|
selection.SingleSelect = SingleSelect;
|
|
selection.RowDoubleTapped += (_, e) => RaiseEvent(new RoutedEventArgs(ChangeDoubleTappedEvent));
|
|
selection.SelectionChanged += (s, _) =>
|
|
{
|
|
if (!_isSelecting && s is Models.TreeDataGridSelectionModel<ChangeTreeNode> model)
|
|
{
|
|
var selected = new List<Models.Change>();
|
|
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<Models.Change>(changes)
|
|
{
|
|
Columns = { new TemplateColumn<Models.Change>(null, template, null, GridLength.Auto) }
|
|
};
|
|
|
|
var selection = new Models.TreeDataGridSelectionModel<Models.Change>(source, null);
|
|
selection.SingleSelect = SingleSelect;
|
|
selection.RowDoubleTapped += (_, e) => RaiseEvent(new RoutedEventArgs(ChangeDoubleTappedEvent));
|
|
selection.SelectionChanged += (s, _) =>
|
|
{
|
|
if (!_isSelecting && s is Models.TreeDataGridSelectionModel<Models.Change> model)
|
|
{
|
|
var selected = new List<Models.Change>();
|
|
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<Models.Change>(changes)
|
|
{
|
|
Columns = { new TemplateColumn<Models.Change>(null, template, null, GridLength.Auto) },
|
|
};
|
|
|
|
var selection = new Models.TreeDataGridSelectionModel<Models.Change>(source, null);
|
|
selection.SingleSelect = SingleSelect;
|
|
selection.RowDoubleTapped += (_, e) => RaiseEvent(new RoutedEventArgs(ChangeDoubleTappedEvent));
|
|
selection.SelectionChanged += (s, _) =>
|
|
{
|
|
if (!_isSelecting && s is Models.TreeDataGridSelectionModel<Models.Change> model)
|
|
{
|
|
var selected = new List<Models.Change>();
|
|
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<Models.Change> changeSelection)
|
|
{
|
|
if (selected == null || selected.Count == 0)
|
|
changeSelection.Clear();
|
|
else
|
|
changeSelection.Select(selected);
|
|
}
|
|
else if (tree.Source.Selection is Models.TreeDataGridSelectionModel<ChangeTreeNode> treeSelection)
|
|
{
|
|
if (selected == null || selected.Count == 0)
|
|
{
|
|
treeSelection.Clear();
|
|
_isSelecting = false;
|
|
return;
|
|
}
|
|
|
|
var set = new HashSet<object>();
|
|
foreach (var c in selected)
|
|
set.Add(c);
|
|
|
|
var nodes = new List<ChangeTreeNode>();
|
|
foreach (var node in tree.Source.Items)
|
|
CollectSelectedNodeByChange(nodes, node as ChangeTreeNode, 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<Models.Change> outs, 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 void CollectSelectedNodeByChange(List<ChangeTreeNode> outs, ChangeTreeNode node, HashSet<object> selected)
|
|
{
|
|
if (node == null)
|
|
return;
|
|
|
|
if (node.IsFolder)
|
|
{
|
|
foreach (var child in node.Children)
|
|
CollectSelectedNodeByChange(outs, child, selected);
|
|
}
|
|
else if (node.Change != null && selected.Contains(node.Change))
|
|
{
|
|
outs.Add(node);
|
|
}
|
|
}
|
|
|
|
private void TrySetSelected(List<Models.Change> 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;
|
|
}
|
|
}
|