diff --git a/Default.sublime-commands b/Default.sublime-commands index 916490e8..89471f96 100644 --- a/Default.sublime-commands +++ b/Default.sublime-commands @@ -261,6 +261,10 @@ "caption": "Git: Add Selected Hunk", "command": "git_add_selected_hunk" } + ,{ + "caption": "Git: Add Selected Hunk (Edit)", + "command": "git_add_selected_hunk", "args": { "edit_patch": "True" } + } ,{ "caption": "Git: Commit Selected Hunk", "command": "git_commit_selected_hunk" diff --git a/Main.sublime-menu b/Main.sublime-menu index cb8d1f96..0cf84578 100644 --- a/Main.sublime-menu +++ b/Main.sublime-menu @@ -20,6 +20,7 @@ ,{ "caption": "-" } ,{ "caption": "Add", "command": "git_raw", "args": { "command": "git add", "append_current_file": true } } ,{ "caption": "Add Selected Hunk", "command": "git_add_selected_hunk" } + ,{ "caption": "Add Selected Hunk (Edit)", "command": "git_add_selected_hunk", "args": { "edit_patch": "True" } } ,{ "caption": "-" } ,{ "caption": "Move/Rename...", "command": "git_mv"} ,{ "caption": "Remove/Delete", "command": "git_raw", "args": { "command": "git rm", "append_current_file": true } } diff --git a/git/add.py b/git/add.py index 2e808732..599ad2db 100644 --- a/git/add.py +++ b/git/add.py @@ -6,6 +6,8 @@ import sublime from . import GitTextCommand, GitWindowCommand, git_root from .status import GitStatusCommand +from collections import namedtuple +from .diff import get_GitDiffRootInView class GitAddChoiceCommand(GitStatusCommand): @@ -45,10 +47,28 @@ def rerun(self, result): class GitAddSelectedHunkCommand(GitTextCommand): - def run(self, edit): - self.run_command(['git', 'diff', '--no-color', '-U1', self.get_file_name()], self.cull_diff) + def is_gitDiffView(self, view): + return view.name() == "Git Diff" and get_GitDiffRootInView(view) is not None - def cull_diff(self, result): + def is_enabled(self): + + view = self.active_view() + if self.is_gitDiffView(view): + return True + + # First, is this actually a file on the file system? + return super().is_enabled() + + def run(self, edit, edit_patch=False): + if self.is_gitDiffView(self.view): + kwargs = {} + kwargs['working_dir'] = get_GitDiffRootInView(self.view) + full_diff = self.view.substr(sublime.Region(0, self.view.size())) + self.cull_diff(full_diff, edit_patch=edit_patch, direct_select=True, **kwargs) + else: + self.run_command(['git', 'diff', '--no-color', '-U1', self.get_file_name()], lambda result: self.cull_diff(result, edit_patch)) + + def cull_diff(self, result, edit_patch=False, direct_select=False, **kwargs): selection = [] for sel in self.view.sel(): selection.append({ @@ -56,12 +76,17 @@ def cull_diff(self, result): "end": self.view.rowcol(sel.end())[0] + 1, }) - hunks = [{"diff": ""}] - i = 0 + # We devide the diff output into hunk groups. A file header starts a new group. + # Each group can contain zero or more hunks. + HunkGroup = namedtuple("HunkGroup", ["fileHeader", "hunkList"]) + section = [] + hunks = [HunkGroup(section, [])] # Initial lines before hunks matcher = re.compile('^@@ -([0-9]*)(?:,([0-9]*))? \+([0-9]*)(?:,([0-9]*))? @@') - for line in result.splitlines(): - if line.startswith('@@'): - i += 1 + for line_num, line in enumerate(result.splitlines(keepends=True)): # if different line endings, patch will not apply + if line.startswith('diff'): # new file + section = [] + hunks.append(HunkGroup(section, [])) + elif line.startswith('@@'): # new hunk match = matcher.match(line) start = int(match.group(3)) end = match.group(4) @@ -69,26 +94,58 @@ def cull_diff(self, result): end = start + int(end) else: end = start - hunks.append({"diff": "", "start": start, "end": end}) - hunks[i]["diff"] += line + "\n" + section = [] + hunks[-1].hunkList.append({"diff": section, "start": start, "end": end, "diff_start": line_num + 1, "diff_end": line_num + 1}) + elif hunks[-1].hunkList: # new line for file header or hunk + hunks[-1].hunkList[-1]["diff_end"] = line_num + 1 # update hunk end + + section.append(line) - diffs = hunks[0]["diff"] + diffs = "".join(hunks[0][0]) hunks.pop(0) selection_is_hunky = False - for hunk in hunks: - for sel in selection: - if sel["end"] < hunk["start"]: - continue - if sel["start"] > hunk["end"]: - continue - diffs += hunk["diff"] # + "\n\nEND OF HUNK\n\n" - selection_is_hunky = True + for file_header, hunkL in hunks: + file_header = "".join(file_header) + file_header_added = False + for hunk in hunkL: + for sel in selection: + # In direct mode the selected view lines correspond directly to the lines of the diff file + # In indirect mode the selected view lines correspond to the lines in the "@@" hunk header + if direct_select: + hunk_start = hunk["diff_start"] + hunk_end = hunk["diff_end"] + else: + hunk_start = hunk["start"] + hunk_end = hunk["end"] + if sel["end"] < hunk_start: + continue + if sel["start"] > hunk_end: + continue + # Only print the file header once + if not file_header_added: + file_header_added = True + diffs += file_header + hunk_str = "".join(hunk["diff"]) + diffs += hunk_str # + "\n\nEND OF HUNK\n\n" + selection_is_hunky = True if selection_is_hunky: - self.run_command(['git', 'apply', '--cached'], stdin=diffs) + if edit_patch: # open an input panel to modify the patch + patch_view = self.get_window().show_input_panel( + "Message", diffs, + lambda edited_patch: self.on_input(edited_patch, **kwargs), None, None + ) + s = sublime.load_settings("Git.sublime-settings") + syntax = s.get("diff_syntax", "Packages/Diff/Diff.tmLanguage") + patch_view.set_syntax_file(syntax) + patch_view.settings().set('word_wrap', False) + else: + self.on_input(diffs, **kwargs) else: sublime.status_message("No selected hunk") + def on_input(self, patch, **kwargs): + self.run_command(['git', 'apply', '--cached'], stdin=patch, **kwargs) # Also, sometimes we want to undo adds diff --git a/git/diff.py b/git/diff.py index af815c76..db8e5481 100644 --- a/git/diff.py +++ b/git/diff.py @@ -7,6 +7,22 @@ from . import GitTextCommand, GitWindowCommand, do_when, goto_xy, git_root, get_open_folder_from_window +gitDiffRootPrefix = "## git_diff_root: " + + +def get_GitDiffRootInView(view): + git_root_prefix = gitDiffRootPrefix + line_0 = view.substr(view.line(0)) + if git_root_prefix == line_0[:len(git_root_prefix)]: + git_root = line_0[len(git_root_prefix):].strip("\"") + return git_root + return None + + +def add_gitDiffRootToDiffOutput(output, gitRoot): + return gitDiffRootPrefix + "\"" + gitRoot + "\"\n" + output + + class GitDiff (object): def run(self, edit=None, ignore_whitespace=False): command = ['git', 'diff', '--no-color'] @@ -21,6 +37,8 @@ def diff_done(self, result): return s = sublime.load_settings("Git.sublime-settings") syntax = s.get("diff_syntax", "Packages/Diff/Diff.tmLanguage") + # We add meta-information from which we can infer the git_root after sublime restart + result = add_gitDiffRootToDiffOutput(result, git_root(self.get_working_dir())) if s.get('diff_panel'): self.panel(result, syntax=syntax) else: