mirror of
https://github.com/sourcegit-scm/sourcegit.git
synced 2025-01-11 23:57:21 -08:00
feature<Welcome>: supports set bookmark of selected repository from context menu.
This commit is contained in:
parent
7e1b1d7324
commit
5d3088d520
6 changed files with 97 additions and 78 deletions
|
@ -14,6 +14,11 @@ namespace SourceGit.Models {
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static event Action<Repository> Opened;
|
public static event Action<Repository> Opened;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 仓库的书签变化了
|
||||||
|
/// </summary>
|
||||||
|
public static event Action<string, int> BookmarkChanged;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 跳转到指定提交的事件
|
/// 跳转到指定提交的事件
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
@ -97,6 +102,15 @@ namespace SourceGit.Models {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 设置仓库标签变化
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="repo"></param>
|
||||||
|
/// <param name="bookmark"></param>
|
||||||
|
public static void SetBookmark(string repo, int bookmark) {
|
||||||
|
BookmarkChanged?.Invoke(repo, bookmark);
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 跳转到指定的提交
|
/// 跳转到指定的提交
|
||||||
/// </summary>
|
/// </summary>
|
||||||
|
|
|
@ -1,54 +0,0 @@
|
||||||
using System.ComponentModel;
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
|
|
||||||
namespace SourceGit.Views.Controls {
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Implementation of <see cref="INotifyPropertyChanged" /> to simplify models.
|
|
||||||
/// </summary>
|
|
||||||
public abstract class BindableBase : INotifyPropertyChanged {
|
|
||||||
/// <summary>
|
|
||||||
/// Multicast event for property change notifications.
|
|
||||||
/// </summary>
|
|
||||||
public event PropertyChangedEventHandler PropertyChanged;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Checks if a property already matches a desired value. Sets the property and
|
|
||||||
/// notifies listeners only when necessary.
|
|
||||||
/// </summary>
|
|
||||||
/// <typeparam name="T">Type of the property.</typeparam>
|
|
||||||
/// <param name="storage">Reference to a property with both getter and setter.</param>
|
|
||||||
/// <param name="value">Desired value for the property.</param>
|
|
||||||
/// <param name="propertyName">
|
|
||||||
/// Name of the property used to notify listeners. This
|
|
||||||
/// value is optional and can be provided automatically when invoked from compilers that
|
|
||||||
/// support CallerMemberName.
|
|
||||||
/// </param>
|
|
||||||
/// <returns>
|
|
||||||
/// True if the value was changed, false if the existing value matched the
|
|
||||||
/// desired value.
|
|
||||||
/// </returns>
|
|
||||||
protected bool SetProperty<T>(ref T storage, T value, [CallerMemberName] string propertyName = null) {
|
|
||||||
if (Equals(storage, value)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
storage = value;
|
|
||||||
this.OnPropertyChanged(propertyName);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Notifies listeners that a property value has changed.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="propertyName">
|
|
||||||
/// Name of the property used to notify listeners. This
|
|
||||||
/// value is optional and can be provided automatically when invoked from compilers
|
|
||||||
/// that support <see cref="CallerMemberNameAttribute" />.
|
|
||||||
/// </param>
|
|
||||||
protected void OnPropertyChanged([CallerMemberName] string propertyName = null) {
|
|
||||||
this.PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -19,14 +19,6 @@ namespace SourceGit.Views {
|
||||||
Models.Watcher.Opened += OpenRepository;
|
Models.Watcher.Opened += OpenRepository;
|
||||||
InitializeComponent();
|
InitializeComponent();
|
||||||
tabs.Add();
|
tabs.Add();
|
||||||
|
|
||||||
tabs.OnTabEdited += (t) => {
|
|
||||||
foreach (var tab in tabs.Tabs) {
|
|
||||||
if (tab.IsRepository) continue;
|
|
||||||
var page = container.Get(tab.Id) as Widgets.Welcome;
|
|
||||||
if (page != null) page.UpdateVisibles();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnClosing(object sender, CancelEventArgs e) {
|
private void OnClosing(object sender, CancelEventArgs e) {
|
||||||
|
|
|
@ -1,6 +1,8 @@
|
||||||
using System;
|
using System;
|
||||||
using System.Collections.ObjectModel;
|
using System.Collections.ObjectModel;
|
||||||
|
using System.ComponentModel;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
using System.Windows;
|
using System.Windows;
|
||||||
using System.Windows.Controls;
|
using System.Windows.Controls;
|
||||||
using System.Windows.Input;
|
using System.Windows.Input;
|
||||||
|
@ -16,7 +18,9 @@ namespace SourceGit.Views.Widgets {
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 标签数据
|
/// 标签数据
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class Tab : Controls.BindableBase {
|
public class Tab : INotifyPropertyChanged {
|
||||||
|
public event PropertyChangedEventHandler PropertyChanged;
|
||||||
|
|
||||||
public string Id { get; set; }
|
public string Id { get; set; }
|
||||||
public bool IsRepository { get; set; }
|
public bool IsRepository { get; set; }
|
||||||
|
|
||||||
|
@ -39,12 +43,13 @@ namespace SourceGit.Views.Widgets {
|
||||||
get => isSeperatorVisible;
|
get => isSeperatorVisible;
|
||||||
set => SetProperty(ref isSeperatorVisible, value);
|
set => SetProperty(ref isSeperatorVisible, value);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
public void SetProperty<T>(ref T storage, T value, [CallerMemberName] string propName = null) {
|
||||||
/// 仓库标签页编辑事件参数
|
if (Equals(storage, value)) return;
|
||||||
/// </summary>
|
storage = value;
|
||||||
public event Action<Tab> OnTabEdited;
|
PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propName));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 标签相关事件参数
|
/// 标签相关事件参数
|
||||||
|
@ -99,6 +104,15 @@ namespace SourceGit.Views.Widgets {
|
||||||
public PageTabBar() {
|
public PageTabBar() {
|
||||||
Tabs = new ObservableCollection<Tab>();
|
Tabs = new ObservableCollection<Tab>();
|
||||||
InitializeComponent();
|
InitializeComponent();
|
||||||
|
|
||||||
|
Models.Watcher.BookmarkChanged += (repoPath, bookmark) => {
|
||||||
|
foreach (var tab in Tabs) {
|
||||||
|
if (tab.Id == repoPath) {
|
||||||
|
tab.Bookmark = bookmark;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
public void Add() {
|
public void Add() {
|
||||||
|
@ -350,12 +364,7 @@ namespace SourceGit.Views.Widgets {
|
||||||
|
|
||||||
var refIdx = i;
|
var refIdx = i;
|
||||||
mark.Click += (o, ev) => {
|
mark.Click += (o, ev) => {
|
||||||
var repo = Models.Preference.Instance.FindRepository(tab.Id);
|
Models.Watcher.SetBookmark(tab.Id, refIdx);
|
||||||
if (repo != null) {
|
|
||||||
repo.Bookmark = refIdx;
|
|
||||||
tab.Bookmark = refIdx;
|
|
||||||
OnTabEdited?.Invoke(tab);
|
|
||||||
}
|
|
||||||
ev.Handled = true;
|
ev.Handled = true;
|
||||||
};
|
};
|
||||||
bookmark.Items.Add(mark);
|
bookmark.Items.Add(mark);
|
||||||
|
|
|
@ -107,7 +107,7 @@
|
||||||
|
|
||||||
<ItemsControl.ItemTemplate>
|
<ItemsControl.ItemTemplate>
|
||||||
<DataTemplate>
|
<DataTemplate>
|
||||||
<Control MouseDoubleClick="OnDoubleClickRepository">
|
<Control MouseDoubleClick="OnDoubleClickRepository" ContextMenuOpening="OnRepositoryContextMenuOpening">
|
||||||
<Control.Template>
|
<Control.Template>
|
||||||
<ControlTemplate>
|
<ControlTemplate>
|
||||||
<Grid Height="38" Margin="2,0,2,6">
|
<Grid Height="38" Margin="2,0,2,6">
|
||||||
|
|
|
@ -19,6 +19,14 @@ namespace SourceGit.Views.Widgets {
|
||||||
InitializeComponent();
|
InitializeComponent();
|
||||||
UpdateVisibles();
|
UpdateVisibles();
|
||||||
Models.Theme.AddListener(this, UpdateVisibles);
|
Models.Theme.AddListener(this, UpdateVisibles);
|
||||||
|
|
||||||
|
Models.Watcher.BookmarkChanged += (repoPath, bookmark) => {
|
||||||
|
var repo = Models.Preference.Instance.FindRepository(repoPath);
|
||||||
|
if (repo != null) {
|
||||||
|
repo.Bookmark = bookmark;
|
||||||
|
UpdateVisibles();
|
||||||
|
}
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
#region FUNC_EVENTS
|
#region FUNC_EVENTS
|
||||||
|
@ -89,7 +97,7 @@ namespace SourceGit.Views.Widgets {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnRemoveRepository(object sender, RoutedEventArgs e) {
|
private void OnRemoveRepository(object sender, RoutedEventArgs e) {
|
||||||
var repo = (sender as Button).DataContext as Models.Repository;
|
var repo = (sender as Control).DataContext as Models.Repository;
|
||||||
if (repo == null) return;
|
if (repo == null) return;
|
||||||
|
|
||||||
var confirmDialog = new ConfirmDialog(
|
var confirmDialog = new ConfirmDialog(
|
||||||
|
@ -108,6 +116,56 @@ namespace SourceGit.Views.Widgets {
|
||||||
OnOpenRepository(sender, e);
|
OnOpenRepository(sender, e);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void OnRepositoryContextMenuOpening(object sender, ContextMenuEventArgs e) {
|
||||||
|
var control = sender as Control;
|
||||||
|
if (control == null) return;
|
||||||
|
|
||||||
|
var repo = control.DataContext as Models.Repository;
|
||||||
|
if (repo == null) return;
|
||||||
|
|
||||||
|
var menu = new ContextMenu();
|
||||||
|
menu.Placement = PlacementMode.MousePoint;
|
||||||
|
menu.PlacementTarget = control;
|
||||||
|
menu.StaysOpen = false;
|
||||||
|
menu.Focusable = true;
|
||||||
|
|
||||||
|
var open = new MenuItem();
|
||||||
|
open.Header = App.Text("RepoCM.Open");
|
||||||
|
open.Click += OnOpenRepository;
|
||||||
|
menu.Items.Add(open);
|
||||||
|
menu.Items.Add(new Separator());
|
||||||
|
|
||||||
|
var bookmark = new MenuItem();
|
||||||
|
bookmark.Header = App.Text("PageTabBar.Tab.Bookmark");
|
||||||
|
for (int i = 0; i < Converters.IntToBookmarkBrush.COLORS.Length; i++) {
|
||||||
|
var icon = new System.Windows.Shapes.Path();
|
||||||
|
icon.Data = new EllipseGeometry(new Point(0, 0), 12, 12);
|
||||||
|
icon.Fill = Converters.IntToBookmarkBrush.COLORS[i];
|
||||||
|
icon.Width = 12;
|
||||||
|
|
||||||
|
var mark = new MenuItem();
|
||||||
|
mark.Icon = icon;
|
||||||
|
mark.Header = $"{i}";
|
||||||
|
|
||||||
|
var refIdx = i;
|
||||||
|
mark.Click += (o, ev) => {
|
||||||
|
Models.Watcher.SetBookmark(repo.Path, refIdx);
|
||||||
|
ev.Handled = true;
|
||||||
|
};
|
||||||
|
bookmark.Items.Add(mark);
|
||||||
|
}
|
||||||
|
menu.Items.Add(bookmark);
|
||||||
|
menu.Items.Add(new Separator());
|
||||||
|
|
||||||
|
var remove = new MenuItem();
|
||||||
|
remove.Header = App.Text("Welcome.Delete");
|
||||||
|
remove.Click += OnRemoveRepository;
|
||||||
|
menu.Items.Add(remove);
|
||||||
|
|
||||||
|
menu.IsOpen = true;
|
||||||
|
e.Handled = true;
|
||||||
|
}
|
||||||
|
|
||||||
private void OnOpenRepository(object sender, RoutedEventArgs e) {
|
private void OnOpenRepository(object sender, RoutedEventArgs e) {
|
||||||
var repo = (sender as Control).DataContext as Models.Repository;
|
var repo = (sender as Control).DataContext as Models.Repository;
|
||||||
if (repo == null) return;
|
if (repo == null) return;
|
||||||
|
@ -125,7 +183,7 @@ namespace SourceGit.Views.Widgets {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void OnOpenRepositoryTerminal(object sender, RoutedEventArgs e) {
|
private void OnOpenRepositoryTerminal(object sender, RoutedEventArgs e) {
|
||||||
var repo = (sender as Button).DataContext as Models.Repository;
|
var repo = (sender as Control).DataContext as Models.Repository;
|
||||||
if (repo == null) return;
|
if (repo == null) return;
|
||||||
|
|
||||||
var bash = Path.Combine(Models.Preference.Instance.Git.Path, "..", "bash.exe");
|
var bash = Path.Combine(Models.Preference.Instance.Git.Path, "..", "bash.exe");
|
||||||
|
|
Loading…
Reference in a new issue