git.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
From: "ToBoMi via GitGitGadget" <gitgitgadget@gmail.com>
To: git@vger.kernel.org
Cc: Johannes Sixt <j6t@kdbg.org>,
	tobias.boesch@miele.com, ToBoMi <tobias.boesch@miele.com>,
	Tobias Boesch <tobias.boesch@miele.com>
Subject: [PATCH v5] gitk: add external diff file rename detection
Date: Tue, 10 Jun 2025 08:29:34 +0000	[thread overview]
Message-ID: <pull.1774.v5.git.1749544174590.gitgitgadget@gmail.com> (raw)
In-Reply-To: <pull.1774.v4.git.1745830037917.gitgitgadget@gmail.com>

From: Tobias Boesch <tobias.boesch@miele.com>

If a file is renamed between commits and an external diff is started
through gitk on the original or the renamed file name,
gitk is unable to open the renamed file in the external diff editor.
It fails to fetch the renamed file from git, because it fetches it
using its original path in contrast to using the renamed path of the
file.
Detect the rename and open the external diff with the original and
the renamed file instead of no file (fetch the renamed file path and
name from git) no matter if the original or the renamed file is
selected in gitk.
Since moved or renamed file are handled the same way do this also
for moved files.

Signed-off-by: Tobias Boesch <tobias.boesch@miele.com>
---
    gitk: add external diff file rename detection
    
    Changes since v1:
    
     * Commit message ident
     * Commit message line length
    
    Changes since v2:
    
     * Removed option for rename detection (Adding GUI options seems to be
       not desired - which is understandable)
     * Rebased on current master of git-for-windows
     * Renamed variables for a better understanding
     * Made rename detection also work when the renamed file is selected in
       gitk
    
    Changes since v3:
    
     * Changed message to use present tense, removed bullet points and
       described changes in imperative mood
    
    Changes sine v4:
    
     * Use a git command to gather the changed file paths rather than
       parsing the text from the diff window panel for efficiency and to
       avoid regex containing the filename as a variable.
     * Change != to ne in string comparison
     * removed extra set of parentheses around &&
     * shorter variable names

Published-As: https://github.com/gitgitgadget/git/releases/tag/pr-1774%2FToBoMi%2Fdetect_renamed_files_when_opening_diff-v5
Fetch-It-Via: git fetch https://github.com/gitgitgadget/git pr-1774/ToBoMi/detect_renamed_files_when_opening_diff-v5
Pull-Request: https://github.com/gitgitgadget/git/pull/1774

