diff --git a/src/Commands/SubTree.cs b/src/Commands/SubTree.cs new file mode 100644 index 00000000..6c963d09 --- /dev/null +++ b/src/Commands/SubTree.cs @@ -0,0 +1,39 @@ +using System; + +namespace SourceGit.Commands { + /// + /// 子树相关操作 + /// + public class SubTree : Command { + private Action handler = null; + + public SubTree(string repo) { + Cwd = repo; + TraitErrorAsOutput = true; + } + + public override void OnReadline(string line) { + handler?.Invoke(line); + } + + public bool Add(string prefix, string source, string revision, bool squash, Action onProgress) { + handler = onProgress; + Args = $"subtree add --prefix=\"{prefix}\" {source} {revision}"; + if (squash) Args += " --squash"; + return Exec(); + } + + public void Pull(string prefix, string source, string branch, bool squash, Action onProgress) { + handler = onProgress; + Args = $"subtree pull --prefix=\"{prefix}\" {source} {branch}"; + if (squash) Args += " --squash"; + Exec(); + } + + public void Push(string prefix, string source, string branch, Action onProgress) { + handler = onProgress; + Args = $"subtree push --prefix=\"{prefix}\" {source} {branch}"; + Exec(); + } + } +} diff --git a/src/Models/Repository.cs b/src/Models/Repository.cs index d900f12e..36df309b 100644 --- a/src/Models/Repository.cs +++ b/src/Models/Repository.cs @@ -19,6 +19,7 @@ namespace SourceGit.Models { public string GitDir { get; set; } = ""; public string GroupId { get; set; } = ""; public int Bookmark { get; set; } = 0; + public List SubTrees { get; set; } = new List(); public List Filters { get; set; } = new List(); public List CommitMessages { get; set; } = new List(); #endregion diff --git a/src/Models/SubTree.cs b/src/Models/SubTree.cs new file mode 100644 index 00000000..fdbad2fe --- /dev/null +++ b/src/Models/SubTree.cs @@ -0,0 +1,10 @@ +namespace SourceGit.Models { + /// + /// 子树 + /// + public class SubTree { + public string Prefix { get; set; } + public string Remote { get; set; } + public string Branch { get; set; } = "master"; + } +} diff --git a/src/Models/Watcher.cs b/src/Models/Watcher.cs index d1c16927..784faa6a 100644 --- a/src/Models/Watcher.cs +++ b/src/Models/Watcher.cs @@ -38,6 +38,10 @@ namespace SourceGit.Models { /// 子模块变更 /// public event Action SubmoduleChanged; + /// + /// 树更新 + /// + public event Action SubTreeChanged; /// /// 打开仓库事件 @@ -119,6 +123,13 @@ namespace SourceGit.Models { WorkingCopyChanged?.Invoke(); } + /// + /// 通知更新子树列表 + /// + public void RefreshSubTrees() { + SubTreeChanged?.Invoke(); + } + private void Start(string repo, string gitDir) { wcWatcher = new FileSystemWatcher(); wcWatcher.Path = repo; @@ -163,6 +174,7 @@ namespace SourceGit.Models { TagChanged = null; StashChanged = null; SubmoduleChanged = null; + SubTreeChanged = null; } private void OnRepositoryChanged(object o, FileSystemEventArgs e) { diff --git a/src/Resources/Icons.xaml b/src/Resources/Icons.xaml index 1149947b..b23ea37e 100644 --- a/src/Resources/Icons.xaml +++ b/src/Resources/Icons.xaml @@ -2,13 +2,15 @@ xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"> M1004.8 466.4 557.7 19.3c-25.7-25.8-67.5-25.8-93.3 0L360.6 123.2l78.2 78.2c12.5-6 26.6-9.4 41.4-9.4c53 0 96 43 96 96c0 14.8-3.4 28.9-9.4 41.4l128 128c12.5-6 26.6-9.4 41.4-9.4c53 0 96 43 96 96s-43 96-96 96s-96-43-96-96c0-14.8 3.4-28.9 9.4-41.4L521.5 374.6a88.8 88.8 0 01-9.4 3.9v267c37.3 13.2 64 48.7 64 90.5c0 53-43 96-96 96s-96-43-96-96c0-41.8 26.7-77.3 64-90.5V378.5c-37.3-13.2-64-48.7-64-90.5c0-14.8 3.4-28.9 9.4-41.4l-78.2-78.2L19.4 464.3c-25.8 25.8-25.8 67.5 0 93.3l447.1 447.1c25.7 25.8 67.5 25.8 93.3 0l445-445c25.8-25.8 25.8-67.6 0-93.3z M557.7 545.3 789.9 402.7c24-15 31.3-46.5 16.4-70.5c-14.8-23.8-46-31.2-70-16.7L506.5 456.6 277.1 315.4c-24.1-14.8-55.6-7.3-70.5 16.8c-14.8 24.1-7.3 55.6 16.8 70.5l231.8 142.6V819.1c0 28.3 22.9 51.2 51.2 51.2c28.3 0 51.2-22.9 51.2-51.2V545.3h.1zM506.5 0l443.4 256v511.9L506.5 1023.9 63.1 767.9v-511.9L506.5 0z + M491 256h469c13 0 21-9 21-21v-171c0-13-9-21-21-21h-469c-13 0-21 9-21 21V128H256V64c0-13-9-21-21-21h-171c-13 0-21 9-21 21v171c0 13 9 21 21 21H128v597h341v64c0 13 9 21 21 21h469c13 0 21-9 21-21v-171c0-13-9-21-21-21h-469c-13 0-21 9-21 21V811H171v-299h299v64c0 13 9 21 21 21h469c13 0 21-9 21-21v-171c0-13-9-21-21-21h-469c-13 0-21 9-21 21V469H171V256h64c13 0 21-9 21-21V171h213v64c0 13 9 21 21 21z M170 470l0 84 86 0 0-84-86 0zM86 598l0-172 852 0 0 172-852 0zM256 298l0-84-86 0 0 84 86 0zM86 170l852 0 0 172-852 0 0-172zM170 726l0 84 86 0 0-84-86 0zM86 854l0-172 852 0 0 172-852 0z M853.3 960H170.7V64h469.3l213.3 213.3zM821.3 298.7H618.7V96z M192 0l0 1024 320-320 320 320 0-1024z M888.8 0H135.2c-32.3 0-58.9 26.1-58.9 58.9v906.2c0 32.3 26.1 58.9 58.9 58.9h753.2c32.3 0 58.9-26.1 58.9-58.9v-906.2c.5-32.8-26.1-58.9-58.4-58.9zm-164.9 176.6c30.7 0 55.8 25.1 55.8 55.8s-25.1 55.8-55.8 55.8s-55.8-25.1-55.8-55.8s24.6-55.8 55.8-55.8zm-212 0c30.7 0 55.8 25.1 55.8 55.8S542.7 288.3 512 288.3s-55.8-25.1-55.8-55.8S481.3 176.6 512 176.6zm-212 0c30.7 0 55.8 25.1 55.8 55.8s-25.1 55.8-55.8 55.8s-55.8-25.1-55.8-55.8s25.1-55.8 55.8-55.8zm208.9 606.2H285.2c-24.6 0-44-20-44-44c0-24.6 20-44 44-44h223.7c24.6 0 44 20 44 44c0 24.1-19.5 44-44 44zm229.9-212H285.2c-24.6 0-44-20-44-44c0-24.6 20-44 44-44h453.1c24.6 0 44 20 44 44c.5 24.1-19.5 44-43.5 44z M490.7 85.3l42.7 0 0 853.3-42.7 0 0-853.3zM85.3 490.7l853.3 0 0 42.7-853.3 0 0-42.7z M682.7 42.7H85.3v682.7h85.3V128h512V42.7zM256 213.3l4.5 768H896V213.3H256zm554.7 682.7H341.3V298.7h469.3v597.3z - + M204 291c45-11 77-49 77-96c0-53-43-98-98-98c-53 0-98 45-98 98c0 47 34 87 77 96v91c0 13 9 21 21 21h236c2 38 32 68 70 68h372c41 0 73-32 73-73v-38c0-41-32-73-73-73h-370c-38 0-70 30-70 68H204V291zm258 74h2c0-15 13-30 30-30h372c15 0 30 13 30 30v38c0 15-13 30-30 30h-375c-15 0-30-13-30-30v-38zM183 250c-30 0-55-26-55-55s26-55 55-55s55 26 55 55s-26 55-55 55zM679 495c-134 0-244 109-244 244s109 244 244 244c134 0 244-109 244-244s-109-244-244-244zm159 268h-134v134h-50V764H521v-50h134v-134h50v134h134V764zM244 766H185c-13 0-23-11-23-23s11-23 23-23h59c13 0 23 11 23 23s-11 23-23 23zM368 766h-42c-9 0-17-8-17-17v-13c0-9 8-17 17-17h42c9 0 17 8 17 17v13c0 9-8 17-17 17zM183 766c-12 0-21-9-21-21V320c0-12 9-21 21-21c12 0 21 9 21 21v425c0 12-10 21-21 21z + M256 811h512v85H256z M922 205v614H154V205h768zm-51 51H205v512h666V256z M899 203 821 125 512 434 203 125 125 203 434 512 125 821 203 899 512 590 821 899 899 821 590 512z diff --git a/src/Resources/Locales/en_US.xaml b/src/Resources/Locales/en_US.xaml index d1365c13..977b632f 100644 --- a/src/Resources/Locales/en_US.xaml +++ b/src/Resources/Locales/en_US.xaml @@ -128,6 +128,8 @@ SUBMODULES ADD SUBMODULE UPDATE SUBMODULE + SUBTREES + ADD/LINK SUBTREE RESOLVE CONTINUE ABORT @@ -425,7 +427,33 @@ NOTICE Restart required to apply changes in preference. Restart now? - + + Add/Link SubTree + Source URL : + Branch/Commit : + Local Relative Path : + Squash commits? + + Edit SubTree + Source URL : + Local Relative Path : + + Unlink SubTree + Local Relative Path : + This will only remove links. + + Pull Changes Of SubTree + Push Changes Of SubTree + Local Relative Path : + Remote : + Branch : + Squash commits? + + Edit ... + Unlink ... + Pull ... + Push ... + Git has NOT been configured.\nPlease to go [Preference] and configure it first. Path[{0}] not exists! Can NOT locate bash.exe. Make sure bash.exe exists under the same folder with git.exe @@ -448,6 +476,7 @@ Duplicated tag name! Commit subject can NOT be empty Invalid path for patch file - Invalid path for submodules + Invalid relative path Invalid path for archive file + This field is required \ No newline at end of file diff --git a/src/Resources/Locales/zh_CN.xaml b/src/Resources/Locales/zh_CN.xaml index 5f08e071..7218c9e4 100644 --- a/src/Resources/Locales/zh_CN.xaml +++ b/src/Resources/Locales/zh_CN.xaml @@ -128,6 +128,8 @@ 子模块列表 添加子模块 更新子模块 + 子树列表 + 添加子树 解决冲突 下一步 终止冲突解决 @@ -426,6 +428,32 @@ 系统提示 本次配置变更需要在重启后生效,是否立即重启? + 添加子树 + 远程地址: + 分支或提交ID: + 本地相对路径: + 合并提交为单一提交 + + 编辑子树信息 + 远程地址: + 本地相对路径: + + 删除子树 + 本地相对路径: + 本操作仅将子树信息删除,相关文件及提交不会更改 + + 拉取子树更新 + 推送子树更新到远程 + 本地相对路径: + 远程地址: + 远程分支: + 合并提交为单一提交 + + 编辑子树 ... + 删除子树 ... + 拉取子树更新 + 推送子树变更 + GIT尚未配置。请打开【偏好设置】配置GIT路径。 路径({0})不存在或不可读取! 无法找到bash.exe,请确保其在git.exe同目录中! @@ -448,6 +476,7 @@ 标签名已存在! 提交信息未填写! 补丁文件不存在或不可访问! - 非法的子模块路径! + 非法的子路径! 非法的存档文件路径! + 内容未填写! \ No newline at end of file diff --git a/src/Views/Popups/AddSubTree.xaml b/src/Views/Popups/AddSubTree.xaml new file mode 100644 index 00000000..e416860d --- /dev/null +++ b/src/Views/Popups/AddSubTree.xaml @@ -0,0 +1,86 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/Views/Popups/AddSubTree.xaml.cs b/src/Views/Popups/AddSubTree.xaml.cs new file mode 100644 index 00000000..0086a0a2 --- /dev/null +++ b/src/Views/Popups/AddSubTree.xaml.cs @@ -0,0 +1,57 @@ +using System.Threading.Tasks; +using System.Windows.Controls; + +namespace SourceGit.Views.Popups { + + /// + /// 添加子树面板 + /// + public partial class AddSubTree : Controls.PopupWidget { + private Models.Repository repo = null; + + public string Source { get; set; } + public string Ref { get; set; } + public string Prefix { get; set; } + + public AddSubTree(Models.Repository repo) { + this.repo = repo; + InitializeComponent(); + } + + public override string GetTitle() { + return App.Text("AddSubTree"); + } + + public override Task Start() { + txtSource.GetBindingExpression(TextBox.TextProperty).UpdateSource(); + if (Validation.GetHasError(txtSource)) return null; + + txtPrefix.GetBindingExpression(TextBox.TextProperty).UpdateSource(); + if (Validation.GetHasError(txtPrefix)) return null; + + txtRef.GetBindingExpression(TextBox.TextProperty).UpdateSource(); + if (Validation.GetHasError(txtRef)) return null; + + var squash = chkSquash.IsChecked == true; + if (repo.SubTrees.FindIndex(x => x.Prefix == Prefix) >= 0) { + Models.Exception.Raise($"Subtree add failed. Prefix({Prefix}) already exists!"); + return null; + } + + return Task.Run(() => { + Models.Watcher.SetEnabled(repo.Path, false); + var succ = new Commands.SubTree(repo.Path).Add(Prefix, Source, Ref, squash, UpdateProgress); + if (succ) { + repo.SubTrees.Add(new Models.SubTree() { + Prefix = Prefix, + Remote = Source, + }); + Models.Preference.Save(); + Models.Watcher.Get(repo.Path)?.RefreshSubTrees(); + } + Models.Watcher.SetEnabled(repo.Path, true); + return succ; + }); + } + } +} diff --git a/src/Views/Popups/AddSubmodule.xaml b/src/Views/Popups/AddSubmodule.xaml index 3b50799d..1b8233e7 100644 --- a/src/Views/Popups/AddSubmodule.xaml +++ b/src/Views/Popups/AddSubmodule.xaml @@ -53,7 +53,7 @@ - + diff --git a/src/Views/Popups/EditSubTree.xaml b/src/Views/Popups/EditSubTree.xaml new file mode 100644 index 00000000..5a5c034f --- /dev/null +++ b/src/Views/Popups/EditSubTree.xaml @@ -0,0 +1,54 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/Views/Popups/EditSubTree.xaml.cs b/src/Views/Popups/EditSubTree.xaml.cs new file mode 100644 index 00000000..32e2d538 --- /dev/null +++ b/src/Views/Popups/EditSubTree.xaml.cs @@ -0,0 +1,34 @@ +using System.Threading.Tasks; +using System.Windows.Controls; + +namespace SourceGit.Views.Popups { + /// + /// 编辑子树 + /// + public partial class EditSubTree : Controls.PopupWidget { + private Models.Repository repo; + private Models.SubTree subtree; + + public string Source { + get { return subtree.Remote; } + set { subtree.Remote = value; } + } + + public EditSubTree(Models.Repository repo, string prefix) { + this.repo = repo; + this.subtree = repo.SubTrees.Find(x => x.Prefix == prefix); + InitializeComponent(); + txtPrefix.Text = prefix; + } + + public override string GetTitle() { + return App.Text("EditSubTree"); + } + + public override Task Start() { + txtSource.GetBindingExpression(TextBox.TextProperty).UpdateSource(); + if (Validation.GetHasError(txtSource)) return null; + return Task.Run(() => true); + } + } +} diff --git a/src/Views/Popups/SubTreePull.xaml b/src/Views/Popups/SubTreePull.xaml new file mode 100644 index 00000000..18446afa --- /dev/null +++ b/src/Views/Popups/SubTreePull.xaml @@ -0,0 +1,76 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/Views/Popups/SubTreePull.xaml.cs b/src/Views/Popups/SubTreePull.xaml.cs new file mode 100644 index 00000000..52149368 --- /dev/null +++ b/src/Views/Popups/SubTreePull.xaml.cs @@ -0,0 +1,43 @@ +using System.Threading.Tasks; +using System.Windows.Controls; + +namespace SourceGit.Views.Popups { + /// + /// 拉取 + /// + public partial class SubTreePull : Controls.PopupWidget { + private string repo; + private Models.SubTree subtree; + + public string Branch { + get { return subtree.Branch; } + set { subtree.Branch = value; } + } + + public SubTreePull(string repo, Models.SubTree subtree) { + this.repo = repo; + this.subtree = subtree; + InitializeComponent(); + txtPrefix.Text = subtree.Prefix; + txtSource.Text = subtree.Remote; + } + + public override string GetTitle() { + return App.Text("SubTreePullOrPush.Pull"); + } + + public override Task Start() { + txtBranch.GetBindingExpression(TextBox.TextProperty).UpdateSource(); + if (Validation.GetHasError(txtBranch)) return null; + + var squash = chkSquash.IsChecked == true; + + return Task.Run(() => { + Models.Watcher.SetEnabled(repo, false); + new Commands.SubTree(repo).Pull(subtree.Prefix, subtree.Remote, Branch, squash, UpdateProgress); + Models.Watcher.SetEnabled(repo, true); + return true; + }); + } + } +} diff --git a/src/Views/Popups/SubTreePush.xaml b/src/Views/Popups/SubTreePush.xaml new file mode 100644 index 00000000..f7036261 --- /dev/null +++ b/src/Views/Popups/SubTreePush.xaml @@ -0,0 +1,68 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/Views/Popups/SubTreePush.xaml.cs b/src/Views/Popups/SubTreePush.xaml.cs new file mode 100644 index 00000000..7cef2871 --- /dev/null +++ b/src/Views/Popups/SubTreePush.xaml.cs @@ -0,0 +1,41 @@ +using System.Threading.Tasks; +using System.Windows.Controls; + +namespace SourceGit.Views.Popups { + /// + /// 推送 + /// + public partial class SubTreePush : Controls.PopupWidget { + private string repo; + private Models.SubTree subtree; + + public string Branch { + get { return subtree.Branch; } + set { subtree.Branch = value; } + } + + public SubTreePush(string repo, Models.SubTree subtree) { + this.repo = repo; + this.subtree = subtree; + InitializeComponent(); + txtPrefix.Text = subtree.Prefix; + txtSource.Text = subtree.Remote; + } + + public override string GetTitle() { + return App.Text("SubTreePullOrPush.Push"); + } + + public override Task Start() { + txtBranch.GetBindingExpression(TextBox.TextProperty).UpdateSource(); + if (Validation.GetHasError(txtBranch)) return null; + + return Task.Run(() => { + Models.Watcher.SetEnabled(repo, false); + new Commands.SubTree(repo).Push(subtree.Prefix, subtree.Remote, Branch, UpdateProgress); + Models.Watcher.SetEnabled(repo, true); + return true; + }); + } + } +} diff --git a/src/Views/Popups/UnlinkSubTree.xaml b/src/Views/Popups/UnlinkSubTree.xaml new file mode 100644 index 00000000..4468956f --- /dev/null +++ b/src/Views/Popups/UnlinkSubTree.xaml @@ -0,0 +1,40 @@ + + + + + + + + + + + + + + + + + + + + + diff --git a/src/Views/Popups/UnlinkSubTree.xaml.cs b/src/Views/Popups/UnlinkSubTree.xaml.cs new file mode 100644 index 00000000..372bf305 --- /dev/null +++ b/src/Views/Popups/UnlinkSubTree.xaml.cs @@ -0,0 +1,34 @@ +using System.Threading.Tasks; + +namespace SourceGit.Views.Popups { + /// + /// 删除子树 + /// + public partial class UnlinkSubTree : Controls.PopupWidget { + private Models.Repository repo; + private string prefix; + + public UnlinkSubTree(Models.Repository repo, string prefix) { + this.repo = repo; + this.prefix = prefix; + InitializeComponent(); + txtPrefix.Text = prefix; + } + + public override string GetTitle() { + return App.Text("UnlinkSubTree"); + } + + public override Task Start() { + return Task.Run(() => { + var idx = repo.SubTrees.FindIndex(x => x.Prefix == prefix); + if (idx >= 0) { + repo.SubTrees.RemoveAt(idx); + Models.Preference.Save(); + Models.Watcher.Get(repo.Path)?.RefreshSubTrees(); + } + return true; + }); + } + } +} diff --git a/src/Views/Validations/SubmodulePath.cs b/src/Views/Validations/RelativePath.cs similarity index 81% rename from src/Views/Validations/SubmodulePath.cs rename to src/Views/Validations/RelativePath.cs index 880ae4ea..4f7496a2 100644 --- a/src/Views/Validations/SubmodulePath.cs +++ b/src/Views/Validations/RelativePath.cs @@ -1,17 +1,16 @@ using System.Globalization; -using System.IO; using System.Text.RegularExpressions; using System.Windows.Controls; namespace SourceGit.Views.Validations { - public class SubmodulePath : ValidationRule { + public class RelativePath : ValidationRule { public override ValidationResult Validate(object value, CultureInfo cultureInfo) { var path = value as string; if (string.IsNullOrEmpty(path)) return ValidationResult.ValidResult; var regex = new Regex(@"^[\w\-\._/]+$"); var succ = regex.IsMatch(path.Trim()); - return !succ ? new ValidationResult(false, App.Text("BadSubmodulePath")) : ValidationResult.ValidResult; + return !succ ? new ValidationResult(false, App.Text("BadRelativePath")) : ValidationResult.ValidResult; } } } \ No newline at end of file diff --git a/src/Views/Validations/Required.cs b/src/Views/Validations/Required.cs new file mode 100644 index 00000000..a0cdc076 --- /dev/null +++ b/src/Views/Validations/Required.cs @@ -0,0 +1,13 @@ +using System.Globalization; +using System.Windows.Controls; + +namespace SourceGit.Views.Validations { + public class Required : ValidationRule { + public override ValidationResult Validate(object value, CultureInfo cultureInfo) { + var path = value as string; + return string.IsNullOrEmpty(path) ? + new ValidationResult(false, App.Text("Required")) : + ValidationResult.ValidResult; + } + } +} diff --git a/src/Views/Widgets/Dashboard.xaml b/src/Views/Widgets/Dashboard.xaml index c5217d22..72d900ac 100644 --- a/src/Views/Widgets/Dashboard.xaml +++ b/src/Views/Widgets/Dashboard.xaml @@ -110,6 +110,8 @@ + + @@ -326,7 +328,7 @@ Grid.Row="7" x:Name="tagList" RowHeight="24" - Height="200" + MaxHeight="200" ScrollViewer.HorizontalScrollBarVisibility="Disabled" ScrollViewer.VerticalScrollBarVisibility="Auto" Visibility="{Binding ElementName=tglTags, Path=IsChecked, Converter={StaticResource BoolToCollapsed}}" @@ -386,7 +388,7 @@ Grid.Row="9" x:Name="submoduleList" RowHeight="24" - Height="100" + MaxHeight="100" ScrollViewer.HorizontalScrollBarVisibility="Disabled" ScrollViewer.VerticalScrollBarVisibility="Auto" Visibility="{Binding ElementName=tglSubmodules, Path=IsChecked, Converter={StaticResource BoolToCollapsed}}" @@ -411,6 +413,52 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/Views/Widgets/Dashboard.xaml.cs b/src/Views/Widgets/Dashboard.xaml.cs index 5a12e642..e2e20020 100644 --- a/src/Views/Widgets/Dashboard.xaml.cs +++ b/src/Views/Widgets/Dashboard.xaml.cs @@ -60,6 +60,7 @@ namespace SourceGit.Views.Widgets { UpdateStashes(); UpdateTags(); UpdateSubmodules(); + UpdateSubTrees(); var watcher = Models.Watcher.Get(repo.Path); watcher.Navigate += NavigateTo; @@ -68,6 +69,7 @@ namespace SourceGit.Views.Widgets { watcher.StashChanged += UpdateStashes; watcher.TagChanged += UpdateTags; watcher.SubmoduleChanged += UpdateSubmodules; + watcher.SubTreeChanged += UpdateSubTrees; Unloaded += (o, e) => { localBranches.Clear(); @@ -257,6 +259,14 @@ namespace SourceGit.Views.Widgets { }); }); } + + private void UpdateSubTrees() { + Dispatcher.Invoke(() => { + txtSubTreeCount.Text = $"({repo.SubTrees.Count})"; + subTreeList.ItemsSource = null; + subTreeList.ItemsSource = repo.SubTrees; + }); + } #endregion #region TOOLBAR_COMMANDS @@ -909,6 +919,55 @@ namespace SourceGit.Views.Widgets { } #endregion + #region SUBTREES + private void OpenAddSubTree(object sender, RoutedEventArgs e) { + new Popups.AddSubTree(repo).Show(); + e.Handled = true; + } + + private void OnSubTreeContextMenuOpening(object sender, ContextMenuEventArgs e) { + var subtree = subTreeList.SelectedItem as Models.SubTree; + if (subtree == null) return; + + var edit = new MenuItem(); + edit.Header = App.Text("SubTree.Edit"); + edit.Click += (o, ev) => { + new Popups.EditSubTree(repo, subtree.Prefix).Show(); + ev.Handled = true; + }; + + var unlink = new MenuItem(); + unlink.Header = App.Text("SubTree.Unlink"); + unlink.Click += (o, ev) => { + new Popups.UnlinkSubTree(repo, subtree.Prefix).Show(); + ev.Handled = true; + }; + + var pull = new MenuItem(); + pull.Header = App.Text("SubTree.Pull"); + pull.Click += (o, ev) => { + new Popups.SubTreePull(repo.Path, subtree).Show(); + ev.Handled = true; + }; + + var push = new MenuItem(); + push.Header = App.Text("SubTree.Push"); + push.Click += (o, ev) => { + new Popups.SubTreePush(repo.Path, subtree).Show(); + ev.Handled = true; + }; + + var menu = new ContextMenu(); + menu.Items.Add(edit); + menu.Items.Add(unlink); + menu.Items.Add(new Separator()); + menu.Items.Add(pull); + menu.Items.Add(push); + menu.IsOpen = true; + e.Handled = true; + } + #endregion + #region FILTERS private void OnFilterChanged(object sender, RoutedEventArgs e) { var toggle = sender as ToggleButton;