From: Max Kirillov <max@max630.net>
To: Paul Mackerras <paulus@samba.org>
Cc: git@vger.kernel.org
Subject: [PATCH 1/3] gitk: refactor: separate generic hunk parsing out of find_hunk_blamespecs{}
Date: Tue, 4 Feb 2014 00:41:52 +0200 [thread overview]
Message-ID: <20140203224152.GA14305@wheezy.local> (raw)
In-Reply-To: <20140203223346.GA14202@wheezy.local>
For requesting a region blame, it is necessary to parse a hunk and
find the region in the parent file corresponding to the selected region.
There is already hunk parsin functionality in the find_hunk_blamespec{},
but returns only information for a single line.
The new function, resolve_hunk_lines{}, scans the hunk once and returns
for all hunk lines between $start_diffline and $end_diffline, in which parent
each of them exists and which is its number there.
Signed-off-by: Max Kirillov <max@max630.net>
---
gitk | 93 ++++++++++++++++++++++++++++++++++++++++++--------------------------
1 file changed, 57 insertions(+), 36 deletions(-)
diff --git a/gitk b/gitk
index dfac4fd..7699a66 100755
--- a/gitk
+++ b/gitk
@@ -3590,11 +3590,11 @@ proc external_diff {} {
}
}
-proc find_hunk_blamespec {base line} {
+proc resolve_hunk_lines {base start_diffline end_diffline} {
global ctext
# Find and parse the hunk header
- set s_lix [$ctext search -backwards -regexp ^@@ "$line.0 lineend" $base.0]
+ set s_lix [$ctext search -backwards -regexp ^@@ "$start_diffline.0 lineend" $base.0]
if {$s_lix eq {}} return
set s_line [$ctext get $s_lix "$s_lix + 1 lines"]
@@ -3614,49 +3614,70 @@ proc find_hunk_blamespec {base line} {
}
# Now scan the lines to determine offset within the hunk
- set max_parent [expr {[llength $base_lines]-2}]
- set dline 0
+ set max_parent [expr {[llength $base_lines]-1}]
set s_lno [lindex [split $s_lix "."] 0]
- # Determine if the line is removed
- set chunk [$ctext get $line.0 "$line.1 + $max_parent chars"]
- if {[string match {[-+ ]*} $chunk]} {
- set removed_idx [string first "-" $chunk]
- # Choose a parent index
- if {$removed_idx >= 0} {
- set parent $removed_idx
+ set commitlines_by_diffline {}
+ array unset commit_lines
+ for {set p 0} {$p <= $max_parent} {incr p} {
+ set commit_lines($p) [expr [lindex $base_lines $p] - 1]
+ }
+ for {set diffline [expr $s_lno + 1]} {$diffline <= $end_diffline} {incr diffline} {
+ set chunk [$ctext get $diffline.0 "$diffline.0 + $max_parent chars"]
+ if {$chunk eq {} || [string match "\[\n@\]*" $chunk]} {
+ # region is larger than hunk
+ return {}
+ }
+ set is_removed [expr [string first "-" $chunk] >= 0]
+ if {!$is_removed} {
+ incr commit_lines(0)
+ set commitlines [list [list 0 $commit_lines(0)]]
} else {
- set unchanged_idx [string first " " $chunk]
- if {$unchanged_idx >= 0} {
- set parent $unchanged_idx
- } else {
- # blame the current commit
- set parent -1
- }
- }
- # then count other lines that belong to it
- for {set i $line} {[incr i -1] > $s_lno} {} {
- set chunk [$ctext get $i.0 "$i.1 + $max_parent chars"]
- # Determine if the line is removed
- set removed_idx [string first "-" $chunk]
- if {$parent >= 0} {
- set code [string index $chunk $parent]
- if {$code eq "-" || ($removed_idx < 0 && $code ne "+")} {
- incr dline
+ set commitlines {}
+ }
+ for {set p 1} {$p <= $max_parent} {incr p} {
+ switch -- [string index $chunk "$p-1"] {
+ "+" {
}
- } else {
- if {$removed_idx < 0} {
- incr dline
+ "-" {
+ incr commit_lines($p)
+ lappend commitlines [list $p $commit_lines($p)]
+ }
+ " " {
+ if {!$is_removed} {
+ incr commit_lines($p)
+ lappend commitlines [list $p $commit_lines($p)]
+ }
+ }
+ default {
+ error_popup "resolve_hunk_lines: unexpected diff line($diffline): $chunk"
+ break
}
}
}
- incr parent
- } else {
- set parent 0
+ if {$diffline >= $start_diffline} {
+ lappend commitlines_by_diffline [list $diffline $commitlines]
+ }
}
+ return $commitlines_by_diffline
+}
- incr dline [lindex $base_lines $parent]
- return [list $parent $dline]
+proc find_hunk_blamespec {base line} {
+ foreach cl_spec [resolve_hunk_lines $base $line $line] {
+ if {[lindex $cl_spec 0] == $line} {
+ set commitlines [lindex $cl_spec 1]
+ if {[llength $commitlines] > 0} {
+ if {[llength $commitlines] > 1 && [lindex $commitlines 0 0] eq 0} {
+ return [lindex $commitlines 1]
+ } else {
+ return [lindex $commitlines 0]
+ }
+ } else {
+ error_popup "find_hunk_blamespec: invalid commitlines: $commitlines"
+ }
+ }
+ }
+ return {}
}
proc external_blame_diff {} {
--
1.8.5.2.421.g4cdf8d0
next prev parent reply other threads:[~2014-02-03 22:42 UTC|newest]
Thread overview: 8+ messages / expand[flat|nested] mbox.gz Atom feed top
2014-02-03 20:53 [PATCH] gitk: use single blamestuff for all show_line_source{} calls Max Kirillov
2014-02-03 22:34 ` [PATCH 0/3] gitk: show latest change to region Max Kirillov
2014-02-03 22:41 ` Max Kirillov [this message]
2014-02-03 23:20 ` [PATCH 1/3] gitk: refactor: separate generic hunk parsing out of find_hunk_blamespecs{} Eric Sunshine
2014-06-24 18:27 ` Max Kirillov
2014-02-03 22:42 ` [PATCH 2/3] gitk: refactor: separate io from logic in the searching origin of line Max Kirillov
2014-02-03 22:42 ` [PATCH 3/3] gitk: pick selection for region blame Max Kirillov
2014-02-03 22:48 ` [PATCH 3/3 v2] gitk: show latest change to region Max Kirillov
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=20140203224152.GA14305@wheezy.local \
--to=max@max630.net \
--cc=git@vger.kernel.org \
--cc=paulus@samba.org \
/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).