2024-06-28 03:57:56 -07:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
2024-10-17 00:37:14 -07:00
|
|
|
|
using System.IO;
|
2024-02-05 23:08:37 -08:00
|
|
|
|
using System.Threading.Tasks;
|
2024-06-06 00:31:02 -07:00
|
|
|
|
|
2024-05-30 00:13:59 -07:00
|
|
|
|
using Avalonia.Controls;
|
2024-03-17 18:37:06 -07:00
|
|
|
|
using Avalonia.Threading;
|
|
|
|
|
|
|
|
|
|
using CommunityToolkit.Mvvm.ComponentModel;
|
|
|
|
|
|
|
|
|
|
namespace SourceGit.ViewModels
|
|
|
|
|
{
|
|
|
|
|
public class StashesPage : ObservableObject
|
|
|
|
|
{
|
|
|
|
|
public List<Models.Stash> Stashes
|
|
|
|
|
{
|
2024-02-05 23:08:37 -08:00
|
|
|
|
get => _stashes;
|
2024-03-17 18:37:06 -07:00
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
if (SetProperty(ref _stashes, value))
|
2024-06-28 03:57:56 -07:00
|
|
|
|
RefreshVisible();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public List<Models.Stash> VisibleStashes
|
|
|
|
|
{
|
|
|
|
|
get => _visibleStashes;
|
|
|
|
|
private set
|
|
|
|
|
{
|
|
|
|
|
if (SetProperty(ref _visibleStashes, value))
|
2024-02-05 23:08:37 -08:00
|
|
|
|
SelectedStash = null;
|
2024-06-28 03:57:56 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public string SearchFilter
|
|
|
|
|
{
|
|
|
|
|
get => _searchFilter;
|
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
if (SetProperty(ref _searchFilter, value))
|
|
|
|
|
RefreshVisible();
|
2024-02-05 23:08:37 -08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-17 18:37:06 -07:00
|
|
|
|
public Models.Stash SelectedStash
|
|
|
|
|
{
|
2024-02-05 23:08:37 -08:00
|
|
|
|
get => _selectedStash;
|
2024-03-17 18:37:06 -07:00
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
if (SetProperty(ref _selectedStash, value))
|
|
|
|
|
{
|
|
|
|
|
if (value == null)
|
|
|
|
|
{
|
2024-02-05 23:08:37 -08:00
|
|
|
|
Changes = null;
|
2024-03-17 18:37:06 -07:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
Task.Run(() =>
|
|
|
|
|
{
|
2024-02-05 23:08:37 -08:00
|
|
|
|
var changes = new Commands.QueryStashChanges(_repo.FullPath, value.SHA).Result();
|
2024-06-28 03:57:56 -07:00
|
|
|
|
Dispatcher.UIThread.Invoke(() => Changes = changes);
|
2024-02-05 23:08:37 -08:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-17 18:37:06 -07:00
|
|
|
|
public List<Models.Change> Changes
|
|
|
|
|
{
|
2024-02-05 23:08:37 -08:00
|
|
|
|
get => _changes;
|
2024-03-17 18:37:06 -07:00
|
|
|
|
private set
|
|
|
|
|
{
|
|
|
|
|
if (SetProperty(ref _changes, value))
|
2024-02-05 23:08:37 -08:00
|
|
|
|
SelectedChange = null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-17 18:37:06 -07:00
|
|
|
|
public Models.Change SelectedChange
|
|
|
|
|
{
|
2024-02-05 23:08:37 -08:00
|
|
|
|
get => _selectedChange;
|
2024-03-17 18:37:06 -07:00
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
if (SetProperty(ref _selectedChange, value))
|
|
|
|
|
{
|
|
|
|
|
if (value == null)
|
2024-02-05 23:08:37 -08:00
|
|
|
|
DiffContext = null;
|
2024-03-17 18:37:06 -07:00
|
|
|
|
else
|
2024-03-20 03:27:48 -07:00
|
|
|
|
DiffContext = new DiffContext(_repo.FullPath, new Models.DiffOption($"{_selectedStash.SHA}^", _selectedStash.SHA, value), _diffContext);
|
2024-02-05 23:08:37 -08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-17 18:37:06 -07:00
|
|
|
|
public DiffContext DiffContext
|
|
|
|
|
{
|
2024-02-05 23:08:37 -08:00
|
|
|
|
get => _diffContext;
|
|
|
|
|
private set => SetProperty(ref _diffContext, value);
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-17 18:37:06 -07:00
|
|
|
|
public StashesPage(Repository repo)
|
|
|
|
|
{
|
2024-02-05 23:08:37 -08:00
|
|
|
|
_repo = repo;
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-17 18:37:06 -07:00
|
|
|
|
public void Cleanup()
|
|
|
|
|
{
|
2024-02-20 02:27:59 -08:00
|
|
|
|
_repo = null;
|
2024-03-31 01:54:29 -07:00
|
|
|
|
if (_stashes != null)
|
|
|
|
|
_stashes.Clear();
|
2024-02-20 02:27:59 -08:00
|
|
|
|
_selectedStash = null;
|
2024-03-31 01:54:29 -07:00
|
|
|
|
if (_changes != null)
|
|
|
|
|
_changes.Clear();
|
2024-02-20 02:27:59 -08:00
|
|
|
|
_selectedChange = null;
|
|
|
|
|
_diffContext = null;
|
|
|
|
|
}
|
|
|
|
|
|
2024-05-30 00:13:59 -07:00
|
|
|
|
public ContextMenu MakeContextMenu(Models.Stash stash)
|
2024-03-17 18:37:06 -07:00
|
|
|
|
{
|
2024-05-30 00:13:59 -07:00
|
|
|
|
if (stash == null)
|
|
|
|
|
return null;
|
|
|
|
|
|
|
|
|
|
var apply = new MenuItem();
|
|
|
|
|
apply.Header = App.Text("StashCM.Apply");
|
2024-07-14 09:30:31 -07:00
|
|
|
|
apply.Click += (_, ev) =>
|
2024-03-17 18:37:06 -07:00
|
|
|
|
{
|
2024-05-30 00:13:59 -07:00
|
|
|
|
Task.Run(() => new Commands.Stash(_repo.FullPath).Apply(stash.Name));
|
|
|
|
|
ev.Handled = true;
|
|
|
|
|
};
|
2024-02-05 23:08:37 -08:00
|
|
|
|
|
2024-05-30 00:13:59 -07:00
|
|
|
|
var pop = new MenuItem();
|
|
|
|
|
pop.Header = App.Text("StashCM.Pop");
|
2024-07-14 09:30:31 -07:00
|
|
|
|
pop.Click += (_, ev) =>
|
2024-03-17 18:37:06 -07:00
|
|
|
|
{
|
2024-05-30 00:13:59 -07:00
|
|
|
|
Task.Run(() => new Commands.Stash(_repo.FullPath).Pop(stash.Name));
|
|
|
|
|
ev.Handled = true;
|
|
|
|
|
};
|
2024-02-05 23:08:37 -08:00
|
|
|
|
|
2024-05-30 00:13:59 -07:00
|
|
|
|
var drop = new MenuItem();
|
|
|
|
|
drop.Header = App.Text("StashCM.Drop");
|
2024-07-14 09:30:31 -07:00
|
|
|
|
drop.Click += (_, ev) =>
|
2024-03-17 18:37:06 -07:00
|
|
|
|
{
|
2024-05-30 00:13:59 -07:00
|
|
|
|
if (PopupHost.CanCreatePopup())
|
|
|
|
|
PopupHost.ShowPopup(new DropStash(_repo.FullPath, stash));
|
|
|
|
|
|
|
|
|
|
ev.Handled = true;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var menu = new ContextMenu();
|
|
|
|
|
menu.Items.Add(apply);
|
|
|
|
|
menu.Items.Add(pop);
|
|
|
|
|
menu.Items.Add(drop);
|
|
|
|
|
return menu;
|
2024-02-05 23:08:37 -08:00
|
|
|
|
}
|
|
|
|
|
|
2024-10-17 00:37:14 -07:00
|
|
|
|
public ContextMenu MakeContextMenuForChange(Models.Change change)
|
|
|
|
|
{
|
|
|
|
|
if (change == null)
|
|
|
|
|
return null;
|
|
|
|
|
|
|
|
|
|
var diffWithMerger = new MenuItem();
|
|
|
|
|
diffWithMerger.Header = App.Text("DiffWithMerger");
|
|
|
|
|
diffWithMerger.Icon = App.CreateMenuIcon("Icons.OpenWith");
|
|
|
|
|
diffWithMerger.Click += (_, ev) =>
|
|
|
|
|
{
|
|
|
|
|
var toolType = Preference.Instance.ExternalMergeToolType;
|
|
|
|
|
var toolPath = Preference.Instance.ExternalMergeToolPath;
|
|
|
|
|
var opt = new Models.DiffOption($"{_selectedStash.SHA}^", _selectedStash.SHA, change);
|
|
|
|
|
|
|
|
|
|
Task.Run(() => Commands.MergeTool.OpenForDiff(_repo.FullPath, toolType, toolPath, opt));
|
|
|
|
|
ev.Handled = true;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var fullPath = Path.Combine(_repo.FullPath, change.Path);
|
|
|
|
|
var explore = new MenuItem();
|
|
|
|
|
explore.Header = App.Text("RevealFile");
|
|
|
|
|
explore.Icon = App.CreateMenuIcon("Icons.Explore");
|
|
|
|
|
explore.IsEnabled = File.Exists(fullPath);
|
|
|
|
|
explore.Click += (_, ev) =>
|
|
|
|
|
{
|
|
|
|
|
Native.OS.OpenInFileManager(fullPath, true);
|
|
|
|
|
ev.Handled = true;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var resetToThisRevision = new MenuItem();
|
|
|
|
|
resetToThisRevision.Header = App.Text("ChangeCM.CheckoutThisRevision");
|
|
|
|
|
resetToThisRevision.Icon = App.CreateMenuIcon("Icons.File.Checkout");
|
|
|
|
|
resetToThisRevision.Click += (_, ev) =>
|
|
|
|
|
{
|
|
|
|
|
new Commands.Checkout(_repo.FullPath).FileWithRevision(change.Path, $"{_selectedStash.SHA}");
|
|
|
|
|
ev.Handled = true;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var copyPath = new MenuItem();
|
|
|
|
|
copyPath.Header = App.Text("CopyPath");
|
|
|
|
|
copyPath.Icon = App.CreateMenuIcon("Icons.Copy");
|
|
|
|
|
copyPath.Click += (_, ev) =>
|
|
|
|
|
{
|
|
|
|
|
App.CopyText(change.Path);
|
|
|
|
|
ev.Handled = true;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var copyFileName = new MenuItem();
|
|
|
|
|
copyFileName.Header = App.Text("CopyFileName");
|
|
|
|
|
copyFileName.Icon = App.CreateMenuIcon("Icons.Copy");
|
|
|
|
|
copyFileName.Click += (_, e) =>
|
|
|
|
|
{
|
|
|
|
|
App.CopyText(Path.GetFileName(change.Path));
|
|
|
|
|
e.Handled = true;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var menu = new ContextMenu();
|
|
|
|
|
menu.Items.Add(diffWithMerger);
|
|
|
|
|
menu.Items.Add(explore);
|
|
|
|
|
menu.Items.Add(new MenuItem { Header = "-" });
|
|
|
|
|
menu.Items.Add(resetToThisRevision);
|
|
|
|
|
menu.Items.Add(new MenuItem { Header = "-" });
|
|
|
|
|
menu.Items.Add(copyPath);
|
|
|
|
|
menu.Items.Add(copyFileName);
|
|
|
|
|
|
|
|
|
|
return menu;
|
|
|
|
|
}
|
|
|
|
|
|
2024-03-17 18:37:06 -07:00
|
|
|
|
public void Clear()
|
|
|
|
|
{
|
|
|
|
|
if (PopupHost.CanCreatePopup())
|
2024-02-05 23:08:37 -08:00
|
|
|
|
PopupHost.ShowPopup(new ClearStashes(_repo));
|
2024-06-28 03:57:56 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void ClearSearchFilter()
|
|
|
|
|
{
|
|
|
|
|
SearchFilter = string.Empty;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void RefreshVisible()
|
|
|
|
|
{
|
|
|
|
|
if (string.IsNullOrEmpty(_searchFilter))
|
|
|
|
|
{
|
|
|
|
|
VisibleStashes = _stashes;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
var visible = new List<Models.Stash>();
|
|
|
|
|
foreach (var s in _stashes)
|
|
|
|
|
{
|
|
|
|
|
if (s.Message.Contains(_searchFilter, StringComparison.OrdinalIgnoreCase))
|
|
|
|
|
visible.Add(s);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
VisibleStashes = visible;
|
2024-02-05 23:08:37 -08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private Repository _repo = null;
|
2024-06-28 03:57:56 -07:00
|
|
|
|
private List<Models.Stash> _stashes = new List<Models.Stash>();
|
|
|
|
|
private List<Models.Stash> _visibleStashes = new List<Models.Stash>();
|
|
|
|
|
private string _searchFilter = string.Empty;
|
2024-02-05 23:08:37 -08:00
|
|
|
|
private Models.Stash _selectedStash = null;
|
|
|
|
|
private List<Models.Change> _changes = null;
|
|
|
|
|
private Models.Change _selectedChange = null;
|
|
|
|
|
private DiffContext _diffContext = null;
|
|
|
|
|
}
|
2024-03-31 01:54:29 -07:00
|
|
|
|
}
|