mirror of
https://github.com/sourcegit-scm/sourcegit
synced 2025-05-22 04:34:59 +00:00
feature: supports sort branches by committer date (#1192)
Signed-off-by: leo <longshuang@msn.cn>
This commit is contained in:
parent
b7fa04d141
commit
e45e37d305
10 changed files with 217 additions and 20 deletions
|
@ -10,6 +10,7 @@ namespace SourceGit.ViewModels
|
|||
public string Name { get; private set; } = string.Empty;
|
||||
public string Path { get; private set; } = string.Empty;
|
||||
public object Backend { get; private set; } = null;
|
||||
public ulong TimeToSort { get; private set; } = 0;
|
||||
public int Depth { get; set; } = 0;
|
||||
public bool IsSelected { get; set; } = false;
|
||||
public List<BranchTreeNode> Children { get; private set; } = new List<BranchTreeNode>();
|
||||
|
@ -62,6 +63,12 @@ namespace SourceGit.ViewModels
|
|||
public List<BranchTreeNode> Remotes => _remotes;
|
||||
public List<string> InvalidExpandedNodes => _invalidExpandedNodes;
|
||||
|
||||
public Builder(Models.BranchSortMode localSortMode, Models.BranchSortMode remoteSortMode)
|
||||
{
|
||||
_localSortMode = localSortMode;
|
||||
_remoteSortMode = remoteSortMode;
|
||||
}
|
||||
|
||||
public void SetExpandedNodes(List<string> expanded)
|
||||
{
|
||||
foreach (var node in expanded)
|
||||
|
@ -72,6 +79,7 @@ namespace SourceGit.ViewModels
|
|||
{
|
||||
var folders = new Dictionary<string, BranchTreeNode>();
|
||||
|
||||
var fakeRemoteTime = (ulong)remotes.Count;
|
||||
foreach (var remote in remotes)
|
||||
{
|
||||
var path = $"refs/remotes/{remote.Name}";
|
||||
|
@ -81,8 +89,10 @@ namespace SourceGit.ViewModels
|
|||
Path = path,
|
||||
Backend = remote,
|
||||
IsExpanded = bForceExpanded || _expanded.Contains(path),
|
||||
TimeToSort = fakeRemoteTime,
|
||||
};
|
||||
|
||||
fakeRemoteTime--;
|
||||
folders.Add(path, node);
|
||||
_remotes.Add(node);
|
||||
}
|
||||
|
@ -108,8 +118,26 @@ namespace SourceGit.ViewModels
|
|||
}
|
||||
|
||||
folders.Clear();
|
||||
SortNodes(_locals);
|
||||
SortNodes(_remotes);
|
||||
|
||||
if (_localSortMode == Models.BranchSortMode.Name)
|
||||
{
|
||||
SortNodesByName(_locals);
|
||||
}
|
||||
else
|
||||
{
|
||||
SetTimeToSortRecusive(_locals);
|
||||
SortNodesByTime(_locals);
|
||||
}
|
||||
|
||||
if (_remoteSortMode == Models.BranchSortMode.Name)
|
||||
{
|
||||
SortNodesByName(_remotes);
|
||||
}
|
||||
else
|
||||
{
|
||||
SetTimeToSortRecusive(_remotes);
|
||||
SortNodesByTime(_remotes);
|
||||
}
|
||||
}
|
||||
|
||||
private void MakeBranchNode(Models.Branch branch, List<BranchTreeNode> roots, Dictionary<string, BranchTreeNode> folders, string prefix, bool bForceExpanded)
|
||||
|
@ -124,6 +152,7 @@ namespace SourceGit.ViewModels
|
|||
Path = fullpath,
|
||||
Backend = branch,
|
||||
IsExpanded = false,
|
||||
TimeToSort = branch.CommitterDate,
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
@ -175,10 +204,11 @@ namespace SourceGit.ViewModels
|
|||
Path = fullpath,
|
||||
Backend = branch,
|
||||
IsExpanded = false,
|
||||
TimeToSort = branch.CommitterDate,
|
||||
});
|
||||
}
|
||||
|
||||
private void SortNodes(List<BranchTreeNode> nodes)
|
||||
private void SortNodesByName(List<BranchTreeNode> nodes)
|
||||
{
|
||||
nodes.Sort((l, r) =>
|
||||
{
|
||||
|
@ -192,9 +222,61 @@ namespace SourceGit.ViewModels
|
|||
});
|
||||
|
||||
foreach (var node in nodes)
|
||||
SortNodes(node.Children);
|
||||
SortNodesByName(node.Children);
|
||||
}
|
||||
|
||||
private void SortNodesByTime(List<BranchTreeNode> nodes)
|
||||
{
|
||||
nodes.Sort((l, r) =>
|
||||
{
|
||||
if (l.Backend is Models.Branch { IsDetachedHead: true })
|
||||
return -1;
|
||||
|
||||
if (l.Backend is Models.Branch)
|
||||
{
|
||||
if (r.Backend is Models.Branch)
|
||||
return r.TimeToSort == l.TimeToSort ? Models.NumericSort.Compare(l.Name, r.Name) : r.TimeToSort.CompareTo(l.TimeToSort);
|
||||
else
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (r.Backend is Models.Branch)
|
||||
return -1;
|
||||
|
||||
if (r.TimeToSort == l.TimeToSort)
|
||||
return Models.NumericSort.Compare(l.Name, r.Name);
|
||||
|
||||
return r.TimeToSort.CompareTo(l.TimeToSort);
|
||||
});
|
||||
|
||||
foreach (var node in nodes)
|
||||
SortNodesByTime(node.Children);
|
||||
}
|
||||
|
||||
private ulong SetTimeToSortRecusive(List<BranchTreeNode> nodes)
|
||||
{
|
||||
var recent = (ulong)0;
|
||||
|
||||
foreach (var node in nodes)
|
||||
{
|
||||
if (node.Backend is Models.Branch)
|
||||
{
|
||||
recent = Math.Max(recent, node.TimeToSort);
|
||||
continue;
|
||||
}
|
||||
|
||||
var time = SetTimeToSortRecusive(node.Children);
|
||||
recent = Math.Max(recent, time);
|
||||
|
||||
if (node.Backend is not Models.Remote)
|
||||
node.TimeToSort = time;
|
||||
}
|
||||
|
||||
return recent;
|
||||
}
|
||||
|
||||
private readonly Models.BranchSortMode _localSortMode = Models.BranchSortMode.Name;
|
||||
private readonly Models.BranchSortMode _remoteSortMode = Models.BranchSortMode.Name;
|
||||
private readonly List<BranchTreeNode> _locals = new List<BranchTreeNode>();
|
||||
private readonly List<BranchTreeNode> _remotes = new List<BranchTreeNode>();
|
||||
private readonly List<string> _invalidExpandedNodes = new List<string>();
|
||||
|
|
|
@ -2229,6 +2229,51 @@ namespace SourceGit.ViewModels
|
|||
return menu;
|
||||
}
|
||||
|
||||
public ContextMenu CreateContextMenuForBranchSortMode(bool local)
|
||||
{
|
||||
var mode = local ? _settings.LocalBranchSortMode : _settings.RemoteBranchSortMode;
|
||||
var changeMode = new Action<Models.BranchSortMode>(m =>
|
||||
{
|
||||
if (local)
|
||||
_settings.LocalBranchSortMode = m;
|
||||
else
|
||||
_settings.RemoteBranchSortMode = m;
|
||||
|
||||
var builder = BuildBranchTree(_branches, _remotes);
|
||||
LocalBranchTrees = builder.Locals;
|
||||
RemoteBranchTrees = builder.Remotes;
|
||||
});
|
||||
|
||||
var byNameAsc = new MenuItem();
|
||||
byNameAsc.Header = App.Text("Repository.BranchSort.ByName");
|
||||
if (mode == Models.BranchSortMode.Name)
|
||||
byNameAsc.Icon = App.CreateMenuIcon("Icons.Check");
|
||||
byNameAsc.Click += (_, ev) =>
|
||||
{
|
||||
if (mode != Models.BranchSortMode.Name)
|
||||
changeMode(Models.BranchSortMode.Name);
|
||||
|
||||
ev.Handled = true;
|
||||
};
|
||||
|
||||
var byCommitterDate = new MenuItem();
|
||||
byCommitterDate.Header = App.Text("Repository.BranchSort.ByCommitterDate");
|
||||
if (mode == Models.BranchSortMode.CommitterDate)
|
||||
byCommitterDate.Icon = App.CreateMenuIcon("Icons.Check");
|
||||
byCommitterDate.Click += (_, ev) =>
|
||||
{
|
||||
if (mode != Models.BranchSortMode.CommitterDate)
|
||||
changeMode(Models.BranchSortMode.CommitterDate);
|
||||
|
||||
ev.Handled = true;
|
||||
};
|
||||
|
||||
var menu = new ContextMenu();
|
||||
menu.Items.Add(byNameAsc);
|
||||
menu.Items.Add(byCommitterDate);
|
||||
return menu;
|
||||
}
|
||||
|
||||
public ContextMenu CreateContextMenuForTagSortMode()
|
||||
{
|
||||
var mode = _settings.TagSortMode;
|
||||
|
@ -2398,7 +2443,7 @@ namespace SourceGit.ViewModels
|
|||
|
||||
private BranchTreeNode.Builder BuildBranchTree(List<Models.Branch> branches, List<Models.Remote> remotes)
|
||||
{
|
||||
var builder = new BranchTreeNode.Builder();
|
||||
var builder = new BranchTreeNode.Builder(_settings.LocalBranchSortMode, _settings.RemoteBranchSortMode);
|
||||
if (string.IsNullOrEmpty(_filter))
|
||||
{
|
||||
builder.SetExpandedNodes(_settings.ExpandedBranchNodesInSideBar);
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue