|
|
@ -19,10 +19,10 @@ proc gitdir {} { |
|
|
|
proc parse_args {rargs} { |
|
|
|
proc parse_args {rargs} { |
|
|
|
global parsed_args |
|
|
|
global parsed_args |
|
|
|
|
|
|
|
|
|
|
|
if [catch { |
|
|
|
if {[catch { |
|
|
|
set parse_args [concat --default HEAD $rargs] |
|
|
|
set parse_args [concat --default HEAD $rargs] |
|
|
|
set parsed_args [split [eval exec git-rev-parse $parse_args] "\n"] |
|
|
|
set parsed_args [split [eval exec git-rev-parse $parse_args] "\n"] |
|
|
|
}] { |
|
|
|
}]} { |
|
|
|
# if git-rev-parse failed for some reason... |
|
|
|
# if git-rev-parse failed for some reason... |
|
|
|
if {$rargs == {}} { |
|
|
|
if {$rargs == {}} { |
|
|
|
set rargs HEAD |
|
|
|
set rargs HEAD |
|
|
@ -39,10 +39,10 @@ proc start_rev_list {rlargs} { |
|
|
|
set startmsecs [clock clicks -milliseconds] |
|
|
|
set startmsecs [clock clicks -milliseconds] |
|
|
|
set nextupdate [expr {$startmsecs + 100}] |
|
|
|
set nextupdate [expr {$startmsecs + 100}] |
|
|
|
set ncmupdate 1 |
|
|
|
set ncmupdate 1 |
|
|
|
if [catch { |
|
|
|
if {[catch { |
|
|
|
set commfd [open [concat | git-rev-list --header --topo-order \ |
|
|
|
set commfd [open [concat | git-rev-list --header --topo-order \ |
|
|
|
--parents $rlargs] r] |
|
|
|
--parents $rlargs] r] |
|
|
|
} err] { |
|
|
|
} err]} { |
|
|
|
puts stderr "Error executing git-rev-list: $err" |
|
|
|
puts stderr "Error executing git-rev-list: $err" |
|
|
|
exit 1 |
|
|
|
exit 1 |
|
|
|
} |
|
|
|
} |
|
|
@ -181,7 +181,7 @@ proc doupdate {reading} { |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
proc readcommit {id} { |
|
|
|
proc readcommit {id} { |
|
|
|
if [catch {set contents [exec git-cat-file commit $id]}] return |
|
|
|
if {[catch {set contents [exec git-cat-file commit $id]}]} return |
|
|
|
parsecommit $id $contents 0 {} |
|
|
|
parsecommit $id $contents 0 {} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
@ -542,8 +542,19 @@ proc makewindow {rargs} { |
|
|
|
$ctext tag conf m2 -fore green |
|
|
|
$ctext tag conf m2 -fore green |
|
|
|
$ctext tag conf m3 -fore purple |
|
|
|
$ctext tag conf m3 -fore purple |
|
|
|
$ctext tag conf m4 -fore brown |
|
|
|
$ctext tag conf m4 -fore brown |
|
|
|
|
|
|
|
$ctext tag conf m5 -fore "#009090" |
|
|
|
|
|
|
|
$ctext tag conf m6 -fore magenta |
|
|
|
|
|
|
|
$ctext tag conf m7 -fore "#808000" |
|
|
|
|
|
|
|
$ctext tag conf m8 -fore "#009000" |
|
|
|
|
|
|
|
$ctext tag conf m9 -fore "#ff0080" |
|
|
|
|
|
|
|
$ctext tag conf m10 -fore cyan |
|
|
|
|
|
|
|
$ctext tag conf m11 -fore "#b07070" |
|
|
|
|
|
|
|
$ctext tag conf m12 -fore "#70b0f0" |
|
|
|
|
|
|
|
$ctext tag conf m13 -fore "#70f0b0" |
|
|
|
|
|
|
|
$ctext tag conf m14 -fore "#f0b070" |
|
|
|
|
|
|
|
$ctext tag conf m15 -fore "#ff70b0" |
|
|
|
$ctext tag conf mmax -fore darkgrey |
|
|
|
$ctext tag conf mmax -fore darkgrey |
|
|
|
set mergemax 5 |
|
|
|
set mergemax 16 |
|
|
|
$ctext tag conf mresult -font [concat $textfont bold] |
|
|
|
$ctext tag conf mresult -font [concat $textfont bold] |
|
|
|
$ctext tag conf msep -font [concat $textfont bold] |
|
|
|
$ctext tag conf msep -font [concat $textfont bold] |
|
|
|
$ctext tag conf found -back yellow |
|
|
|
$ctext tag conf found -back yellow |
|
|
@ -679,7 +690,7 @@ proc savestuff {w} { |
|
|
|
|
|
|
|
|
|
|
|
proc resizeclistpanes {win w} { |
|
|
|
proc resizeclistpanes {win w} { |
|
|
|
global oldwidth |
|
|
|
global oldwidth |
|
|
|
if [info exists oldwidth($win)] { |
|
|
|
if {[info exists oldwidth($win)]} { |
|
|
|
set s0 [$win sash coord 0] |
|
|
|
set s0 [$win sash coord 0] |
|
|
|
set s1 [$win sash coord 1] |
|
|
|
set s1 [$win sash coord 1] |
|
|
|
if {$w < 60} { |
|
|
|
if {$w < 60} { |
|
|
@ -710,7 +721,7 @@ proc resizeclistpanes {win w} { |
|
|
|
|
|
|
|
|
|
|
|
proc resizecdetpanes {win w} { |
|
|
|
proc resizecdetpanes {win w} { |
|
|
|
global oldwidth |
|
|
|
global oldwidth |
|
|
|
if [info exists oldwidth($win)] { |
|
|
|
if {[info exists oldwidth($win)]} { |
|
|
|
set s0 [$win sash coord 0] |
|
|
|
set s0 [$win sash coord 0] |
|
|
|
if {$w < 60} { |
|
|
|
if {$w < 60} { |
|
|
|
set sash0 [expr {int($w*3/4 - 2)}] |
|
|
|
set sash0 [expr {int($w*3/4 - 2)}] |
|
|
@ -768,7 +779,7 @@ proc assigncolor {id} { |
|
|
|
global parents nparents children nchildren |
|
|
|
global parents nparents children nchildren |
|
|
|
global cornercrossings crossings |
|
|
|
global cornercrossings crossings |
|
|
|
|
|
|
|
|
|
|
|
if [info exists colormap($id)] return |
|
|
|
if {[info exists colormap($id)]} return |
|
|
|
set ncolors [llength $colors] |
|
|
|
set ncolors [llength $colors] |
|
|
|
if {$nparents($id) <= 1 && $nchildren($id) == 1} { |
|
|
|
if {$nparents($id) <= 1 && $nchildren($id) == 1} { |
|
|
|
set child [lindex $children($id) 0] |
|
|
|
set child [lindex $children($id) 0] |
|
|
@ -2182,6 +2193,7 @@ proc selectline {l isnew} { |
|
|
|
global canvy0 linespc parents nparents children |
|
|
|
global canvy0 linespc parents nparents children |
|
|
|
global cflist currentid sha1entry |
|
|
|
global cflist currentid sha1entry |
|
|
|
global commentend idtags idline linknum |
|
|
|
global commentend idtags idline linknum |
|
|
|
|
|
|
|
global mergemax |
|
|
|
|
|
|
|
|
|
|
|
$canv delete hover |
|
|
|
$canv delete hover |
|
|
|
normalline |
|
|
|
normalline |
|
|
@ -2265,11 +2277,26 @@ proc selectline {l isnew} { |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
set comment {} |
|
|
|
set comment {} |
|
|
|
if {[info exists parents($id)]} { |
|
|
|
if {$nparents($id) > 1} { |
|
|
|
|
|
|
|
set np 0 |
|
|
|
foreach p $parents($id) { |
|
|
|
foreach p $parents($id) { |
|
|
|
append comment "Parent: [commit_descriptor $p]\n" |
|
|
|
if {$np >= $mergemax} { |
|
|
|
|
|
|
|
set tag mmax |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
set tag m$np |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
$ctext insert end "Parent: " $tag |
|
|
|
|
|
|
|
appendwithlinks [commit_descriptor $p] |
|
|
|
|
|
|
|
incr np |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
if {[info exists parents($id)]} { |
|
|
|
|
|
|
|
foreach p $parents($id) { |
|
|
|
|
|
|
|
append comment "Parent: [commit_descriptor $p]\n" |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
if {[info exists children($id)]} { |
|
|
|
if {[info exists children($id)]} { |
|
|
|
foreach c $children($id) { |
|
|
|
foreach c $children($id) { |
|
|
|
append comment "Child: [commit_descriptor $c]\n" |
|
|
|
append comment "Child: [commit_descriptor $c]\n" |
|
|
@ -2361,529 +2388,100 @@ proc goforw {} { |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
proc mergediff {id} { |
|
|
|
proc mergediff {id} { |
|
|
|
global parents diffmergeid diffmergegca mergefilelist diffpindex |
|
|
|
global parents diffmergeid diffopts mdifffd |
|
|
|
|
|
|
|
global difffilestart |
|
|
|
|
|
|
|
|
|
|
|
set diffmergeid $id |
|
|
|
set diffmergeid $id |
|
|
|
set diffpindex -1 |
|
|
|
catch {unset difffilestart} |
|
|
|
set diffmergegca [findgca $parents($id)] |
|
|
|
# this doesn't seem to actually affect anything... |
|
|
|
if {[info exists mergefilelist($id)]} { |
|
|
|
|
|
|
|
if {$mergefilelist($id) ne {}} { |
|
|
|
|
|
|
|
showmergediff |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
contmergediff {} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
proc findgca {ids} { |
|
|
|
|
|
|
|
set gca {} |
|
|
|
|
|
|
|
foreach id $ids { |
|
|
|
|
|
|
|
if {$gca eq {}} { |
|
|
|
|
|
|
|
set gca $id |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
if {[catch { |
|
|
|
|
|
|
|
set gca [exec git-merge-base $gca $id] |
|
|
|
|
|
|
|
} err]} { |
|
|
|
|
|
|
|
return {} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
return $gca |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
proc contmergediff {ids} { |
|
|
|
|
|
|
|
global diffmergeid diffpindex parents nparents diffmergegca |
|
|
|
|
|
|
|
global treediffs mergefilelist diffids treepending |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# diff the child against each of the parents, and diff |
|
|
|
|
|
|
|
# each of the parents against the GCA. |
|
|
|
|
|
|
|
while 1 { |
|
|
|
|
|
|
|
if {[lindex $ids 1] == $diffmergeid && $diffmergegca ne {}} { |
|
|
|
|
|
|
|
set ids [list $diffmergegca [lindex $ids 0]] |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
if {[incr diffpindex] >= $nparents($diffmergeid)} break |
|
|
|
|
|
|
|
set p [lindex $parents($diffmergeid) $diffpindex] |
|
|
|
|
|
|
|
set ids [list $p $diffmergeid] |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
if {![info exists treediffs($ids)]} { |
|
|
|
|
|
|
|
set diffids $ids |
|
|
|
|
|
|
|
if {![info exists treepending]} { |
|
|
|
|
|
|
|
gettreediffs $ids |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
return |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# If a file in some parent is different from the child and also |
|
|
|
|
|
|
|
# different from the GCA, then it's interesting. |
|
|
|
|
|
|
|
# If we don't have a GCA, then a file is interesting if it is |
|
|
|
|
|
|
|
# different from the child in all the parents. |
|
|
|
|
|
|
|
if {$diffmergegca ne {}} { |
|
|
|
|
|
|
|
set files {} |
|
|
|
|
|
|
|
foreach p $parents($diffmergeid) { |
|
|
|
|
|
|
|
set gcadiffs $treediffs([list $diffmergegca $p]) |
|
|
|
|
|
|
|
foreach f $treediffs([list $p $diffmergeid]) { |
|
|
|
|
|
|
|
if {[lsearch -exact $files $f] < 0 |
|
|
|
|
|
|
|
&& [lsearch -exact $gcadiffs $f] >= 0} { |
|
|
|
|
|
|
|
lappend files $f |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
set files [lsort $files] |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
set p [lindex $parents($diffmergeid) 0] |
|
|
|
|
|
|
|
set files $treediffs([list $diffmergeid $p]) |
|
|
|
|
|
|
|
for {set i 1} {$i < $nparents($diffmergeid) && $files ne {}} {incr i} { |
|
|
|
|
|
|
|
set p [lindex $parents($diffmergeid) $i] |
|
|
|
|
|
|
|
set df $treediffs([list $p $diffmergeid]) |
|
|
|
|
|
|
|
set nf {} |
|
|
|
|
|
|
|
foreach f $files { |
|
|
|
|
|
|
|
if {[lsearch -exact $df $f] >= 0} { |
|
|
|
|
|
|
|
lappend nf $f |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
set files $nf |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
set mergefilelist($diffmergeid) $files |
|
|
|
|
|
|
|
if {$files ne {}} { |
|
|
|
|
|
|
|
showmergediff |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
proc showmergediff {} { |
|
|
|
|
|
|
|
global cflist diffmergeid mergefilelist parents |
|
|
|
|
|
|
|
global diffopts diffinhunk currentfile currenthunk filelines |
|
|
|
|
|
|
|
global diffblocked groupfilelast mergefds groupfilenum grouphunks |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
set files $mergefilelist($diffmergeid) |
|
|
|
|
|
|
|
foreach f $files { |
|
|
|
|
|
|
|
$cflist insert end $f |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
set env(GIT_DIFF_OPTS) $diffopts |
|
|
|
set env(GIT_DIFF_OPTS) $diffopts |
|
|
|
set flist {} |
|
|
|
set cmd [concat | git-diff-tree --no-commit-id --cc $id] |
|
|
|
catch {unset currentfile} |
|
|
|
if {[catch {set mdf [open $cmd r]} err]} { |
|
|
|
catch {unset currenthunk} |
|
|
|
error_popup "Error getting merge diffs: $err" |
|
|
|
catch {unset filelines} |
|
|
|
return |
|
|
|
catch {unset groupfilenum} |
|
|
|
|
|
|
|
catch {unset grouphunks} |
|
|
|
|
|
|
|
set groupfilelast -1 |
|
|
|
|
|
|
|
foreach p $parents($diffmergeid) { |
|
|
|
|
|
|
|
set cmd [list | git-diff-tree -p $p $diffmergeid] |
|
|
|
|
|
|
|
set cmd [concat $cmd $mergefilelist($diffmergeid)] |
|
|
|
|
|
|
|
if {[catch {set f [open $cmd r]} err]} { |
|
|
|
|
|
|
|
error_popup "Error getting diffs: $err" |
|
|
|
|
|
|
|
foreach f $flist { |
|
|
|
|
|
|
|
catch {close $f} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
return |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
lappend flist $f |
|
|
|
|
|
|
|
set ids [list $diffmergeid $p] |
|
|
|
|
|
|
|
set mergefds($ids) $f |
|
|
|
|
|
|
|
set diffinhunk($ids) 0 |
|
|
|
|
|
|
|
set diffblocked($ids) 0 |
|
|
|
|
|
|
|
fconfigure $f -blocking 0 |
|
|
|
|
|
|
|
fileevent $f readable [list getmergediffline $f $ids $diffmergeid] |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
fconfigure $mdf -blocking 0 |
|
|
|
|
|
|
|
set mdifffd($id) $mdf |
|
|
|
|
|
|
|
fileevent $mdf readable [list getmergediffline $mdf $id] |
|
|
|
|
|
|
|
set nextupdate [expr {[clock clicks -milliseconds] + 100}] |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
proc getmergediffline {f ids id} { |
|
|
|
proc getmergediffline {mdf id} { |
|
|
|
global diffmergeid diffinhunk diffoldlines diffnewlines |
|
|
|
global diffmergeid ctext cflist nextupdate nparents mergemax |
|
|
|
global currentfile currenthunk |
|
|
|
global difffilestart |
|
|
|
global diffoldstart diffnewstart diffoldlno diffnewlno |
|
|
|
|
|
|
|
global diffblocked mergefilelist |
|
|
|
|
|
|
|
global noldlines nnewlines difflcounts filelines |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
set n [gets $f line] |
|
|
|
set n [gets $mdf line] |
|
|
|
if {$n < 0} { |
|
|
|
if {$n < 0} { |
|
|
|
if {![eof $f]} return |
|
|
|
if {[eof $mdf]} { |
|
|
|
} |
|
|
|
close $mdf |
|
|
|
|
|
|
|
|
|
|
|
if {!([info exists diffmergeid] && $diffmergeid == $id)} { |
|
|
|
|
|
|
|
if {$n < 0} { |
|
|
|
|
|
|
|
close $f |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
return |
|
|
|
return |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
if {![info exists diffmergeid] || $id != $diffmergeid} { |
|
|
|
if {$diffinhunk($ids) != 0} { |
|
|
|
return |
|
|
|
set fi $currentfile($ids) |
|
|
|
|
|
|
|
if {$n > 0 && [regexp {^[-+ \\]} $line match]} { |
|
|
|
|
|
|
|
# continuing an existing hunk |
|
|
|
|
|
|
|
set line [string range $line 1 end] |
|
|
|
|
|
|
|
set p [lindex $ids 1] |
|
|
|
|
|
|
|
if {$match eq "-" || $match eq " "} { |
|
|
|
|
|
|
|
set filelines($p,$fi,$diffoldlno($ids)) $line |
|
|
|
|
|
|
|
incr diffoldlno($ids) |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
if {$match eq "+" || $match eq " "} { |
|
|
|
|
|
|
|
set filelines($id,$fi,$diffnewlno($ids)) $line |
|
|
|
|
|
|
|
incr diffnewlno($ids) |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
if {$match eq " "} { |
|
|
|
|
|
|
|
if {$diffinhunk($ids) == 2} { |
|
|
|
|
|
|
|
lappend difflcounts($ids) \ |
|
|
|
|
|
|
|
[list $noldlines($ids) $nnewlines($ids)] |
|
|
|
|
|
|
|
set noldlines($ids) 0 |
|
|
|
|
|
|
|
set diffinhunk($ids) 1 |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
incr noldlines($ids) |
|
|
|
|
|
|
|
} elseif {$match eq "-" || $match eq "+"} { |
|
|
|
|
|
|
|
if {$diffinhunk($ids) == 1} { |
|
|
|
|
|
|
|
lappend difflcounts($ids) [list $noldlines($ids)] |
|
|
|
|
|
|
|
set noldlines($ids) 0 |
|
|
|
|
|
|
|
set nnewlines($ids) 0 |
|
|
|
|
|
|
|
set diffinhunk($ids) 2 |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
if {$match eq "-"} { |
|
|
|
|
|
|
|
incr noldlines($ids) |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
incr nnewlines($ids) |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
# and if it's \ No newline at end of line, then what? |
|
|
|
|
|
|
|
return |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
# end of a hunk |
|
|
|
|
|
|
|
if {$diffinhunk($ids) == 1 && $noldlines($ids) != 0} { |
|
|
|
|
|
|
|
lappend difflcounts($ids) [list $noldlines($ids)] |
|
|
|
|
|
|
|
} elseif {$diffinhunk($ids) == 2 |
|
|
|
|
|
|
|
&& ($noldlines($ids) != 0 || $nnewlines($ids) != 0)} { |
|
|
|
|
|
|
|
lappend difflcounts($ids) [list $noldlines($ids) $nnewlines($ids)] |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
set currenthunk($ids) [list $currentfile($ids) \ |
|
|
|
|
|
|
|
$diffoldstart($ids) $diffnewstart($ids) \ |
|
|
|
|
|
|
|
$diffoldlno($ids) $diffnewlno($ids) \ |
|
|
|
|
|
|
|
$difflcounts($ids)] |
|
|
|
|
|
|
|
set diffinhunk($ids) 0 |
|
|
|
|
|
|
|
# -1 = need to block, 0 = unblocked, 1 = is blocked |
|
|
|
|
|
|
|
set diffblocked($ids) -1 |
|
|
|
|
|
|
|
processhunks |
|
|
|
|
|
|
|
if {$diffblocked($ids) == -1} { |
|
|
|
|
|
|
|
fileevent $f readable {} |
|
|
|
|
|
|
|
set diffblocked($ids) 1 |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if {$n < 0} { |
|
|
|
|
|
|
|
# eof |
|
|
|
|
|
|
|
if {!$diffblocked($ids)} { |
|
|
|
|
|
|
|
close $f |
|
|
|
|
|
|
|
set currentfile($ids) [llength $mergefilelist($diffmergeid)] |
|
|
|
|
|
|
|
set currenthunk($ids) [list $currentfile($ids) 0 0 0 0 {}] |
|
|
|
|
|
|
|
processhunks |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} elseif {[regexp {^diff --git a/(.*) b/} $line match fname]} { |
|
|
|
|
|
|
|
# start of a new file |
|
|
|
|
|
|
|
set currentfile($ids) \ |
|
|
|
|
|
|
|
[lsearch -exact $mergefilelist($diffmergeid) $fname] |
|
|
|
|
|
|
|
} elseif {[regexp {^@@ -([0-9]+),([0-9]+) \+([0-9]+),([0-9]+) @@(.*)} \ |
|
|
|
|
|
|
|
$line match f1l f1c f2l f2c rest]} { |
|
|
|
|
|
|
|
if {[info exists currentfile($ids)] && $currentfile($ids) >= 0} { |
|
|
|
|
|
|
|
# start of a new hunk |
|
|
|
|
|
|
|
if {$f1l == 0 && $f1c == 0} { |
|
|
|
|
|
|
|
set f1l 1 |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
if {$f2l == 0 && $f2c == 0} { |
|
|
|
|
|
|
|
set f2l 1 |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
set diffinhunk($ids) 1 |
|
|
|
|
|
|
|
set diffoldstart($ids) $f1l |
|
|
|
|
|
|
|
set diffnewstart($ids) $f2l |
|
|
|
|
|
|
|
set diffoldlno($ids) $f1l |
|
|
|
|
|
|
|
set diffnewlno($ids) $f2l |
|
|
|
|
|
|
|
set difflcounts($ids) {} |
|
|
|
|
|
|
|
set noldlines($ids) 0 |
|
|
|
|
|
|
|
set nnewlines($ids) 0 |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
proc processhunks {} { |
|
|
|
|
|
|
|
global diffmergeid parents nparents currenthunk |
|
|
|
|
|
|
|
global mergefilelist diffblocked mergefds |
|
|
|
|
|
|
|
global grouphunks grouplinestart grouplineend groupfilenum |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
set nfiles [llength $mergefilelist($diffmergeid)] |
|
|
|
|
|
|
|
while 1 { |
|
|
|
|
|
|
|
set fi $nfiles |
|
|
|
|
|
|
|
set lno 0 |
|
|
|
|
|
|
|
# look for the earliest hunk |
|
|
|
|
|
|
|
foreach p $parents($diffmergeid) { |
|
|
|
|
|
|
|
set ids [list $diffmergeid $p] |
|
|
|
|
|
|
|
if {![info exists currenthunk($ids)]} return |
|
|
|
|
|
|
|
set i [lindex $currenthunk($ids) 0] |
|
|
|
|
|
|
|
set l [lindex $currenthunk($ids) 2] |
|
|
|
|
|
|
|
if {$i < $fi || ($i == $fi && $l < $lno)} { |
|
|
|
|
|
|
|
set fi $i |
|
|
|
|
|
|
|
set lno $l |
|
|
|
|
|
|
|
set pi $p |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if {$fi < $nfiles} { |
|
|
|
|
|
|
|
set ids [list $diffmergeid $pi] |
|
|
|
|
|
|
|
set hunk $currenthunk($ids) |
|
|
|
|
|
|
|
unset currenthunk($ids) |
|
|
|
|
|
|
|
if {$diffblocked($ids) > 0} { |
|
|
|
|
|
|
|
fileevent $mergefds($ids) readable \ |
|
|
|
|
|
|
|
[list getmergediffline $mergefds($ids) $ids $diffmergeid] |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
set diffblocked($ids) 0 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if {[info exists groupfilenum] && $groupfilenum == $fi |
|
|
|
|
|
|
|
&& $lno <= $grouplineend} { |
|
|
|
|
|
|
|
# add this hunk to the pending group |
|
|
|
|
|
|
|
lappend grouphunks($pi) $hunk |
|
|
|
|
|
|
|
set endln [lindex $hunk 4] |
|
|
|
|
|
|
|
if {$endln > $grouplineend} { |
|
|
|
|
|
|
|
set grouplineend $endln |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
continue |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# succeeding stuff doesn't belong in this group, so |
|
|
|
|
|
|
|
# process the group now |
|
|
|
|
|
|
|
if {[info exists groupfilenum]} { |
|
|
|
|
|
|
|
processgroup |
|
|
|
|
|
|
|
unset groupfilenum |
|
|
|
|
|
|
|
unset grouphunks |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if {$fi >= $nfiles} break |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
# start a new group |
|
|
|
|
|
|
|
set groupfilenum $fi |
|
|
|
|
|
|
|
set grouphunks($pi) [list $hunk] |
|
|
|
|
|
|
|
set grouplinestart $lno |
|
|
|
|
|
|
|
set grouplineend [lindex $hunk 4] |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
proc processgroup {} { |
|
|
|
|
|
|
|
global groupfilelast groupfilenum difffilestart |
|
|
|
|
|
|
|
global mergefilelist diffmergeid ctext filelines |
|
|
|
|
|
|
|
global parents diffmergeid diffoffset |
|
|
|
|
|
|
|
global grouphunks grouplinestart grouplineend nparents |
|
|
|
|
|
|
|
global mergemax |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
$ctext conf -state normal |
|
|
|
$ctext conf -state normal |
|
|
|
set id $diffmergeid |
|
|
|
if {[regexp {^diff --cc (.*)} $line match fname]} { |
|
|
|
set f $groupfilenum |
|
|
|
# start of a new file |
|
|
|
if {$groupfilelast != $f} { |
|
|
|
|
|
|
|
$ctext insert end "\n" |
|
|
|
$ctext insert end "\n" |
|
|
|
set here [$ctext index "end - 1c"] |
|
|
|
set here [$ctext index "end - 1c"] |
|
|
|
set difffilestart($f) $here |
|
|
|
set i [$cflist index end] |
|
|
|
set mark fmark.[expr {$f + 1}] |
|
|
|
$ctext mark set fmark.$i $here |
|
|
|
$ctext mark set $mark $here |
|
|
|
$ctext mark gravity fmark.$i left |
|
|
|
$ctext mark gravity $mark left |
|
|
|
set difffilestart([expr {$i-1}]) $here |
|
|
|
set header [lindex $mergefilelist($id) $f] |
|
|
|
$cflist insert end $fname |
|
|
|
set l [expr {(78 - [string length $header]) / 2}] |
|
|
|
set l [expr {(78 - [string length $fname]) / 2}] |
|
|
|
set pad [string range "----------------------------------------" 1 $l] |
|
|
|
set pad [string range "----------------------------------------" 1 $l] |
|
|
|
$ctext insert end "$pad $header $pad\n" filesep |
|
|
|
$ctext insert end "$pad $fname $pad\n" filesep |
|
|
|
set groupfilelast $f |
|
|
|
} elseif {[regexp {^@@} $line]} { |
|
|
|
foreach p $parents($id) { |
|
|
|
$ctext insert end "$line\n" hunksep |
|
|
|
set diffoffset($p) 0 |
|
|
|
} elseif {[regexp {^[0-9a-f]{40}$} $line] || [regexp {^index} $line]} { |
|
|
|
} |
|
|
|
# do nothing |
|
|
|
} |
|
|
|
} else { |
|
|
|
|
|
|
|
# parse the prefix - one ' ', '-' or '+' for each parent |
|
|
|
$ctext insert end "@@" msep |
|
|
|
set np $nparents($id) |
|
|
|
set nlines [expr {$grouplineend - $grouplinestart}] |
|
|
|
set spaces {} |
|
|
|
set events {} |
|
|
|
set minuses {} |
|
|
|
set pnum 0 |
|
|
|
set pluses {} |
|
|
|
foreach p $parents($id) { |
|
|
|
set isbad 0 |
|
|
|
set startline [expr {$grouplinestart + $diffoffset($p)}] |
|
|
|
for {set j 0} {$j < $np} {incr j} { |
|
|
|
set ol $startline |
|
|
|
set c [string range $line $j $j] |
|
|
|
set nl $grouplinestart |
|
|
|
if {$c == " "} { |
|
|
|
if {[info exists grouphunks($p)]} { |
|
|
|
lappend spaces $j |
|
|
|
foreach h $grouphunks($p) { |
|
|
|
} elseif {$c == "-"} { |
|
|
|
set l [lindex $h 2] |
|
|
|
lappend minuses $j |
|
|
|
if {$nl < $l} { |
|
|
|
} elseif {$c == "+"} { |
|
|
|
for {} {$nl < $l} {incr nl} { |
|
|
|
lappend pluses $j |
|
|
|
set filelines($p,$f,$ol) $filelines($id,$f,$nl) |
|
|
|
|
|
|
|
incr ol |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
foreach chunk [lindex $h 5] { |
|
|
|
|
|
|
|
if {[llength $chunk] == 2} { |
|
|
|
|
|
|
|
set olc [lindex $chunk 0] |
|
|
|
|
|
|
|
set nlc [lindex $chunk 1] |
|
|
|
|
|
|
|
set nnl [expr {$nl + $nlc}] |
|
|
|
|
|
|
|
lappend events [list $nl $nnl $pnum $olc $nlc] |
|
|
|
|
|
|
|
incr ol $olc |
|
|
|
|
|
|
|
set nl $nnl |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
incr ol [lindex $chunk 0] |
|
|
|
|
|
|
|
incr nl [lindex $chunk 0] |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
if {$nl < $grouplineend} { |
|
|
|
|
|
|
|
for {} {$nl < $grouplineend} {incr nl} { |
|
|
|
|
|
|
|
set filelines($p,$f,$ol) $filelines($id,$f,$nl) |
|
|
|
|
|
|
|
incr ol |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
set nlines [expr {$ol - $startline}] |
|
|
|
|
|
|
|
$ctext insert end " -$startline,$nlines" msep |
|
|
|
|
|
|
|
incr pnum |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
set nlines [expr {$grouplineend - $grouplinestart}] |
|
|
|
|
|
|
|
$ctext insert end " +$grouplinestart,$nlines @@\n" msep |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
set events [lsort -integer -index 0 $events] |
|
|
|
|
|
|
|
set nevents [llength $events] |
|
|
|
|
|
|
|
set nmerge $nparents($diffmergeid) |
|
|
|
|
|
|
|
set l $grouplinestart |
|
|
|
|
|
|
|
for {set i 0} {$i < $nevents} {set i $j} { |
|
|
|
|
|
|
|
set nl [lindex $events $i 0] |
|
|
|
|
|
|
|
while {$l < $nl} { |
|
|
|
|
|
|
|
$ctext insert end " $filelines($id,$f,$l)\n" |
|
|
|
|
|
|
|
incr l |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
set e [lindex $events $i] |
|
|
|
|
|
|
|
set enl [lindex $e 1] |
|
|
|
|
|
|
|
set j $i |
|
|
|
|
|
|
|
set active {} |
|
|
|
|
|
|
|
while 1 { |
|
|
|
|
|
|
|
set pnum [lindex $e 2] |
|
|
|
|
|
|
|
set olc [lindex $e 3] |
|
|
|
|
|
|
|
set nlc [lindex $e 4] |
|
|
|
|
|
|
|
if {![info exists delta($pnum)]} { |
|
|
|
|
|
|
|
set delta($pnum) [expr {$olc - $nlc}] |
|
|
|
|
|
|
|
lappend active $pnum |
|
|
|
|
|
|
|
} else { |
|
|
|
} else { |
|
|
|
incr delta($pnum) [expr {$olc - $nlc}] |
|
|
|
set isbad 1 |
|
|
|
} |
|
|
|
break |
|
|
|
if {[incr j] >= $nevents} break |
|
|
|
|
|
|
|
set e [lindex $events $j] |
|
|
|
|
|
|
|
if {[lindex $e 0] >= $enl} break |
|
|
|
|
|
|
|
if {[lindex $e 1] > $enl} { |
|
|
|
|
|
|
|
set enl [lindex $e 1] |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
set nlc [expr {$enl - $l}] |
|
|
|
|
|
|
|
set ncol mresult |
|
|
|
|
|
|
|
set bestpn -1 |
|
|
|
|
|
|
|
if {[llength $active] == $nmerge - 1} { |
|
|
|
|
|
|
|
# no diff for one of the parents, i.e. it's identical |
|
|
|
|
|
|
|
for {set pnum 0} {$pnum < $nmerge} {incr pnum} { |
|
|
|
|
|
|
|
if {![info exists delta($pnum)]} { |
|
|
|
|
|
|
|
if {$pnum < $mergemax} { |
|
|
|
|
|
|
|
lappend ncol m$pnum |
|
|
|
|
|
|
|
} else { |
|
|
|
|
|
|
|
lappend ncol mmax |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
break |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} elseif {[llength $active] == $nmerge} { |
|
|
|
|
|
|
|
# all parents are different, see if one is very similar |
|
|
|
|
|
|
|
set bestsim 30 |
|
|
|
|
|
|
|
for {set pnum 0} {$pnum < $nmerge} {incr pnum} { |
|
|
|
|
|
|
|
set sim [similarity $pnum $l $nlc $f \ |
|
|
|
|
|
|
|
[lrange $events $i [expr {$j-1}]]] |
|
|
|
|
|
|
|
if {$sim > $bestsim} { |
|
|
|
|
|
|
|
set bestsim $sim |
|
|
|
|
|
|
|
set bestpn $pnum |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
if {$bestpn >= 0} { |
|
|
|
|
|
|
|
lappend ncol m$bestpn |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
set pnum -1 |
|
|
|
set tags {} |
|
|
|
foreach p $parents($id) { |
|
|
|
set num {} |
|
|
|
incr pnum |
|
|
|
if {!$isbad && $minuses ne {} && $pluses eq {}} { |
|
|
|
if {![info exists delta($pnum)] || $pnum == $bestpn} continue |
|
|
|
# line doesn't appear in result, parents in $minuses have the line |
|
|
|
set olc [expr {$nlc + $delta($pnum)}] |
|
|
|
set num [lindex $minuses 0] |
|
|
|
set ol [expr {$l + $diffoffset($p)}] |
|
|
|
} elseif {!$isbad && $pluses ne {} && $minuses eq {}} { |
|
|
|
incr diffoffset($p) $delta($pnum) |
|
|
|
# line appears in result, parents in $pluses don't have the line |
|
|
|
unset delta($pnum) |
|
|
|
lappend tags mresult |
|
|
|
for {} {$olc > 0} {incr olc -1} { |
|
|
|
set num [lindex $spaces 0] |
|
|
|
$ctext insert end "-$filelines($p,$f,$ol)\n" m$pnum |
|
|
|
|
|
|
|
incr ol |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
set endl [expr {$l + $nlc}] |
|
|
|
if {$num ne {}} { |
|
|
|
if {$bestpn >= 0} { |
|
|
|
if {$num >= $mergemax} { |
|
|
|
# show this pretty much as a normal diff |
|
|
|
set num "max" |
|
|
|
set p [lindex $parents($id) $bestpn] |
|
|
|
|
|
|
|
set ol [expr {$l + $diffoffset($p)}] |
|
|
|
|
|
|
|
incr diffoffset($p) $delta($bestpn) |
|
|
|
|
|
|
|
unset delta($bestpn) |
|
|
|
|
|
|
|
for {set k $i} {$k < $j} {incr k} { |
|
|
|
|
|
|
|
set e [lindex $events $k] |
|
|
|
|
|
|
|
if {[lindex $e 2] != $bestpn} continue |
|
|
|
|
|
|
|
set nl [lindex $e 0] |
|
|
|
|
|
|
|
set ol [expr {$ol + $nl - $l}] |
|
|
|
|
|
|
|
for {} {$l < $nl} {incr l} { |
|
|
|
|
|
|
|
$ctext insert end "+$filelines($id,$f,$l)\n" $ncol |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
set c [lindex $e 3] |
|
|
|
|
|
|
|
for {} {$c > 0} {incr c -1} { |
|
|
|
|
|
|
|
$ctext insert end "-$filelines($p,$f,$ol)\n" m$bestpn |
|
|
|
|
|
|
|
incr ol |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
set nl [lindex $e 1] |
|
|
|
|
|
|
|
for {} {$l < $nl} {incr l} { |
|
|
|
|
|
|
|
$ctext insert end "+$filelines($id,$f,$l)\n" mresult |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
lappend tags m$num |
|
|
|
} |
|
|
|
} |
|
|
|
for {} {$l < $endl} {incr l} { |
|
|
|
$ctext insert end "$line\n" $tags |
|
|
|
$ctext insert end "+$filelines($id,$f,$l)\n" $ncol |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
while {$l < $grouplineend} { |
|
|
|
|
|
|
|
$ctext insert end " $filelines($id,$f,$l)\n" |
|
|
|
|
|
|
|
incr l |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
$ctext conf -state disabled |
|
|
|
$ctext conf -state disabled |
|
|
|
} |
|
|
|
if {[clock clicks -milliseconds] >= $nextupdate} { |
|
|
|
|
|
|
|
incr nextupdate 100 |
|
|
|
proc similarity {pnum l nlc f events} { |
|
|
|
fileevent $mdf readable {} |
|
|
|
global diffmergeid parents diffoffset filelines |
|
|
|
update |
|
|
|
|
|
|
|
fileevent $mdf readable [list getmergediffline $mdf $id] |
|
|
|
set id $diffmergeid |
|
|
|
|
|
|
|
set p [lindex $parents($id) $pnum] |
|
|
|
|
|
|
|
set ol [expr {$l + $diffoffset($p)}] |
|
|
|
|
|
|
|
set endl [expr {$l + $nlc}] |
|
|
|
|
|
|
|
set same 0 |
|
|
|
|
|
|
|
set diff 0 |
|
|
|
|
|
|
|
foreach e $events { |
|
|
|
|
|
|
|
if {[lindex $e 2] != $pnum} continue |
|
|
|
|
|
|
|
set nl [lindex $e 0] |
|
|
|
|
|
|
|
set ol [expr {$ol + $nl - $l}] |
|
|
|
|
|
|
|
for {} {$l < $nl} {incr l} { |
|
|
|
|
|
|
|
incr same [string length $filelines($id,$f,$l)] |
|
|
|
|
|
|
|
incr same |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
set oc [lindex $e 3] |
|
|
|
|
|
|
|
for {} {$oc > 0} {incr oc -1} { |
|
|
|
|
|
|
|
incr diff [string length $filelines($p,$f,$ol)] |
|
|
|
|
|
|
|
incr diff |
|
|
|
|
|
|
|
incr ol |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
set nl [lindex $e 1] |
|
|
|
|
|
|
|
for {} {$l < $nl} {incr l} { |
|
|
|
|
|
|
|
incr diff [string length $filelines($id,$f,$l)] |
|
|
|
|
|
|
|
incr diff |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
for {} {$l < $endl} {incr l} { |
|
|
|
|
|
|
|
incr same [string length $filelines($id,$f,$l)] |
|
|
|
|
|
|
|
incr same |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
if {$same == 0} { |
|
|
|
|
|
|
|
return 0 |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
return [expr {200 * $same / (2 * $same + $diff)}] |
|
|
|
|
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
proc startdiff {ids} { |
|
|
|
proc startdiff {ids} { |
|
|
@ -2912,7 +2510,9 @@ proc gettreediffs {ids} { |
|
|
|
global treediff parents treepending |
|
|
|
global treediff parents treepending |
|
|
|
set treepending $ids |
|
|
|
set treepending $ids |
|
|
|
set treediff {} |
|
|
|
set treediff {} |
|
|
|
if [catch {set gdtf [open [concat | git-diff-tree --no-commit-id -r $ids] r]}] return |
|
|
|
if {[catch \ |
|
|
|
|
|
|
|
{set gdtf [open [concat | git-diff-tree --no-commit-id -r $ids] r]} \ |
|
|
|
|
|
|
|
]} return |
|
|
|
fconfigure $gdtf -blocking 0 |
|
|
|
fconfigure $gdtf -blocking 0 |
|
|
|
fileevent $gdtf readable [list gettreediffline $gdtf $ids] |
|
|
|
fileevent $gdtf readable [list gettreediffline $gdtf $ids] |
|
|
|
} |
|
|
|
} |
|
|
|