Range-diff vs v4:

 1:  948b94bef5c ! 1:  0d28f189dc3 gitk: add external diff file rename detection
     @@ gitk-git/gitk: proc external_diff_get_one_file {diffid filename diffdir} {
                      "revision $diffid"]
       }
       
     -+proc check_for_renames_in_diff {filepath} {
     -+    global ctext
     ++proc check_for_renames_in_diff {diffidfrom diffidto filepath} {
     ++    global nullid nullid2
      +
     -+    set renamed_filenames [list {}]
     -+    set filename [file tail $filepath]
     -+    set rename_from_text_identifier_length 12
     -+    set rename_to_text_identifier_length 10
     -+    set reg_expr_rename_from {^rename from (.*$filename)}
     -+    set reg_expr_rename_from [subst -nobackslashes -nocommands $reg_expr_rename_from]
     -+    set rename_from_text_index [$ctext search -elide -regexp -- $reg_expr_rename_from 0.0]
     -+    if { ($rename_from_text_index != {})} {
     -+        set reg_expr_rename_to {^rename to (.*)}
     -+        set rename_to_text_index [$ctext search -elide -regexp -- $reg_expr_rename_to $rename_from_text_index]
     -+        if { ($rename_from_text_index != {}) && ($rename_to_text_index != {}) } {
     -+            lappend renamed_filenames [$ctext get "$rename_from_text_index + $rename_from_text_identifier_length chars" "$rename_from_text_index lineend"]
     -+            lappend renamed_filenames [$ctext get "$rename_to_text_index + $rename_to_text_identifier_length chars" "$rename_to_text_index lineend"]
     -+        }
     -+        return $renamed_filenames
     ++    if {$diffidfrom eq $nullid} {
     ++        set rev [list $diffidto -R]
     ++    } elseif {$diffidfrom eq $nullid2} {
     ++        set rev [list $diffidto --cached -R]
     ++    } elseif {$diffidto eq $nullid} {
     ++        set rev [list $diffidfrom]
     ++    } elseif {$diffidto eq $nullid2} {
     ++        set rev [list $diffidfrom --cached]
     ++    } else {
     ++        set rev [list $diffidfrom..$diffidto]
      +    }
     -+    set reg_expr_rename_to {^rename to (.*$filename)}
     -+    set reg_expr_rename_to [subst -nobackslashes -nocommands $reg_expr_rename_to]
     -+    set rename_to_text_index [$ctext search -elide -regexp -- $reg_expr_rename_to 0.0]
     -+    if { ($rename_to_text_index != {})} {
     -+        set reg_expr_rename_from {^rename from (.*)}
     -+        set rename_from_text_index [$ctext search -backwards -elide -regexp -- $reg_expr_rename_from $rename_to_text_index]
     -+        if { ($rename_to_text_index != {}) && ($rename_from_text_index != {}) } {
     -+            lappend renamed_filenames [$ctext get "$rename_from_text_index + $rename_from_text_identifier_length chars" "$rename_from_text_index lineend"]
     -+            lappend renamed_filenames [$ctext get "$rename_to_text_index + $rename_to_text_identifier_length chars" "$rename_to_text_index lineend"]
     ++
     ++    set renames [list {}]
     ++    if {[catch {eval exec git diff $rev --find-renames --stat --raw --diff-filter=R} cmd_result]} {
     ++        error_popup "[mc "Error getting file rename info for file \"%s\" from commit %s to %s." \
     ++                            $filepath $diffidfrom $diffidto] $cmd_result.\n\n"
     ++    }
     ++    set filename [file tail $filepath]
     ++    set regex_ren {\d+\s\d+\s\S+\s\S+\s\S+\s+(\S+)\s+(\S+)}
     ++    set regex_ren [subst -nobackslashes -nocommands $regex_ren]
     ++    if {[regexp -line -- $regex_ren $cmd_result whole_match ren_from ren_to]} {
     ++        if {$ren_from ne {} && $ren_to ne {}} {
     ++            lappend renames $ren_from
     ++            lappend renames $ren_to
      +        }
     -+        return $renamed_filenames
      +    }
     ++    return $renames
      +}
      +
       proc external_diff {} {
     @@ gitk-git/gitk: proc external_diff {} {
           # gather files to diff
      -    set difffromfile [external_diff_get_one_file $diffidfrom $flist_menu_file $diffdir]
      -    set difftofile [external_diff_get_one_file $diffidto $flist_menu_file $diffdir]
     -+    set renamed_filenames [check_for_renames_in_diff $flist_menu_file]
     ++    set renamed_filenames [check_for_renames_in_diff $diffidfrom $diffidto $flist_menu_file]
      +    set rename_from_filename [lindex $renamed_filenames 1]
      +    set rename_to_filename [lindex $renamed_filenames 2]
      +    if { ($rename_from_filename != {}) && ($rename_to_filename != {}) } {


 gitk-git/gitk | 44 ++++++++++++++++++++++++++++++++++++++++++--
 1 file changed, 42 insertions(+), 2 deletions(-)

diff --git a/gitk-git/gitk b/gitk-git/gitk
index 19689765cde..f97904f5fa2 100755
--- a/gitk-git/gitk
+++ b/gitk-git/gitk
@@ -3775,6 +3775,38 @@ proc external_diff_get_one_file {diffid filename diffdir} {
                "revision $diffid"]
 }
 
+proc check_for_renames_in_diff {diffidfrom diffidto filepath} {
+    global nullid nullid2
+
+    if {$diffidfrom eq $nullid} {
+        set rev [list $diffidto -R]
+    } elseif {$diffidfrom eq $nullid2} {
+        set rev [list $diffidto --cached -R]
+    } elseif {$diffidto eq $nullid} {
+        set rev [list $diffidfrom]
+    } elseif {$diffidto eq $nullid2} {
+        set rev [list $diffidfrom --cached]
+    } else {
+        set rev [list $diffidfrom..$diffidto]
+    }
+
+    set renames [list {}]
+    if {[catch {eval exec git diff $rev --find-renames --stat --raw --diff-filter=R} cmd_result]} {
+        error_popup "[mc "Error getting file rename info for file \"%s\" from commit %s to %s." \
+                            $filepath $diffidfrom $diffidto] $cmd_result.\n\n"
+    }
+    set filename [file tail $filepath]
+    set regex_ren {\d+\s\d+\s\S+\s\S+\s\S+\s+(\S+)\s+(\S+)}
+    set regex_ren [subst -nobackslashes -nocommands $regex_ren]
+    if {[regexp -line -- $regex_ren $cmd_result whole_match ren_from ren_to]} {
+        if {$ren_from ne {} && $ren_to ne {}} {
+            lappend renames $ren_from
+            lappend renames $ren_to
+        }
+    }
+    return $renames
+}
+
 proc external_diff {} {
     global nullid nullid2
     global flist_menu_file
@@ -3805,8 +3837,16 @@ proc external_diff {} {
     if {$diffdir eq {}} return
 
     # gather files to diff
-    set difffromfile [external_diff_get_one_file $diffidfrom $flist_menu_file $diffdir]
-    set difftofile [external_diff_get_one_file $diffidto $flist_menu_file $diffdir]
+    set renamed_filenames [check_for_renames_in_diff $diffidfrom $diffidto $flist_menu_file]
+    set rename_from_filename [lindex $renamed_filenames 1]
+    set rename_to_filename [lindex $renamed_filenames 2]
+    if { ($rename_from_filename != {}) && ($rename_to_filename != {}) } {
+        set difffromfile [external_diff_get_one_file $diffidfrom $rename_from_filename $diffdir]
+        set difftofile [external_diff_get_one_file $diffidto $rename_to_filename $diffdir]
+    } else {
+        set difffromfile [external_diff_get_one_file $diffidfrom $flist_menu_file $diffdir]
+        set difftofile [external_diff_get_one_file $diffidto $flist_menu_file $diffdir]
+    }
 
     if {$difffromfile ne {} && $difftofile ne {}} {
         set cmd [list [shellsplit $extdifftool] $difffromfile $difftofile]

base-commit: 14de3eb34435db79c6e7edc8082c302a26a8330a
-- 
gitgitgadget

  parent reply	other threads:[~2025-06-10  8:29 UTC|newest]

Thread overview: 13+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2024-08-22  9:27 [PATCH] gitk: added external diff file rename detection ToBoMi via GitGitGadget
2024-09-06  7:28 ` [PATCH v2] " ToBoMi via GitGitGadget
2024-10-02  9:47   ` AW: " tobias.boesch
2025-03-04 13:01   ` [PATCH v3] " ToBoMi via GitGitGadget
2025-03-16 16:21     ` Johannes Sixt
2025-04-28  8:52       ` AW: " tobias.boesch
2025-04-28  8:47     ` [PATCH v4] gitk: add " ToBoMi via GitGitGadget
2025-05-06 19:39       ` Johannes Sixt
2025-06-10  8:28         ` AW: " tobias.boesch
2025-06-10  8:29       ` ToBoMi via GitGitGadget [this message]
2025-06-13  8:18         ` AW: [PATCH v5] " tobias.boesch
2025-06-24  9:05         ` [PATCH v6] " ToBoMi via GitGitGadget
2025-06-25  6:23           ` Johannes Sixt

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=pull.1774.v5.git.1749544174590.gitgitgadget@gmail.com \
    --to=gitgitgadget@gmail.com \
    --cc=git@vger.kernel.org \
    --cc=j6t@kdbg.org \
    --cc=tobias.boesch@miele.com \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).