1#!/bin/sh
2# Tcl ignores the next line -*- tcl -*- \
3exec wish "$0" -- "$@"
4
5# Copyright © 2005-2008 Paul Mackerras. All rights reserved.
6# This program is free software; it may be used, copied, modified
7# and distributed under the terms of the GNU General Public Licence,
8# either version 2, or (at your option) any later version.
9
10proc gitdir {} {
11 global env
12 if {[info exists env(GIT_DIR)]} {
13 return $env(GIT_DIR)
14 } else {
15 return [exec git rev-parse --git-dir]
16 }
17}
18
19# A simple scheduler for compute-intensive stuff.
20# The aim is to make sure that event handlers for GUI actions can
21# run at least every 50-100 ms. Unfortunately fileevent handlers are
22# run before X event handlers, so reading from a fast source can
23# make the GUI completely unresponsive.
24proc run args {
25 global isonrunq runq currunq
26
27 set script $args
28 if {[info exists isonrunq($script)]} return
29 if {$runq eq {} && ![info exists currunq]} {
30 after idle dorunq
31 }
32 lappend runq [list {} $script]
33 set isonrunq($script) 1
34}
35
36proc filerun {fd script} {
37 fileevent $fd readable [list filereadable $fd $script]
38}
39
40proc filereadable {fd script} {
41 global runq currunq
42
43 fileevent $fd readable {}
44 if {$runq eq {} && ![info exists currunq]} {
45 after idle dorunq
46 }
47 lappend runq [list $fd $script]
48}
49
50proc nukefile {fd} {
51 global runq
52
53 for {set i 0} {$i < [llength $runq]} {} {
54 if {[lindex $runq $i 0] eq $fd} {
55 set runq [lreplace $runq $i $i]
56 } else {
57 incr i
58 }
59 }
60}
61
62proc dorunq {} {
63 global isonrunq runq currunq
64
65 set tstart [clock clicks -milliseconds]
66 set t0 $tstart
67 while {[llength $runq] > 0} {
68 set fd [lindex $runq 0 0]
69 set script [lindex $runq 0 1]
70 set currunq [lindex $runq 0]
71 set runq [lrange $runq 1 end]
72 set repeat [eval $script]
73 unset currunq
74 set t1 [clock clicks -milliseconds]
75 set t [expr {$t1 - $t0}]
76 if {$repeat ne {} && $repeat} {
77 if {$fd eq {} || $repeat == 2} {
78 # script returns 1 if it wants to be readded
79 # file readers return 2 if they could do more straight away
80 lappend runq [list $fd $script]
81 } else {
82 fileevent $fd readable [list filereadable $fd $script]
83 }
84 } elseif {$fd eq {}} {
85 unset isonrunq($script)
86 }
87 set t0 $t1
88 if {$t1 - $tstart >= 80} break
89 }
90 if {$runq ne {}} {
91 after idle dorunq
92 }
93}
94
95proc reg_instance {fd} {
96 global commfd leftover loginstance
97
98 set i [incr loginstance]
99 set commfd($i) $fd
100 set leftover($i) {}
101 return $i
102}
103
104proc unmerged_files {files} {
105 global nr_unmerged
106
107 # find the list of unmerged files
108 set mlist {}
109 set nr_unmerged 0
110 if {[catch {
111 set fd [open "| git ls-files -u" r]
112 } err]} {
113 show_error {} . "[mc "Couldn't get list of unmerged files:"] $err"
114 exit 1
115 }
116 while {[gets $fd line] >= 0} {
117 set i [string first "\t" $line]
118 if {$i < 0} continue
119 set fname [string range $line [expr {$i+1}] end]
120 if {[lsearch -exact $mlist $fname] >= 0} continue
121 incr nr_unmerged
122 if {$files eq {} || [path_filter $files $fname]} {
123 lappend mlist $fname
124 }
125 }
126 catch {close $fd}
127 return $mlist
128}
129
130proc parseviewargs {n arglist} {
131 global vdatemode vmergeonly vflags vdflags vrevs vfiltered vorigargs
132
133 set vdatemode($n) 0
134 set vmergeonly($n) 0
135 set glflags {}
136 set diffargs {}
137 set nextisval 0
138 set revargs {}
139 set origargs $arglist
140 set allknown 1
141 set filtered 0
142 set i -1
143 foreach arg $arglist {
144 incr i
145 if {$nextisval} {
146 lappend glflags $arg
147 set nextisval 0
148 continue
149 }
150 switch -glob -- $arg {
151 "-d" -
152 "--date-order" {
153 set vdatemode($n) 1
154 # remove from origargs in case we hit an unknown option
155 set origargs [lreplace $origargs $i $i]
156 incr i -1
157 }
158 "-[puabwcrRBMC]" -
159 "--no-renames" - "--full-index" - "--binary" - "--abbrev=*" -
160 "--find-copies-harder" - "-l*" - "--ext-diff" - "--no-ext-diff" -
161 "--src-prefix=*" - "--dst-prefix=*" - "--no-prefix" -
162 "-O*" - "--text" - "--full-diff" - "--ignore-space-at-eol" -
163 "--ignore-space-change" - "-U*" - "--unified=*" {
164 # These request or affect diff output, which we don't want.
165 # Some could be used to set our defaults for diff display.
166 lappend diffargs $arg
167 }
168 "--raw" - "--patch-with-raw" - "--patch-with-stat" -
169 "--name-only" - "--name-status" - "--color" - "--color-words" -
170 "--log-size" - "--pretty=*" - "--decorate" - "--abbrev-commit" -
171 "--cc" - "-z" - "--header" - "--parents" - "--boundary" -
172 "--no-color" - "-g" - "--walk-reflogs" - "--no-walk" -
173 "--timestamp" - "relative-date" - "--date=*" - "--stdin" -
174 "--objects" - "--objects-edge" - "--reverse" {
175 # These cause our parsing of git log's output to fail, or else
176 # they're options we want to set ourselves, so ignore them.
177 }
178 "--stat=*" - "--numstat" - "--shortstat" - "--summary" -
179 "--check" - "--exit-code" - "--quiet" - "--topo-order" -
180 "--full-history" - "--dense" - "--sparse" -
181 "--follow" - "--left-right" - "--encoding=*" {
182 # These are harmless, and some are even useful
183 lappend glflags $arg
184 }
185 "--diff-filter=*" - "--no-merges" - "--unpacked" -
186 "--max-count=*" - "--skip=*" - "--since=*" - "--after=*" -
187 "--until=*" - "--before=*" - "--max-age=*" - "--min-age=*" -
188 "--author=*" - "--committer=*" - "--grep=*" - "-[iE]" -
189 "--remove-empty" - "--first-parent" - "--cherry-pick" -
190 "-S*" - "--pickaxe-all" - "--pickaxe-regex" {
191 # These mean that we get a subset of the commits
192 set filtered 1
193 lappend glflags $arg
194 }
195 "-n" {
196 # This appears to be the only one that has a value as a
197 # separate word following it
198 set filtered 1
199 set nextisval 1
200 lappend glflags $arg
201 }
202 "--not" {
203 set notflag [expr {!$notflag}]
204 lappend revargs $arg
205 }
206 "--all" {
207 lappend revargs $arg
208 }
209 "--merge" {
210 set vmergeonly($n) 1
211 # git rev-parse doesn't understand --merge
212 lappend revargs --gitk-symmetric-diff-marker MERGE_HEAD...HEAD
213 }
214 "-*" {
215 # Other flag arguments including -<n>
216 if {[string is digit -strict [string range $arg 1 end]]} {
217 set filtered 1
218 } else {
219 # a flag argument that we don't recognize;
220 # that means we can't optimize
221 set allknown 0
222 }
223 lappend glflags $arg
224 }
225 default {
226 # Non-flag arguments specify commits or ranges of commits
227 if {[string match "*...*" $arg]} {
228 lappend revargs --gitk-symmetric-diff-marker
229 }
230 lappend revargs $arg
231 }
232 }
233 }
234 set vdflags($n) $diffargs
235 set vflags($n) $glflags
236 set vrevs($n) $revargs
237 set vfiltered($n) $filtered
238 set vorigargs($n) $origargs
239 return $allknown
240}
241
242proc parseviewrevs {view revs} {
243 global vposids vnegids
244
245 if {$revs eq {}} {
246 set revs HEAD
247 }
248 if {[catch {set ids [eval exec git rev-parse $revs]} err]} {
249 # we get stdout followed by stderr in $err
250 # for an unknown rev, git rev-parse echoes it and then errors out
251 set errlines [split $err "\n"]
252 set badrev {}
253 for {set l 0} {$l < [llength $errlines]} {incr l} {
254 set line [lindex $errlines $l]
255 if {!([string length $line] == 40 && [string is xdigit $line])} {
256 if {[string match "fatal:*" $line]} {
257 if {[string match "fatal: ambiguous argument*" $line]
258 && $badrev ne {}} {
259 if {[llength $badrev] == 1} {
260 set err "unknown revision $badrev"
261 } else {
262 set err "unknown revisions: [join $badrev ", "]"
263 }
264 } else {
265 set err [join [lrange $errlines $l end] "\n"]
266 }
267 break
268 }
269 lappend badrev $line
270 }
271 }
272 error_popup "[mc "Error parsing revisions:"] $err"
273 return {}
274 }
275 set ret {}
276 set pos {}
277 set neg {}
278 set sdm 0
279 foreach id [split $ids "\n"] {
280 if {$id eq "--gitk-symmetric-diff-marker"} {
281 set sdm 4
282 } elseif {[string match "^*" $id]} {
283 if {$sdm != 1} {
284 lappend ret $id
285 if {$sdm == 3} {
286 set sdm 0
287 }
288 }
289 lappend neg [string range $id 1 end]
290 } else {
291 if {$sdm != 2} {
292 lappend ret $id
293 } else {
294 lset ret end [lindex $ret end]...$id
295 }
296 lappend pos $id
297 }
298 incr sdm -1
299 }
300 set vposids($view) $pos
301 set vnegids($view) $neg
302 return $ret
303}
304
305# Start off a git log process and arrange to read its output
306proc start_rev_list {view} {
307 global startmsecs commitidx viewcomplete curview
308 global tclencoding
309 global viewargs viewargscmd viewfiles vfilelimit
310 global showlocalchanges
311 global viewactive viewinstances vmergeonly
312 global mainheadid
313 global vcanopt vflags vrevs vorigargs
314
315 set startmsecs [clock clicks -milliseconds]
316 set commitidx($view) 0
317 # these are set this way for the error exits
318 set viewcomplete($view) 1
319 set viewactive($view) 0
320 varcinit $view
321
322 set args $viewargs($view)
323 if {$viewargscmd($view) ne {}} {
324 if {[catch {
325 set str [exec sh -c $viewargscmd($view)]
326 } err]} {
327 error_popup "[mc "Error executing --argscmd command:"] $err"
328 return 0
329 }
330 set args [concat $args [split $str "\n"]]
331 }
332 set vcanopt($view) [parseviewargs $view $args]
333
334 set files $viewfiles($view)
335 if {$vmergeonly($view)} {
336 set files [unmerged_files $files]
337 if {$files eq {}} {
338 global nr_unmerged
339 if {$nr_unmerged == 0} {
340 error_popup [mc "No files selected: --merge specified but\
341 no files are unmerged."]
342 } else {
343 error_popup [mc "No files selected: --merge specified but\
344 no unmerged files are within file limit."]
345 }
346 return 0
347 }
348 }
349 set vfilelimit($view) $files
350
351 if {$vcanopt($view)} {
352 set revs [parseviewrevs $view $vrevs($view)]
353 if {$revs eq {}} {
354 return 0
355 }
356 set args [concat $vflags($view) $revs]
357 } else {
358 set args $vorigargs($view)
359 }
360
361 if {[catch {
362 set fd [open [concat | git log --no-color -z --pretty=raw --parents \
363 --boundary $args "--" $files] r]
364 } err]} {
365 error_popup "[mc "Error executing git log:"] $err"
366 return 0
367 }
368 set i [reg_instance $fd]
369 set viewinstances($view) [list $i]
370 if {$showlocalchanges && $mainheadid ne {}} {
371 interestedin $mainheadid dodiffindex
372 }
373 fconfigure $fd -blocking 0 -translation lf -eofchar {}
374 if {$tclencoding != {}} {
375 fconfigure $fd -encoding $tclencoding
376 }
377 filerun $fd [list getcommitlines $fd $i $view 0]
378 nowbusy $view [mc "Reading"]
379 set viewcomplete($view) 0
380 set viewactive($view) 1
381 return 1
382}
383
384proc stop_instance {inst} {
385 global commfd leftover
386
387 set fd $commfd($inst)
388 catch {
389 set pid [pid $fd]
390
391 if {$::tcl_platform(platform) eq {windows}} {
392 exec kill -f $pid
393 } else {
394 exec kill $pid
395 }
396 }
397 catch {close $fd}
398 nukefile $fd
399 unset commfd($inst)
400 unset leftover($inst)
401}
402
403proc stop_backends {} {
404 global commfd
405
406 foreach inst [array names commfd] {
407 stop_instance $inst
408 }
409}
410
411proc stop_rev_list {view} {
412 global viewinstances
413
414 foreach inst $viewinstances($view) {
415 stop_instance $inst
416 }
417 set viewinstances($view) {}
418}
419
420proc reset_pending_select {selid} {
421 global pending_select mainheadid selectheadid
422
423 if {$selid ne {}} {
424 set pending_select $selid
425 } elseif {$selectheadid ne {}} {
426 set pending_select $selectheadid
427 } else {
428 set pending_select $mainheadid
429 }
430}
431
432proc getcommits {selid} {
433 global canv curview need_redisplay viewactive
434
435 initlayout
436 if {[start_rev_list $curview]} {
437 reset_pending_select $selid
438 show_status [mc "Reading commits..."]
439 set need_redisplay 1
440 } else {
441 show_status [mc "No commits selected"]
442 }
443}
444
445proc updatecommits {} {
446 global curview vcanopt vorigargs vfilelimit viewinstances
447 global viewactive viewcomplete tclencoding
448 global startmsecs showneartags showlocalchanges
449 global mainheadid pending_select
450 global isworktree
451 global varcid vposids vnegids vflags vrevs
452
453 set isworktree [expr {[exec git rev-parse --is-inside-work-tree] == "true"}]
454 set oldmainid $mainheadid
455 rereadrefs
456 if {$showlocalchanges} {
457 if {$mainheadid ne $oldmainid} {
458 dohidelocalchanges
459 }
460 if {[commitinview $mainheadid $curview]} {
461 dodiffindex
462 }
463 }
464 set view $curview
465 if {$vcanopt($view)} {
466 set oldpos $vposids($view)
467 set oldneg $vnegids($view)
468 set revs [parseviewrevs $view $vrevs($view)]
469 if {$revs eq {}} {
470 return
471 }
472 # note: getting the delta when negative refs change is hard,
473 # and could require multiple git log invocations, so in that
474 # case we ask git log for all the commits (not just the delta)
475 if {$oldneg eq $vnegids($view)} {
476 set newrevs {}
477 set npos 0
478 # take out positive refs that we asked for before or
479 # that we have already seen
480 foreach rev $revs {
481 if {[string length $rev] == 40} {
482 if {[lsearch -exact $oldpos $rev] < 0
483 && ![info exists varcid($view,$rev)]} {
484 lappend newrevs $rev
485 incr npos
486 }
487 } else {
488 lappend $newrevs $rev
489 }
490 }
491 if {$npos == 0} return
492 set revs $newrevs
493 set vposids($view) [lsort -unique [concat $oldpos $vposids($view)]]
494 }
495 set args [concat $vflags($view) $revs --not $oldpos]
496 } else {
497 set args $vorigargs($view)
498 }
499 if {[catch {
500 set fd [open [concat | git log --no-color -z --pretty=raw --parents \
501 --boundary $args "--" $vfilelimit($view)] r]
502 } err]} {
503 error_popup "[mc "Error executing git log:"] $err"
504 return
505 }
506 if {$viewactive($view) == 0} {
507 set startmsecs [clock clicks -milliseconds]
508 }
509 set i [reg_instance $fd]
510 lappend viewinstances($view) $i
511 fconfigure $fd -blocking 0 -translation lf -eofchar {}
512 if {$tclencoding != {}} {
513 fconfigure $fd -encoding $tclencoding
514 }
515 filerun $fd [list getcommitlines $fd $i $view 1]
516 incr viewactive($view)
517 set viewcomplete($view) 0
518 reset_pending_select {}
519 nowbusy $view "Reading"
520 if {$showneartags} {
521 getallcommits
522 }
523}
524
525proc reloadcommits {} {
526 global curview viewcomplete selectedline currentid thickerline
527 global showneartags treediffs commitinterest cached_commitrow
528 global targetid
529
530 set selid {}
531 if {$selectedline ne {}} {
532 set selid $currentid
533 }
534
535 if {!$viewcomplete($curview)} {
536 stop_rev_list $curview
537 }
538 resetvarcs $curview
539 set selectedline {}
540 catch {unset currentid}
541 catch {unset thickerline}
542 catch {unset treediffs}
543 readrefs
544 changedrefs
545 if {$showneartags} {
546 getallcommits
547 }
548 clear_display
549 catch {unset commitinterest}
550 catch {unset cached_commitrow}
551 catch {unset targetid}
552 setcanvscroll
553 getcommits $selid
554 return 0
555}
556
557# This makes a string representation of a positive integer which
558# sorts as a string in numerical order
559proc strrep {n} {
560 if {$n < 16} {
561 return [format "%x" $n]
562 } elseif {$n < 256} {
563 return [format "x%.2x" $n]
564 } elseif {$n < 65536} {
565 return [format "y%.4x" $n]
566 }
567 return [format "z%.8x" $n]
568}
569
570# Procedures used in reordering commits from git log (without
571# --topo-order) into the order for display.
572
573proc varcinit {view} {
574 global varcstart vupptr vdownptr vleftptr vbackptr varctok varcrow
575 global vtokmod varcmod vrowmod varcix vlastins
576
577 set varcstart($view) {{}}
578 set vupptr($view) {0}
579 set vdownptr($view) {0}
580 set vleftptr($view) {0}
581 set vbackptr($view) {0}
582 set varctok($view) {{}}
583 set varcrow($view) {{}}
584 set vtokmod($view) {}
585 set varcmod($view) 0
586 set vrowmod($view) 0
587 set varcix($view) {{}}
588 set vlastins($view) {0}
589}
590
591proc resetvarcs {view} {
592 global varcid varccommits parents children vseedcount ordertok
593
594 foreach vid [array names varcid $view,*] {
595 unset varcid($vid)
596 unset children($vid)
597 unset parents($vid)
598 }
599 # some commits might have children but haven't been seen yet
600 foreach vid [array names children $view,*] {
601 unset children($vid)
602 }
603 foreach va [array names varccommits $view,*] {
604 unset varccommits($va)
605 }
606 foreach vd [array names vseedcount $view,*] {
607 unset vseedcount($vd)
608 }
609 catch {unset ordertok}
610}
611
612# returns a list of the commits with no children
613proc seeds {v} {
614 global vdownptr vleftptr varcstart
615
616 set ret {}
617 set a [lindex $vdownptr($v) 0]
618 while {$a != 0} {
619 lappend ret [lindex $varcstart($v) $a]
620 set a [lindex $vleftptr($v) $a]
621 }
622 return $ret
623}
624
625proc newvarc {view id} {
626 global varcid varctok parents children vdatemode
627 global vupptr vdownptr vleftptr vbackptr varcrow varcix varcstart
628 global commitdata commitinfo vseedcount varccommits vlastins
629
630 set a [llength $varctok($view)]
631 set vid $view,$id
632 if {[llength $children($vid)] == 0 || $vdatemode($view)} {
633 if {![info exists commitinfo($id)]} {
634 parsecommit $id $commitdata($id) 1
635 }
636 set cdate [lindex $commitinfo($id) 4]
637 if {![string is integer -strict $cdate]} {
638 set cdate 0
639 }
640 if {![info exists vseedcount($view,$cdate)]} {
641 set vseedcount($view,$cdate) -1
642 }
643 set c [incr vseedcount($view,$cdate)]
644 set cdate [expr {$cdate ^ 0xffffffff}]
645 set tok "s[strrep $cdate][strrep $c]"
646 } else {
647 set tok {}
648 }
649 set ka 0
650 if {[llength $children($vid)] > 0} {
651 set kid [lindex $children($vid) end]
652 set k $varcid($view,$kid)
653 if {[string compare [lindex $varctok($view) $k] $tok] > 0} {
654 set ki $kid
655 set ka $k
656 set tok [lindex $varctok($view) $k]
657 }
658 }
659 if {$ka != 0} {
660 set i [lsearch -exact $parents($view,$ki) $id]
661 set j [expr {[llength $parents($view,$ki)] - 1 - $i}]
662 append tok [strrep $j]
663 }
664 set c [lindex $vlastins($view) $ka]
665 if {$c == 0 || [string compare $tok [lindex $varctok($view) $c]] < 0} {
666 set c $ka
667 set b [lindex $vdownptr($view) $ka]
668 } else {
669 set b [lindex $vleftptr($view) $c]
670 }
671 while {$b != 0 && [string compare $tok [lindex $varctok($view) $b]] >= 0} {
672 set c $b
673 set b [lindex $vleftptr($view) $c]
674 }
675 if {$c == $ka} {
676 lset vdownptr($view) $ka $a
677 lappend vbackptr($view) 0
678 } else {
679 lset vleftptr($view) $c $a
680 lappend vbackptr($view) $c
681 }
682 lset vlastins($view) $ka $a
683 lappend vupptr($view) $ka
684 lappend vleftptr($view) $b
685 if {$b != 0} {
686 lset vbackptr($view) $b $a
687 }
688 lappend varctok($view) $tok
689 lappend varcstart($view) $id
690 lappend vdownptr($view) 0
691 lappend varcrow($view) {}
692 lappend varcix($view) {}
693 set varccommits($view,$a) {}
694 lappend vlastins($view) 0
695 return $a
696}
697
698proc splitvarc {p v} {
699 global varcid varcstart varccommits varctok
700 global vupptr vdownptr vleftptr vbackptr varcix varcrow vlastins
701
702 set oa $varcid($v,$p)
703 set ac $varccommits($v,$oa)
704 set i [lsearch -exact $varccommits($v,$oa) $p]
705 if {$i <= 0} return
706 set na [llength $varctok($v)]
707 # "%" sorts before "0"...
708 set tok "[lindex $varctok($v) $oa]%[strrep $i]"
709 lappend varctok($v) $tok
710 lappend varcrow($v) {}
711 lappend varcix($v) {}
712 set varccommits($v,$oa) [lrange $ac 0 [expr {$i - 1}]]
713 set varccommits($v,$na) [lrange $ac $i end]
714 lappend varcstart($v) $p
715 foreach id $varccommits($v,$na) {
716 set varcid($v,$id) $na
717 }
718 lappend vdownptr($v) [lindex $vdownptr($v) $oa]
719 lappend vlastins($v) [lindex $vlastins($v) $oa]
720 lset vdownptr($v) $oa $na
721 lset vlastins($v) $oa 0
722 lappend vupptr($v) $oa
723 lappend vleftptr($v) 0
724 lappend vbackptr($v) 0
725 for {set b [lindex $vdownptr($v) $na]} {$b != 0} {set b [lindex $vleftptr($v) $b]} {
726 lset vupptr($v) $b $na
727 }
728}
729
730proc renumbervarc {a v} {
731 global parents children varctok varcstart varccommits
732 global vupptr vdownptr vleftptr vbackptr vlastins varcid vtokmod vdatemode
733
734 set t1 [clock clicks -milliseconds]
735 set todo {}
736 set isrelated($a) 1
737 set kidchanged($a) 1
738 set ntot 0
739 while {$a != 0} {
740 if {[info exists isrelated($a)]} {
741 lappend todo $a
742 set id [lindex $varccommits($v,$a) end]
743 foreach p $parents($v,$id) {
744 if {[info exists varcid($v,$p)]} {
745 set isrelated($varcid($v,$p)) 1
746 }
747 }
748 }
749 incr ntot
750 set b [lindex $vdownptr($v) $a]
751 if {$b == 0} {
752 while {$a != 0} {
753 set b [lindex $vleftptr($v) $a]
754 if {$b != 0} break
755 set a [lindex $vupptr($v) $a]
756 }
757 }
758 set a $b
759 }
760 foreach a $todo {
761 if {![info exists kidchanged($a)]} continue
762 set id [lindex $varcstart($v) $a]
763 if {[llength $children($v,$id)] > 1} {
764 set children($v,$id) [lsort -command [list vtokcmp $v] \
765 $children($v,$id)]
766 }
767 set oldtok [lindex $varctok($v) $a]
768 if {!$vdatemode($v)} {
769 set tok {}
770 } else {
771 set tok $oldtok
772 }
773 set ka 0
774 set kid [last_real_child $v,$id]
775 if {$kid ne {}} {
776 set k $varcid($v,$kid)
777 if {[string compare [lindex $varctok($v) $k] $tok] > 0} {
778 set ki $kid
779 set ka $k
780 set tok [lindex $varctok($v) $k]
781 }
782 }
783 if {$ka != 0} {
784 set i [lsearch -exact $parents($v,$ki) $id]
785 set j [expr {[llength $parents($v,$ki)] - 1 - $i}]
786 append tok [strrep $j]
787 }
788 if {$tok eq $oldtok} {
789 continue
790 }
791 set id [lindex $varccommits($v,$a) end]
792 foreach p $parents($v,$id) {
793 if {[info exists varcid($v,$p)]} {
794 set kidchanged($varcid($v,$p)) 1
795 } else {
796 set sortkids($p) 1
797 }
798 }
799 lset varctok($v) $a $tok
800 set b [lindex $vupptr($v) $a]
801 if {$b != $ka} {
802 if {[string compare [lindex $varctok($v) $ka] $vtokmod($v)] < 0} {
803 modify_arc $v $ka
804 }
805 if {[string compare [lindex $varctok($v) $b] $vtokmod($v)] < 0} {
806 modify_arc $v $b
807 }
808 set c [lindex $vbackptr($v) $a]
809 set d [lindex $vleftptr($v) $a]
810 if {$c == 0} {
811 lset vdownptr($v) $b $d
812 } else {
813 lset vleftptr($v) $c $d
814 }
815 if {$d != 0} {
816 lset vbackptr($v) $d $c
817 }
818 if {[lindex $vlastins($v) $b] == $a} {
819 lset vlastins($v) $b $c
820 }
821 lset vupptr($v) $a $ka
822 set c [lindex $vlastins($v) $ka]
823 if {$c == 0 || \
824 [string compare $tok [lindex $varctok($v) $c]] < 0} {
825 set c $ka
826 set b [lindex $vdownptr($v) $ka]
827 } else {
828 set b [lindex $vleftptr($v) $c]
829 }
830 while {$b != 0 && \
831 [string compare $tok [lindex $varctok($v) $b]] >= 0} {
832 set c $b
833 set b [lindex $vleftptr($v) $c]
834 }
835 if {$c == $ka} {
836 lset vdownptr($v) $ka $a
837 lset vbackptr($v) $a 0
838 } else {
839 lset vleftptr($v) $c $a
840 lset vbackptr($v) $a $c
841 }
842 lset vleftptr($v) $a $b
843 if {$b != 0} {
844 lset vbackptr($v) $b $a
845 }
846 lset vlastins($v) $ka $a
847 }
848 }
849 foreach id [array names sortkids] {
850 if {[llength $children($v,$id)] > 1} {
851 set children($v,$id) [lsort -command [list vtokcmp $v] \
852 $children($v,$id)]
853 }
854 }
855 set t2 [clock clicks -milliseconds]
856 #puts "renumbervarc did [llength $todo] of $ntot arcs in [expr {$t2-$t1}]ms"
857}
858
859# Fix up the graph after we have found out that in view $v,
860# $p (a commit that we have already seen) is actually the parent
861# of the last commit in arc $a.
862proc fix_reversal {p a v} {
863 global varcid varcstart varctok vupptr
864
865 set pa $varcid($v,$p)
866 if {$p ne [lindex $varcstart($v) $pa]} {
867 splitvarc $p $v
868 set pa $varcid($v,$p)
869 }
870 # seeds always need to be renumbered
871 if {[lindex $vupptr($v) $pa] == 0 ||
872 [string compare [lindex $varctok($v) $a] \
873 [lindex $varctok($v) $pa]] > 0} {
874 renumbervarc $pa $v
875 }
876}
877
878proc insertrow {id p v} {
879 global cmitlisted children parents varcid varctok vtokmod
880 global varccommits ordertok commitidx numcommits curview
881 global targetid targetrow
882
883 readcommit $id
884 set vid $v,$id
885 set cmitlisted($vid) 1
886 set children($vid) {}
887 set parents($vid) [list $p]
888 set a [newvarc $v $id]
889 set varcid($vid) $a
890 if {[string compare [lindex $varctok($v) $a] $vtokmod($v)] < 0} {
891 modify_arc $v $a
892 }
893 lappend varccommits($v,$a) $id
894 set vp $v,$p
895 if {[llength [lappend children($vp) $id]] > 1} {
896 set children($vp) [lsort -command [list vtokcmp $v] $children($vp)]
897 catch {unset ordertok}
898 }
899 fix_reversal $p $a $v
900 incr commitidx($v)
901 if {$v == $curview} {
902 set numcommits $commitidx($v)
903 setcanvscroll
904 if {[info exists targetid]} {
905 if {![comes_before $targetid $p]} {
906 incr targetrow
907 }
908 }
909 }
910}
911
912proc insertfakerow {id p} {
913 global varcid varccommits parents children cmitlisted
914 global commitidx varctok vtokmod targetid targetrow curview numcommits
915
916 set v $curview
917 set a $varcid($v,$p)
918 set i [lsearch -exact $varccommits($v,$a) $p]
919 if {$i < 0} {
920 puts "oops: insertfakerow can't find [shortids $p] on arc $a"
921 return
922 }
923 set children($v,$id) {}
924 set parents($v,$id) [list $p]
925 set varcid($v,$id) $a
926 lappend children($v,$p) $id
927 set cmitlisted($v,$id) 1
928 set numcommits [incr commitidx($v)]
929 # note we deliberately don't update varcstart($v) even if $i == 0
930 set varccommits($v,$a) [linsert $varccommits($v,$a) $i $id]
931 modify_arc $v $a $i
932 if {[info exists targetid]} {
933 if {![comes_before $targetid $p]} {
934 incr targetrow
935 }
936 }
937 setcanvscroll
938 drawvisible
939}
940
941proc removefakerow {id} {
942 global varcid varccommits parents children commitidx
943 global varctok vtokmod cmitlisted currentid selectedline
944 global targetid curview numcommits
945
946 set v $curview
947 if {[llength $parents($v,$id)] != 1} {
948 puts "oops: removefakerow [shortids $id] has [llength $parents($v,$id)] parents"
949 return
950 }
951 set p [lindex $parents($v,$id) 0]
952 set a $varcid($v,$id)
953 set i [lsearch -exact $varccommits($v,$a) $id]
954 if {$i < 0} {
955 puts "oops: removefakerow can't find [shortids $id] on arc $a"
956 return
957 }
958 unset varcid($v,$id)
959 set varccommits($v,$a) [lreplace $varccommits($v,$a) $i $i]
960 unset parents($v,$id)
961 unset children($v,$id)
962 unset cmitlisted($v,$id)
963 set numcommits [incr commitidx($v) -1]
964 set j [lsearch -exact $children($v,$p) $id]
965 if {$j >= 0} {
966 set children($v,$p) [lreplace $children($v,$p) $j $j]
967 }
968 modify_arc $v $a $i
969 if {[info exist currentid] && $id eq $currentid} {
970 unset currentid
971 set selectedline {}
972 }
973 if {[info exists targetid] && $targetid eq $id} {
974 set targetid $p
975 }
976 setcanvscroll
977 drawvisible
978}
979
980proc first_real_child {vp} {
981 global children nullid nullid2
982
983 foreach id $children($vp) {
984 if {$id ne $nullid && $id ne $nullid2} {
985 return $id
986 }
987 }
988 return {}
989}
990
991proc last_real_child {vp} {
992 global children nullid nullid2
993
994 set kids $children($vp)
995 for {set i [llength $kids]} {[incr i -1] >= 0} {} {
996 set id [lindex $kids $i]
997 if {$id ne $nullid && $id ne $nullid2} {
998 return $id
999 }
1000 }
1001 return {}
1002}
1003
1004proc vtokcmp {v a b} {
1005 global varctok varcid
1006
1007 return [string compare [lindex $varctok($v) $varcid($v,$a)] \
1008 [lindex $varctok($v) $varcid($v,$b)]]
1009}
1010
1011# This assumes that if lim is not given, the caller has checked that
1012# arc a's token is less than $vtokmod($v)
1013proc modify_arc {v a {lim {}}} {
1014 global varctok vtokmod varcmod varcrow vupptr curview vrowmod varccommits
1015
1016 if {$lim ne {}} {
1017 set c [string compare [lindex $varctok($v) $a] $vtokmod($v)]
1018 if {$c > 0} return
1019 if {$c == 0} {
1020 set r [lindex $varcrow($v) $a]
1021 if {$r ne {} && $vrowmod($v) <= $r + $lim} return
1022 }
1023 }
1024 set vtokmod($v) [lindex $varctok($v) $a]
1025 set varcmod($v) $a
1026 if {$v == $curview} {
1027 while {$a != 0 && [lindex $varcrow($v) $a] eq {}} {
1028 set a [lindex $vupptr($v) $a]
1029 set lim {}
1030 }
1031 set r 0
1032 if {$a != 0} {
1033 if {$lim eq {}} {
1034 set lim [llength $varccommits($v,$a)]
1035 }
1036 set r [expr {[lindex $varcrow($v) $a] + $lim}]
1037 }
1038 set vrowmod($v) $r
1039 undolayout $r
1040 }
1041}
1042
1043proc update_arcrows {v} {
1044 global vtokmod varcmod vrowmod varcrow commitidx currentid selectedline
1045 global varcid vrownum varcorder varcix varccommits
1046 global vupptr vdownptr vleftptr varctok
1047 global displayorder parentlist curview cached_commitrow
1048
1049 if {$vrowmod($v) == $commitidx($v)} return
1050 if {$v == $curview} {
1051 if {[llength $displayorder] > $vrowmod($v)} {
1052 set displayorder [lrange $displayorder 0 [expr {$vrowmod($v) - 1}]]
1053 set parentlist [lrange $parentlist 0 [expr {$vrowmod($v) - 1}]]
1054 }
1055 catch {unset cached_commitrow}
1056 }
1057 set narctot [expr {[llength $varctok($v)] - 1}]
1058 set a $varcmod($v)
1059 while {$a != 0 && [lindex $varcix($v) $a] eq {}} {
1060 # go up the tree until we find something that has a row number,
1061 # or we get to a seed
1062 set a [lindex $vupptr($v) $a]
1063 }
1064 if {$a == 0} {
1065 set a [lindex $vdownptr($v) 0]
1066 if {$a == 0} return
1067 set vrownum($v) {0}
1068 set varcorder($v) [list $a]
1069 lset varcix($v) $a 0
1070 lset varcrow($v) $a 0
1071 set arcn 0
1072 set row 0
1073 } else {
1074 set arcn [lindex $varcix($v) $a]
1075 if {[llength $vrownum($v)] > $arcn + 1} {
1076 set vrownum($v) [lrange $vrownum($v) 0 $arcn]
1077 set varcorder($v) [lrange $varcorder($v) 0 $arcn]
1078 }
1079 set row [lindex $varcrow($v) $a]
1080 }
1081 while {1} {
1082 set p $a
1083 incr row [llength $varccommits($v,$a)]
1084 # go down if possible
1085 set b [lindex $vdownptr($v) $a]
1086 if {$b == 0} {
1087 # if not, go left, or go up until we can go left
1088 while {$a != 0} {
1089 set b [lindex $vleftptr($v) $a]
1090 if {$b != 0} break
1091 set a [lindex $vupptr($v) $a]
1092 }
1093 if {$a == 0} break
1094 }
1095 set a $b
1096 incr arcn
1097 lappend vrownum($v) $row
1098 lappend varcorder($v) $a
1099 lset varcix($v) $a $arcn
1100 lset varcrow($v) $a $row
1101 }
1102 set vtokmod($v) [lindex $varctok($v) $p]
1103 set varcmod($v) $p
1104 set vrowmod($v) $row
1105 if {[info exists currentid]} {
1106 set selectedline [rowofcommit $currentid]
1107 }
1108}
1109
1110# Test whether view $v contains commit $id
1111proc commitinview {id v} {
1112 global varcid
1113
1114 return [info exists varcid($v,$id)]
1115}
1116
1117# Return the row number for commit $id in the current view
1118proc rowofcommit {id} {
1119 global varcid varccommits varcrow curview cached_commitrow
1120 global varctok vtokmod
1121
1122 set v $curview
1123 if {![info exists varcid($v,$id)]} {
1124 puts "oops rowofcommit no arc for [shortids $id]"
1125 return {}
1126 }
1127 set a $varcid($v,$id)
1128 if {[string compare [lindex $varctok($v) $a] $vtokmod($v)] >= 0} {
1129 update_arcrows $v
1130 }
1131 if {[info exists cached_commitrow($id)]} {
1132 return $cached_commitrow($id)
1133 }
1134 set i [lsearch -exact $varccommits($v,$a) $id]
1135 if {$i < 0} {
1136 puts "oops didn't find commit [shortids $id] in arc $a"
1137 return {}
1138 }
1139 incr i [lindex $varcrow($v) $a]
1140 set cached_commitrow($id) $i
1141 return $i
1142}
1143
1144# Returns 1 if a is on an earlier row than b, otherwise 0
1145proc comes_before {a b} {
1146 global varcid varctok curview
1147
1148 set v $curview
1149 if {$a eq $b || ![info exists varcid($v,$a)] || \
1150 ![info exists varcid($v,$b)]} {
1151 return 0
1152 }
1153 if {$varcid($v,$a) != $varcid($v,$b)} {
1154 return [expr {[string compare [lindex $varctok($v) $varcid($v,$a)] \
1155 [lindex $varctok($v) $varcid($v,$b)]] < 0}]
1156 }
1157 return [expr {[rowofcommit $a] < [rowofcommit $b]}]
1158}
1159
1160proc bsearch {l elt} {
1161 if {[llength $l] == 0 || $elt <= [lindex $l 0]} {
1162 return 0
1163 }
1164 set lo 0
1165 set hi [llength $l]
1166 while {$hi - $lo > 1} {
1167 set mid [expr {int(($lo + $hi) / 2)}]
1168 set t [lindex $l $mid]
1169 if {$elt < $t} {
1170 set hi $mid
1171 } elseif {$elt > $t} {
1172 set lo $mid
1173 } else {
1174 return $mid
1175 }
1176 }
1177 return $lo
1178}
1179
1180# Make sure rows $start..$end-1 are valid in displayorder and parentlist
1181proc make_disporder {start end} {
1182 global vrownum curview commitidx displayorder parentlist
1183 global varccommits varcorder parents vrowmod varcrow
1184 global d_valid_start d_valid_end
1185
1186 if {$end > $vrowmod($curview)} {
1187 update_arcrows $curview
1188 }
1189 set ai [bsearch $vrownum($curview) $start]
1190 set start [lindex $vrownum($curview) $ai]
1191 set narc [llength $vrownum($curview)]
1192 for {set r $start} {$ai < $narc && $r < $end} {incr ai} {
1193 set a [lindex $varcorder($curview) $ai]
1194 set l [llength $displayorder]
1195 set al [llength $varccommits($curview,$a)]
1196 if {$l < $r + $al} {
1197 if {$l < $r} {
1198 set pad [ntimes [expr {$r - $l}] {}]
1199 set displayorder [concat $displayorder $pad]
1200 set parentlist [concat $parentlist $pad]
1201 } elseif {$l > $r} {
1202 set displayorder [lrange $displayorder 0 [expr {$r - 1}]]
1203 set parentlist [lrange $parentlist 0 [expr {$r - 1}]]
1204 }
1205 foreach id $varccommits($curview,$a) {
1206 lappend displayorder $id
1207 lappend parentlist $parents($curview,$id)
1208 }
1209 } elseif {[lindex $displayorder [expr {$r + $al - 1}]] eq {}} {
1210 set i $r
1211 foreach id $varccommits($curview,$a) {
1212 lset displayorder $i $id
1213 lset parentlist $i $parents($curview,$id)
1214 incr i
1215 }
1216 }
1217 incr r $al
1218 }
1219}
1220
1221proc commitonrow {row} {
1222 global displayorder
1223
1224 set id [lindex $displayorder $row]
1225 if {$id eq {}} {
1226 make_disporder $row [expr {$row + 1}]
1227 set id [lindex $displayorder $row]
1228 }
1229 return $id
1230}
1231
1232proc closevarcs {v} {
1233 global varctok varccommits varcid parents children
1234 global cmitlisted commitidx vtokmod
1235
1236 set missing_parents 0
1237 set scripts {}
1238 set narcs [llength $varctok($v)]
1239 for {set a 1} {$a < $narcs} {incr a} {
1240 set id [lindex $varccommits($v,$a) end]
1241 foreach p $parents($v,$id) {
1242 if {[info exists varcid($v,$p)]} continue
1243 # add p as a new commit
1244 incr missing_parents
1245 set cmitlisted($v,$p) 0
1246 set parents($v,$p) {}
1247 if {[llength $children($v,$p)] == 1 &&
1248 [llength $parents($v,$id)] == 1} {
1249 set b $a
1250 } else {
1251 set b [newvarc $v $p]
1252 }
1253 set varcid($v,$p) $b
1254 if {[string compare [lindex $varctok($v) $b] $vtokmod($v)] < 0} {
1255 modify_arc $v $b
1256 }
1257 lappend varccommits($v,$b) $p
1258 incr commitidx($v)
1259 set scripts [check_interest $p $scripts]
1260 }
1261 }
1262 if {$missing_parents > 0} {
1263 foreach s $scripts {
1264 eval $s
1265 }
1266 }
1267}
1268
1269# Use $rwid as a substitute for $id, i.e. reparent $id's children to $rwid
1270# Assumes we already have an arc for $rwid.
1271proc rewrite_commit {v id rwid} {
1272 global children parents varcid varctok vtokmod varccommits
1273
1274 foreach ch $children($v,$id) {
1275 # make $rwid be $ch's parent in place of $id
1276 set i [lsearch -exact $parents($v,$ch) $id]
1277 if {$i < 0} {
1278 puts "oops rewrite_commit didn't find $id in parent list for $ch"
1279 }
1280 set parents($v,$ch) [lreplace $parents($v,$ch) $i $i $rwid]
1281 # add $ch to $rwid's children and sort the list if necessary
1282 if {[llength [lappend children($v,$rwid) $ch]] > 1} {
1283 set children($v,$rwid) [lsort -command [list vtokcmp $v] \
1284 $children($v,$rwid)]
1285 }
1286 # fix the graph after joining $id to $rwid
1287 set a $varcid($v,$ch)
1288 fix_reversal $rwid $a $v
1289 # parentlist is wrong for the last element of arc $a
1290 # even if displayorder is right, hence the 3rd arg here
1291 modify_arc $v $a [expr {[llength $varccommits($v,$a)] - 1}]
1292 }
1293}
1294
1295# Mechanism for registering a command to be executed when we come
1296# across a particular commit. To handle the case when only the
1297# prefix of the commit is known, the commitinterest array is now
1298# indexed by the first 4 characters of the ID. Each element is a
1299# list of id, cmd pairs.
1300proc interestedin {id cmd} {
1301 global commitinterest
1302
1303 lappend commitinterest([string range $id 0 3]) $id $cmd
1304}
1305
1306proc check_interest {id scripts} {
1307 global commitinterest
1308
1309 set prefix [string range $id 0 3]
1310 if {[info exists commitinterest($prefix)]} {
1311 set newlist {}
1312 foreach {i script} $commitinterest($prefix) {
1313 if {[string match "$i*" $id]} {
1314 lappend scripts [string map [list "%I" $id "%P" $i] $script]
1315 } else {
1316 lappend newlist $i $script
1317 }
1318 }
1319 if {$newlist ne {}} {
1320 set commitinterest($prefix) $newlist
1321 } else {
1322 unset commitinterest($prefix)
1323 }
1324 }
1325 return $scripts
1326}
1327
1328proc getcommitlines {fd inst view updating} {
1329 global cmitlisted leftover
1330 global commitidx commitdata vdatemode
1331 global parents children curview hlview
1332 global idpending ordertok
1333 global varccommits varcid varctok vtokmod vfilelimit
1334
1335 set stuff [read $fd 500000]
1336 # git log doesn't terminate the last commit with a null...
1337 if {$stuff == {} && $leftover($inst) ne {} && [eof $fd]} {
1338 set stuff "\0"
1339 }
1340 if {$stuff == {}} {
1341 if {![eof $fd]} {
1342 return 1
1343 }
1344 global commfd viewcomplete viewactive viewname
1345 global viewinstances
1346 unset commfd($inst)
1347 set i [lsearch -exact $viewinstances($view) $inst]
1348 if {$i >= 0} {
1349 set viewinstances($view) [lreplace $viewinstances($view) $i $i]
1350 }
1351 # set it blocking so we wait for the process to terminate
1352 fconfigure $fd -blocking 1
1353 if {[catch {close $fd} err]} {
1354 set fv {}
1355 if {$view != $curview} {
1356 set fv " for the \"$viewname($view)\" view"
1357 }
1358 if {[string range $err 0 4] == "usage"} {
1359 set err "Gitk: error reading commits$fv:\
1360 bad arguments to git log."
1361 if {$viewname($view) eq "Command line"} {
1362 append err \
1363 " (Note: arguments to gitk are passed to git log\
1364 to allow selection of commits to be displayed.)"
1365 }
1366 } else {
1367 set err "Error reading commits$fv: $err"
1368 }
1369 error_popup $err
1370 }
1371 if {[incr viewactive($view) -1] <= 0} {
1372 set viewcomplete($view) 1
1373 # Check if we have seen any ids listed as parents that haven't
1374 # appeared in the list
1375 closevarcs $view
1376 notbusy $view
1377 }
1378 if {$view == $curview} {
1379 run chewcommits
1380 }
1381 return 0
1382 }
1383 set start 0
1384 set gotsome 0
1385 set scripts {}
1386 while 1 {
1387 set i [string first "\0" $stuff $start]
1388 if {$i < 0} {
1389 append leftover($inst) [string range $stuff $start end]
1390 break
1391 }
1392 if {$start == 0} {
1393 set cmit $leftover($inst)
1394 append cmit [string range $stuff 0 [expr {$i - 1}]]
1395 set leftover($inst) {}
1396 } else {
1397 set cmit [string range $stuff $start [expr {$i - 1}]]
1398 }
1399 set start [expr {$i + 1}]
1400 set j [string first "\n" $cmit]
1401 set ok 0
1402 set listed 1
1403 if {$j >= 0 && [string match "commit *" $cmit]} {
1404 set ids [string range $cmit 7 [expr {$j - 1}]]
1405 if {[string match {[-^<>]*} $ids]} {
1406 switch -- [string index $ids 0] {
1407 "-" {set listed 0}
1408 "^" {set listed 2}
1409 "<" {set listed 3}
1410 ">" {set listed 4}
1411 }
1412 set ids [string range $ids 1 end]
1413 }
1414 set ok 1
1415 foreach id $ids {
1416 if {[string length $id] != 40} {
1417 set ok 0
1418 break
1419 }
1420 }
1421 }
1422 if {!$ok} {
1423 set shortcmit $cmit
1424 if {[string length $shortcmit] > 80} {
1425 set shortcmit "[string range $shortcmit 0 80]..."
1426 }
1427 error_popup "[mc "Can't parse git log output:"] {$shortcmit}"
1428 exit 1
1429 }
1430 set id [lindex $ids 0]
1431 set vid $view,$id
1432
1433 if {!$listed && $updating && ![info exists varcid($vid)] &&
1434 $vfilelimit($view) ne {}} {
1435 # git log doesn't rewrite parents for unlisted commits
1436 # when doing path limiting, so work around that here
1437 # by working out the rewritten parent with git rev-list
1438 # and if we already know about it, using the rewritten
1439 # parent as a substitute parent for $id's children.
1440 if {![catch {
1441 set rwid [exec git rev-list --first-parent --max-count=1 \
1442 $id -- $vfilelimit($view)]
1443 }]} {
1444 if {$rwid ne {} && [info exists varcid($view,$rwid)]} {
1445 # use $rwid in place of $id
1446 rewrite_commit $view $id $rwid
1447 continue
1448 }
1449 }
1450 }
1451
1452 set a 0
1453 if {[info exists varcid($vid)]} {
1454 if {$cmitlisted($vid) || !$listed} continue
1455 set a $varcid($vid)
1456 }
1457 if {$listed} {
1458 set olds [lrange $ids 1 end]
1459 } else {
1460 set olds {}
1461 }
1462 set commitdata($id) [string range $cmit [expr {$j + 1}] end]
1463 set cmitlisted($vid) $listed
1464 set parents($vid) $olds
1465 if {![info exists children($vid)]} {
1466 set children($vid) {}
1467 } elseif {$a == 0 && [llength $children($vid)] == 1} {
1468 set k [lindex $children($vid) 0]
1469 if {[llength $parents($view,$k)] == 1 &&
1470 (!$vdatemode($view) ||
1471 $varcid($view,$k) == [llength $varctok($view)] - 1)} {
1472 set a $varcid($view,$k)
1473 }
1474 }
1475 if {$a == 0} {
1476 # new arc
1477 set a [newvarc $view $id]
1478 }
1479 if {[string compare [lindex $varctok($view) $a] $vtokmod($view)] < 0} {
1480 modify_arc $view $a
1481 }
1482 if {![info exists varcid($vid)]} {
1483 set varcid($vid) $a
1484 lappend varccommits($view,$a) $id
1485 incr commitidx($view)
1486 }
1487
1488 set i 0
1489 foreach p $olds {
1490 if {$i == 0 || [lsearch -exact $olds $p] >= $i} {
1491 set vp $view,$p
1492 if {[llength [lappend children($vp) $id]] > 1 &&
1493 [vtokcmp $view [lindex $children($vp) end-1] $id] > 0} {
1494 set children($vp) [lsort -command [list vtokcmp $view] \
1495 $children($vp)]
1496 catch {unset ordertok}
1497 }
1498 if {[info exists varcid($view,$p)]} {
1499 fix_reversal $p $a $view
1500 }
1501 }
1502 incr i
1503 }
1504
1505 set scripts [check_interest $id $scripts]
1506 set gotsome 1
1507 }
1508 if {$gotsome} {
1509 global numcommits hlview
1510
1511 if {$view == $curview} {
1512 set numcommits $commitidx($view)
1513 run chewcommits
1514 }
1515 if {[info exists hlview] && $view == $hlview} {
1516 # we never actually get here...
1517 run vhighlightmore
1518 }
1519 foreach s $scripts {
1520 eval $s
1521 }
1522 }
1523 return 2
1524}
1525
1526proc chewcommits {} {
1527 global curview hlview viewcomplete
1528 global pending_select
1529
1530 layoutmore
1531 if {$viewcomplete($curview)} {
1532 global commitidx varctok
1533 global numcommits startmsecs
1534
1535 if {[info exists pending_select]} {
1536 update
1537 reset_pending_select {}
1538
1539 if {[commitinview $pending_select $curview]} {
1540 selectline [rowofcommit $pending_select] 1
1541 } else {
1542 set row [first_real_row]
1543 selectline $row 1
1544 }
1545 }
1546 if {$commitidx($curview) > 0} {
1547 #set ms [expr {[clock clicks -milliseconds] - $startmsecs}]
1548 #puts "overall $ms ms for $numcommits commits"
1549 #puts "[llength $varctok($view)] arcs, $commitidx($view) commits"
1550 } else {
1551 show_status [mc "No commits selected"]
1552 }
1553 notbusy layout
1554 }
1555 return 0
1556}
1557
1558proc do_readcommit {id} {
1559 global tclencoding
1560
1561 # Invoke git-log to handle automatic encoding conversion
1562 set fd [open [concat | git log --no-color --pretty=raw -1 $id] r]
1563 # Read the results using i18n.logoutputencoding
1564 fconfigure $fd -translation lf -eofchar {}
1565 if {$tclencoding != {}} {
1566 fconfigure $fd -encoding $tclencoding
1567 }
1568 set contents [read $fd]
1569 close $fd
1570 # Remove the heading line
1571 regsub {^commit [0-9a-f]+\n} $contents {} contents
1572
1573 return $contents
1574}
1575
1576proc readcommit {id} {
1577 if {[catch {set contents [do_readcommit $id]}]} return
1578 parsecommit $id $contents 1
1579}
1580
1581proc parsecommit {id contents listed} {
1582 global commitinfo cdate
1583
1584 set inhdr 1
1585 set comment {}
1586 set headline {}
1587 set auname {}
1588 set audate {}
1589 set comname {}
1590 set comdate {}
1591 set hdrend [string first "\n\n" $contents]
1592 if {$hdrend < 0} {
1593 # should never happen...
1594 set hdrend [string length $contents]
1595 }
1596 set header [string range $contents 0 [expr {$hdrend - 1}]]
1597 set comment [string range $contents [expr {$hdrend + 2}] end]
1598 foreach line [split $header "\n"] {
1599 set tag [lindex $line 0]
1600 if {$tag == "author"} {
1601 set audate [lindex $line end-1]
1602 set auname [lrange $line 1 end-2]
1603 } elseif {$tag == "committer"} {
1604 set comdate [lindex $line end-1]
1605 set comname [lrange $line 1 end-2]
1606 }
1607 }
1608 set headline {}
1609 # take the first non-blank line of the comment as the headline
1610 set headline [string trimleft $comment]
1611 set i [string first "\n" $headline]
1612 if {$i >= 0} {
1613 set headline [string range $headline 0 $i]
1614 }
1615 set headline [string trimright $headline]
1616 set i [string first "\r" $headline]
1617 if {$i >= 0} {
1618 set headline [string trimright [string range $headline 0 $i]]
1619 }
1620 if {!$listed} {
1621 # git log indents the comment by 4 spaces;
1622 # if we got this via git cat-file, add the indentation
1623 set newcomment {}
1624 foreach line [split $comment "\n"] {
1625 append newcomment " "
1626 append newcomment $line
1627 append newcomment "\n"
1628 }
1629 set comment $newcomment
1630 }
1631 if {$comdate != {}} {
1632 set cdate($id) $comdate
1633 }
1634 set commitinfo($id) [list $headline $auname $audate \
1635 $comname $comdate $comment]
1636}
1637
1638proc getcommit {id} {
1639 global commitdata commitinfo
1640
1641 if {[info exists commitdata($id)]} {
1642 parsecommit $id $commitdata($id) 1
1643 } else {
1644 readcommit $id
1645 if {![info exists commitinfo($id)]} {
1646 set commitinfo($id) [list [mc "No commit information available"]]
1647 }
1648 }
1649 return 1
1650}
1651
1652# Expand an abbreviated commit ID to a list of full 40-char IDs that match
1653# and are present in the current view.
1654# This is fairly slow...
1655proc longid {prefix} {
1656 global varcid curview
1657
1658 set ids {}
1659 foreach match [array names varcid "$curview,$prefix*"] {
1660 lappend ids [lindex [split $match ","] 1]
1661 }
1662 return $ids
1663}
1664
1665proc readrefs {} {
1666 global tagids idtags headids idheads tagobjid
1667 global otherrefids idotherrefs mainhead mainheadid
1668 global selecthead selectheadid
1669
1670 foreach v {tagids idtags headids idheads otherrefids idotherrefs} {
1671 catch {unset $v}
1672 }
1673 set refd [open [list | git show-ref -d] r]
1674 while {[gets $refd line] >= 0} {
1675 if {[string index $line 40] ne " "} continue
1676 set id [string range $line 0 39]
1677 set ref [string range $line 41 end]
1678 if {![string match "refs/*" $ref]} continue
1679 set name [string range $ref 5 end]
1680 if {[string match "remotes/*" $name]} {
1681 if {![string match "*/HEAD" $name]} {
1682 set headids($name) $id
1683 lappend idheads($id) $name
1684 }
1685 } elseif {[string match "heads/*" $name]} {
1686 set name [string range $name 6 end]
1687 set headids($name) $id
1688 lappend idheads($id) $name
1689 } elseif {[string match "tags/*" $name]} {
1690 # this lets refs/tags/foo^{} overwrite refs/tags/foo,
1691 # which is what we want since the former is the commit ID
1692 set name [string range $name 5 end]
1693 if {[string match "*^{}" $name]} {
1694 set name [string range $name 0 end-3]
1695 } else {
1696 set tagobjid($name) $id
1697 }
1698 set tagids($name) $id
1699 lappend idtags($id) $name
1700 } else {
1701 set otherrefids($name) $id
1702 lappend idotherrefs($id) $name
1703 }
1704 }
1705 catch {close $refd}
1706 set mainhead {}
1707 set mainheadid {}
1708 catch {
1709 set mainheadid [exec git rev-parse HEAD]
1710 set thehead [exec git symbolic-ref HEAD]
1711 if {[string match "refs/heads/*" $thehead]} {
1712 set mainhead [string range $thehead 11 end]
1713 }
1714 }
1715 set selectheadid {}
1716 if {$selecthead ne {}} {
1717 catch {
1718 set selectheadid [exec git rev-parse --verify $selecthead]
1719 }
1720 }
1721}
1722
1723# skip over fake commits
1724proc first_real_row {} {
1725 global nullid nullid2 numcommits
1726
1727 for {set row 0} {$row < $numcommits} {incr row} {
1728 set id [commitonrow $row]
1729 if {$id ne $nullid && $id ne $nullid2} {
1730 break
1731 }
1732 }
1733 return $row
1734}
1735
1736# update things for a head moved to a child of its previous location
1737proc movehead {id name} {
1738 global headids idheads
1739
1740 removehead $headids($name) $name
1741 set headids($name) $id
1742 lappend idheads($id) $name
1743}
1744
1745# update things when a head has been removed
1746proc removehead {id name} {
1747 global headids idheads
1748
1749 if {$idheads($id) eq $name} {
1750 unset idheads($id)
1751 } else {
1752 set i [lsearch -exact $idheads($id) $name]
1753 if {$i >= 0} {
1754 set idheads($id) [lreplace $idheads($id) $i $i]
1755 }
1756 }
1757 unset headids($name)
1758}
1759
1760proc make_transient {window origin} {
1761 global have_tk85
1762
1763 # In MacOS Tk 8.4 transient appears to work by setting
1764 # overrideredirect, which is utterly useless, since the
1765 # windows get no border, and are not even kept above
1766 # the parent.
1767 if {!$have_tk85 && [tk windowingsystem] eq {aqua}} return
1768
1769 wm transient $window $origin
1770
1771 # Windows fails to place transient windows normally, so
1772 # schedule a callback to center them on the parent.
1773 if {[tk windowingsystem] eq {win32}} {
1774 after idle [list tk::PlaceWindow $window widget $origin]
1775 }
1776}
1777
1778proc show_error {w top msg} {
1779 message $w.m -text $msg -justify center -aspect 400
1780 pack $w.m -side top -fill x -padx 20 -pady 20
1781 button $w.ok -text [mc OK] -command "destroy $top"
1782 pack $w.ok -side bottom -fill x
1783 bind $top <Visibility> "grab $top; focus $top"
1784 bind $top <Key-Return> "destroy $top"
1785 bind $top <Key-space> "destroy $top"
1786 bind $top <Key-Escape> "destroy $top"
1787 tkwait window $top
1788}
1789
1790proc error_popup {msg {owner .}} {
1791 set w .error
1792 toplevel $w
1793 make_transient $w $owner
1794 show_error $w $w $msg
1795}
1796
1797proc confirm_popup {msg {owner .}} {
1798 global confirm_ok
1799 set confirm_ok 0
1800 set w .confirm
1801 toplevel $w
1802 make_transient $w $owner
1803 message $w.m -text $msg -justify center -aspect 400
1804 pack $w.m -side top -fill x -padx 20 -pady 20
1805 button $w.ok -text [mc OK] -command "set confirm_ok 1; destroy $w"
1806 pack $w.ok -side left -fill x
1807 button $w.cancel -text [mc Cancel] -command "destroy $w"
1808 pack $w.cancel -side right -fill x
1809 bind $w <Visibility> "grab $w; focus $w"
1810 bind $w <Key-Return> "set confirm_ok 1; destroy $w"
1811 bind $w <Key-space> "set confirm_ok 1; destroy $w"
1812 bind $w <Key-Escape> "destroy $w"
1813 tkwait window $w
1814 return $confirm_ok
1815}
1816
1817proc setoptions {} {
1818 option add *Panedwindow.showHandle 1 startupFile
1819 option add *Panedwindow.sashRelief raised startupFile
1820 option add *Button.font uifont startupFile
1821 option add *Checkbutton.font uifont startupFile
1822 option add *Radiobutton.font uifont startupFile
1823 option add *Menu.font uifont startupFile
1824 option add *Menubutton.font uifont startupFile
1825 option add *Label.font uifont startupFile
1826 option add *Message.font uifont startupFile
1827 option add *Entry.font uifont startupFile
1828}
1829
1830# Make a menu and submenus.
1831# m is the window name for the menu, items is the list of menu items to add.
1832# Each item is a list {mc label type description options...}
1833# mc is ignored; it's so we can put mc there to alert xgettext
1834# label is the string that appears in the menu
1835# type is cascade, command or radiobutton (should add checkbutton)
1836# description depends on type; it's the sublist for cascade, the
1837# command to invoke for command, or {variable value} for radiobutton
1838proc makemenu {m items} {
1839 menu $m
1840 if {[tk windowingsystem] eq {aqua}} {
1841 set Meta1 Cmd
1842 } else {
1843 set Meta1 Ctrl
1844 }
1845 foreach i $items {
1846 set name [mc [lindex $i 1]]
1847 set type [lindex $i 2]
1848 set thing [lindex $i 3]
1849 set params [list $type]
1850 if {$name ne {}} {
1851 set u [string first "&" [string map {&& x} $name]]
1852 lappend params -label [string map {&& & & {}} $name]
1853 if {$u >= 0} {
1854 lappend params -underline $u
1855 }
1856 }
1857 switch -- $type {
1858 "cascade" {
1859 set submenu [string tolower [string map {& ""} [lindex $i 1]]]
1860 lappend params -menu $m.$submenu
1861 }
1862 "command" {
1863 lappend params -command $thing
1864 }
1865 "radiobutton" {
1866 lappend params -variable [lindex $thing 0] \
1867 -value [lindex $thing 1]
1868 }
1869 }
1870 set tail [lrange $i 4 end]
1871 regsub -all {\yMeta1\y} $tail $Meta1 tail
1872 eval $m add $params $tail
1873 if {$type eq "cascade"} {
1874 makemenu $m.$submenu $thing
1875 }
1876 }
1877}
1878
1879# translate string and remove ampersands
1880proc mca {str} {
1881 return [string map {&& & & {}} [mc $str]]
1882}
1883
1884proc makewindow {} {
1885 global canv canv2 canv3 linespc charspc ctext cflist cscroll
1886 global tabstop
1887 global findtype findtypemenu findloc findstring fstring geometry
1888 global entries sha1entry sha1string sha1but
1889 global diffcontextstring diffcontext
1890 global ignorespace
1891 global maincursor textcursor curtextcursor
1892 global rowctxmenu fakerowmenu mergemax wrapcomment
1893 global highlight_files gdttype
1894 global searchstring sstring
1895 global bgcolor fgcolor bglist fglist diffcolors selectbgcolor
1896 global headctxmenu progresscanv progressitem progresscoords statusw
1897 global fprogitem fprogcoord lastprogupdate progupdatepending
1898 global rprogitem rprogcoord rownumsel numcommits
1899 global have_tk85
1900
1901 # The "mc" arguments here are purely so that xgettext
1902 # sees the following string as needing to be translated
1903 makemenu .bar {
1904 {mc "File" cascade {
1905 {mc "Update" command updatecommits -accelerator F5}
1906 {mc "Reload" command reloadcommits -accelerator Meta1-F5}
1907 {mc "Reread references" command rereadrefs}
1908 {mc "List references" command showrefs -accelerator F2}
1909 {mc "Quit" command doquit -accelerator Meta1-Q}
1910 }}
1911 {mc "Edit" cascade {
1912 {mc "Preferences" command doprefs}
1913 }}
1914 {mc "View" cascade {
1915 {mc "New view..." command {newview 0} -accelerator Shift-F4}
1916 {mc "Edit view..." command editview -state disabled -accelerator F4}
1917 {mc "Delete view" command delview -state disabled}
1918 {xx "" separator}
1919 {mc "All files" radiobutton {selectedview 0} -command {showview 0}}
1920 }}
1921 {mc "Help" cascade {
1922 {mc "About gitk" command about}
1923 {mc "Key bindings" command keys}
1924 }}
1925 }
1926 . configure -menu .bar
1927
1928 # the gui has upper and lower half, parts of a paned window.
1929 panedwindow .ctop -orient vertical
1930
1931 # possibly use assumed geometry
1932 if {![info exists geometry(pwsash0)]} {
1933 set geometry(topheight) [expr {15 * $linespc}]
1934 set geometry(topwidth) [expr {80 * $charspc}]
1935 set geometry(botheight) [expr {15 * $linespc}]
1936 set geometry(botwidth) [expr {50 * $charspc}]
1937 set geometry(pwsash0) "[expr {40 * $charspc}] 2"
1938 set geometry(pwsash1) "[expr {60 * $charspc}] 2"
1939 }
1940
1941 # the upper half will have a paned window, a scroll bar to the right, and some stuff below
1942 frame .tf -height $geometry(topheight) -width $geometry(topwidth)
1943 frame .tf.histframe
1944 panedwindow .tf.histframe.pwclist -orient horizontal -sashpad 0 -handlesize 4
1945
1946 # create three canvases
1947 set cscroll .tf.histframe.csb
1948 set canv .tf.histframe.pwclist.canv
1949 canvas $canv \
1950 -selectbackground $selectbgcolor \
1951 -background $bgcolor -bd 0 \
1952 -yscrollincr $linespc -yscrollcommand "scrollcanv $cscroll"
1953 .tf.histframe.pwclist add $canv
1954 set canv2 .tf.histframe.pwclist.canv2
1955 canvas $canv2 \
1956 -selectbackground $selectbgcolor \
1957 -background $bgcolor -bd 0 -yscrollincr $linespc
1958 .tf.histframe.pwclist add $canv2
1959 set canv3 .tf.histframe.pwclist.canv3
1960 canvas $canv3 \
1961 -selectbackground $selectbgcolor \
1962 -background $bgcolor -bd 0 -yscrollincr $linespc
1963 .tf.histframe.pwclist add $canv3
1964 eval .tf.histframe.pwclist sash place 0 $geometry(pwsash0)
1965 eval .tf.histframe.pwclist sash place 1 $geometry(pwsash1)
1966
1967 # a scroll bar to rule them
1968 scrollbar $cscroll -command {allcanvs yview} -highlightthickness 0
1969 pack $cscroll -side right -fill y
1970 bind .tf.histframe.pwclist <Configure> {resizeclistpanes %W %w}
1971 lappend bglist $canv $canv2 $canv3
1972 pack .tf.histframe.pwclist -fill both -expand 1 -side left
1973
1974 # we have two button bars at bottom of top frame. Bar 1
1975 frame .tf.bar
1976 frame .tf.lbar -height 15
1977
1978 set sha1entry .tf.bar.sha1
1979 set entries $sha1entry
1980 set sha1but .tf.bar.sha1label
1981 button $sha1but -text [mc "SHA1 ID: "] -state disabled -relief flat \
1982 -command gotocommit -width 8
1983 $sha1but conf -disabledforeground [$sha1but cget -foreground]
1984 pack .tf.bar.sha1label -side left
1985 entry $sha1entry -width 40 -font textfont -textvariable sha1string
1986 trace add variable sha1string write sha1change
1987 pack $sha1entry -side left -pady 2
1988
1989 image create bitmap bm-left -data {
1990 #define left_width 16
1991 #define left_height 16
1992 static unsigned char left_bits[] = {
1993 0x00, 0x00, 0xc0, 0x01, 0xe0, 0x00, 0x70, 0x00, 0x38, 0x00, 0x1c, 0x00,
1994 0x0e, 0x00, 0xff, 0x7f, 0xff, 0x7f, 0xff, 0x7f, 0x0e, 0x00, 0x1c, 0x00,
1995 0x38, 0x00, 0x70, 0x00, 0xe0, 0x00, 0xc0, 0x01};
1996 }
1997 image create bitmap bm-right -data {
1998 #define right_width 16
1999 #define right_height 16
2000 static unsigned char right_bits[] = {
2001 0x00, 0x00, 0xc0, 0x01, 0x80, 0x03, 0x00, 0x07, 0x00, 0x0e, 0x00, 0x1c,
2002 0x00, 0x38, 0xff, 0x7f, 0xff, 0x7f, 0xff, 0x7f, 0x00, 0x38, 0x00, 0x1c,
2003 0x00, 0x0e, 0x00, 0x07, 0x80, 0x03, 0xc0, 0x01};
2004 }
2005 button .tf.bar.leftbut -image bm-left -command goback \
2006 -state disabled -width 26
2007 pack .tf.bar.leftbut -side left -fill y
2008 button .tf.bar.rightbut -image bm-right -command goforw \
2009 -state disabled -width 26
2010 pack .tf.bar.rightbut -side left -fill y
2011
2012 label .tf.bar.rowlabel -text [mc "Row"]
2013 set rownumsel {}
2014 label .tf.bar.rownum -width 7 -font textfont -textvariable rownumsel \
2015 -relief sunken -anchor e
2016 label .tf.bar.rowlabel2 -text "/"
2017 label .tf.bar.numcommits -width 7 -font textfont -textvariable numcommits \
2018 -relief sunken -anchor e
2019 pack .tf.bar.rowlabel .tf.bar.rownum .tf.bar.rowlabel2 .tf.bar.numcommits \
2020 -side left
2021 global selectedline
2022 trace add variable selectedline write selectedline_change
2023
2024 # Status label and progress bar
2025 set statusw .tf.bar.status
2026 label $statusw -width 15 -relief sunken
2027 pack $statusw -side left -padx 5
2028 set h [expr {[font metrics uifont -linespace] + 2}]
2029 set progresscanv .tf.bar.progress
2030 canvas $progresscanv -relief sunken -height $h -borderwidth 2
2031 set progressitem [$progresscanv create rect -1 0 0 $h -fill green]
2032 set fprogitem [$progresscanv create rect -1 0 0 $h -fill yellow]
2033 set rprogitem [$progresscanv create rect -1 0 0 $h -fill red]
2034 pack $progresscanv -side right -expand 1 -fill x
2035 set progresscoords {0 0}
2036 set fprogcoord 0
2037 set rprogcoord 0
2038 bind $progresscanv <Configure> adjustprogress
2039 set lastprogupdate [clock clicks -milliseconds]
2040 set progupdatepending 0
2041
2042 # build up the bottom bar of upper window
2043 label .tf.lbar.flabel -text "[mc "Find"] "
2044 button .tf.lbar.fnext -text [mc "next"] -command {dofind 1 1}
2045 button .tf.lbar.fprev -text [mc "prev"] -command {dofind -1 1}
2046 label .tf.lbar.flab2 -text " [mc "commit"] "
2047 pack .tf.lbar.flabel .tf.lbar.fnext .tf.lbar.fprev .tf.lbar.flab2 \
2048 -side left -fill y
2049 set gdttype [mc "containing:"]
2050 set gm [tk_optionMenu .tf.lbar.gdttype gdttype \
2051 [mc "containing:"] \
2052 [mc "touching paths:"] \
2053 [mc "adding/removing string:"]]
2054 trace add variable gdttype write gdttype_change
2055 pack .tf.lbar.gdttype -side left -fill y
2056
2057 set findstring {}
2058 set fstring .tf.lbar.findstring
2059 lappend entries $fstring
2060 entry $fstring -width 30 -font textfont -textvariable findstring
2061 trace add variable findstring write find_change
2062 set findtype [mc "Exact"]
2063 set findtypemenu [tk_optionMenu .tf.lbar.findtype \
2064 findtype [mc "Exact"] [mc "IgnCase"] [mc "Regexp"]]
2065 trace add variable findtype write findcom_change
2066 set findloc [mc "All fields"]
2067 tk_optionMenu .tf.lbar.findloc findloc [mc "All fields"] [mc "Headline"] \
2068 [mc "Comments"] [mc "Author"] [mc "Committer"]
2069 trace add variable findloc write find_change
2070 pack .tf.lbar.findloc -side right
2071 pack .tf.lbar.findtype -side right
2072 pack $fstring -side left -expand 1 -fill x
2073
2074 # Finish putting the upper half of the viewer together
2075 pack .tf.lbar -in .tf -side bottom -fill x
2076 pack .tf.bar -in .tf -side bottom -fill x
2077 pack .tf.histframe -fill both -side top -expand 1
2078 .ctop add .tf
2079 .ctop paneconfigure .tf -height $geometry(topheight)
2080 .ctop paneconfigure .tf -width $geometry(topwidth)
2081
2082 # now build up the bottom
2083 panedwindow .pwbottom -orient horizontal
2084
2085 # lower left, a text box over search bar, scroll bar to the right
2086 # if we know window height, then that will set the lower text height, otherwise
2087 # we set lower text height which will drive window height
2088 if {[info exists geometry(main)]} {
2089 frame .bleft -width $geometry(botwidth)
2090 } else {
2091 frame .bleft -width $geometry(botwidth) -height $geometry(botheight)
2092 }
2093 frame .bleft.top
2094 frame .bleft.mid
2095 frame .bleft.bottom
2096
2097 button .bleft.top.search -text [mc "Search"] -command dosearch
2098 pack .bleft.top.search -side left -padx 5
2099 set sstring .bleft.top.sstring
2100 entry $sstring -width 20 -font textfont -textvariable searchstring
2101 lappend entries $sstring
2102 trace add variable searchstring write incrsearch
2103 pack $sstring -side left -expand 1 -fill x
2104 radiobutton .bleft.mid.diff -text [mc "Diff"] \
2105 -command changediffdisp -variable diffelide -value {0 0}
2106 radiobutton .bleft.mid.old -text [mc "Old version"] \
2107 -command changediffdisp -variable diffelide -value {0 1}
2108 radiobutton .bleft.mid.new -text [mc "New version"] \
2109 -command changediffdisp -variable diffelide -value {1 0}
2110 label .bleft.mid.labeldiffcontext -text " [mc "Lines of context"]: "
2111 pack .bleft.mid.diff .bleft.mid.old .bleft.mid.new -side left
2112 spinbox .bleft.mid.diffcontext -width 5 -font textfont \
2113 -from 1 -increment 1 -to 10000000 \
2114 -validate all -validatecommand "diffcontextvalidate %P" \
2115 -textvariable diffcontextstring
2116 .bleft.mid.diffcontext set $diffcontext
2117 trace add variable diffcontextstring write diffcontextchange
2118 lappend entries .bleft.mid.diffcontext
2119 pack .bleft.mid.labeldiffcontext .bleft.mid.diffcontext -side left
2120 checkbutton .bleft.mid.ignspace -text [mc "Ignore space change"] \
2121 -command changeignorespace -variable ignorespace
2122 pack .bleft.mid.ignspace -side left -padx 5
2123 set ctext .bleft.bottom.ctext
2124 text $ctext -background $bgcolor -foreground $fgcolor \
2125 -state disabled -font textfont \
2126 -yscrollcommand scrolltext -wrap none \
2127 -xscrollcommand ".bleft.bottom.sbhorizontal set"
2128 if {$have_tk85} {
2129 $ctext conf -tabstyle wordprocessor
2130 }
2131 scrollbar .bleft.bottom.sb -command "$ctext yview"
2132 scrollbar .bleft.bottom.sbhorizontal -command "$ctext xview" -orient h \
2133 -width 10
2134 pack .bleft.top -side top -fill x
2135 pack .bleft.mid -side top -fill x
2136 grid $ctext .bleft.bottom.sb -sticky nsew
2137 grid .bleft.bottom.sbhorizontal -sticky ew
2138 grid columnconfigure .bleft.bottom 0 -weight 1
2139 grid rowconfigure .bleft.bottom 0 -weight 1
2140 grid rowconfigure .bleft.bottom 1 -weight 0
2141 pack .bleft.bottom -side top -fill both -expand 1
2142 lappend bglist $ctext
2143 lappend fglist $ctext
2144
2145 $ctext tag conf comment -wrap $wrapcomment
2146 $ctext tag conf filesep -font textfontbold -back "#aaaaaa"
2147 $ctext tag conf hunksep -fore [lindex $diffcolors 2]
2148 $ctext tag conf d0 -fore [lindex $diffcolors 0]
2149 $ctext tag conf dresult -fore [lindex $diffcolors 1]
2150 $ctext tag conf m0 -fore red
2151 $ctext tag conf m1 -fore blue
2152 $ctext tag conf m2 -fore green
2153 $ctext tag conf m3 -fore purple
2154 $ctext tag conf m4 -fore brown
2155 $ctext tag conf m5 -fore "#009090"
2156 $ctext tag conf m6 -fore magenta
2157 $ctext tag conf m7 -fore "#808000"
2158 $ctext tag conf m8 -fore "#009000"
2159 $ctext tag conf m9 -fore "#ff0080"
2160 $ctext tag conf m10 -fore cyan
2161 $ctext tag conf m11 -fore "#b07070"
2162 $ctext tag conf m12 -fore "#70b0f0"
2163 $ctext tag conf m13 -fore "#70f0b0"
2164 $ctext tag conf m14 -fore "#f0b070"
2165 $ctext tag conf m15 -fore "#ff70b0"
2166 $ctext tag conf mmax -fore darkgrey
2167 set mergemax 16
2168 $ctext tag conf mresult -font textfontbold
2169 $ctext tag conf msep -font textfontbold
2170 $ctext tag conf found -back yellow
2171
2172 .pwbottom add .bleft
2173 .pwbottom paneconfigure .bleft -width $geometry(botwidth)
2174
2175 # lower right
2176 frame .bright
2177 frame .bright.mode
2178 radiobutton .bright.mode.patch -text [mc "Patch"] \
2179 -command reselectline -variable cmitmode -value "patch"
2180 radiobutton .bright.mode.tree -text [mc "Tree"] \
2181 -command reselectline -variable cmitmode -value "tree"
2182 grid .bright.mode.patch .bright.mode.tree -sticky ew
2183 pack .bright.mode -side top -fill x
2184 set cflist .bright.cfiles
2185 set indent [font measure mainfont "nn"]
2186 text $cflist \
2187 -selectbackground $selectbgcolor \
2188 -background $bgcolor -foreground $fgcolor \
2189 -font mainfont \
2190 -tabs [list $indent [expr {2 * $indent}]] \
2191 -yscrollcommand ".bright.sb set" \
2192 -cursor [. cget -cursor] \
2193 -spacing1 1 -spacing3 1
2194 lappend bglist $cflist
2195 lappend fglist $cflist
2196 scrollbar .bright.sb -command "$cflist yview"
2197 pack .bright.sb -side right -fill y
2198 pack $cflist -side left -fill both -expand 1
2199 $cflist tag configure highlight \
2200 -background [$cflist cget -selectbackground]
2201 $cflist tag configure bold -font mainfontbold
2202
2203 .pwbottom add .bright
2204 .ctop add .pwbottom
2205
2206 # restore window width & height if known
2207 if {[info exists geometry(main)]} {
2208 if {[scan $geometry(main) "%dx%d" w h] >= 2} {
2209 if {$w > [winfo screenwidth .]} {
2210 set w [winfo screenwidth .]
2211 }
2212 if {$h > [winfo screenheight .]} {
2213 set h [winfo screenheight .]
2214 }
2215 wm geometry . "${w}x$h"
2216 }
2217 }
2218
2219 if {[tk windowingsystem] eq {aqua}} {
2220 set M1B M1
2221 } else {
2222 set M1B Control
2223 }
2224
2225 bind .pwbottom <Configure> {resizecdetpanes %W %w}
2226 pack .ctop -fill both -expand 1
2227 bindall <1> {selcanvline %W %x %y}
2228 #bindall <B1-Motion> {selcanvline %W %x %y}
2229 if {[tk windowingsystem] == "win32"} {
2230 bind . <MouseWheel> { windows_mousewheel_redirector %W %X %Y %D }
2231 bind $ctext <MouseWheel> { windows_mousewheel_redirector %W %X %Y %D ; break }
2232 } else {
2233 bindall <ButtonRelease-4> "allcanvs yview scroll -5 units"
2234 bindall <ButtonRelease-5> "allcanvs yview scroll 5 units"
2235 if {[tk windowingsystem] eq "aqua"} {
2236 bindall <MouseWheel> {
2237 set delta [expr {- (%D)}]
2238 allcanvs yview scroll $delta units
2239 }
2240 }
2241 }
2242 bindall <2> "canvscan mark %W %x %y"
2243 bindall <B2-Motion> "canvscan dragto %W %x %y"
2244 bindkey <Home> selfirstline
2245 bindkey <End> sellastline
2246 bind . <Key-Up> "selnextline -1"
2247 bind . <Key-Down> "selnextline 1"
2248 bind . <Shift-Key-Up> "dofind -1 0"
2249 bind . <Shift-Key-Down> "dofind 1 0"
2250 bindkey <Key-Right> "goforw"
2251 bindkey <Key-Left> "goback"
2252 bind . <Key-Prior> "selnextpage -1"
2253 bind . <Key-Next> "selnextpage 1"
2254 bind . <$M1B-Home> "allcanvs yview moveto 0.0"
2255 bind . <$M1B-End> "allcanvs yview moveto 1.0"
2256 bind . <$M1B-Key-Up> "allcanvs yview scroll -1 units"
2257 bind . <$M1B-Key-Down> "allcanvs yview scroll 1 units"
2258 bind . <$M1B-Key-Prior> "allcanvs yview scroll -1 pages"
2259 bind . <$M1B-Key-Next> "allcanvs yview scroll 1 pages"
2260 bindkey <Key-Delete> "$ctext yview scroll -1 pages"
2261 bindkey <Key-BackSpace> "$ctext yview scroll -1 pages"
2262 bindkey <Key-space> "$ctext yview scroll 1 pages"
2263 bindkey p "selnextline -1"
2264 bindkey n "selnextline 1"
2265 bindkey z "goback"
2266 bindkey x "goforw"
2267 bindkey i "selnextline -1"
2268 bindkey k "selnextline 1"
2269 bindkey j "goback"
2270 bindkey l "goforw"
2271 bindkey b prevfile
2272 bindkey d "$ctext yview scroll 18 units"
2273 bindkey u "$ctext yview scroll -18 units"
2274 bindkey / {dofind 1 1}
2275 bindkey <Key-Return> {dofind 1 1}
2276 bindkey ? {dofind -1 1}
2277 bindkey f nextfile
2278 bind . <F5> updatecommits
2279 bind . <$M1B-F5> reloadcommits
2280 bind . <F2> showrefs
2281 bind . <Shift-F4> {newview 0}
2282 catch { bind . <Shift-Key-XF86_Switch_VT_4> {newview 0} }
2283 bind . <F4> edit_or_newview
2284 bind . <$M1B-q> doquit
2285 bind . <$M1B-f> {dofind 1 1}
2286 bind . <$M1B-g> {dofind 1 0}
2287 bind . <$M1B-r> dosearchback
2288 bind . <$M1B-s> dosearch
2289 bind . <$M1B-equal> {incrfont 1}
2290 bind . <$M1B-plus> {incrfont 1}
2291 bind . <$M1B-KP_Add> {incrfont 1}
2292 bind . <$M1B-minus> {incrfont -1}
2293 bind . <$M1B-KP_Subtract> {incrfont -1}
2294 wm protocol . WM_DELETE_WINDOW doquit
2295 bind . <Destroy> {stop_backends}
2296 bind . <Button-1> "click %W"
2297 bind $fstring <Key-Return> {dofind 1 1}
2298 bind $sha1entry <Key-Return> {gotocommit; break}
2299 bind $sha1entry <<PasteSelection>> clearsha1
2300 bind $cflist <1> {sel_flist %W %x %y; break}
2301 bind $cflist <B1-Motion> {sel_flist %W %x %y; break}
2302 bind $cflist <ButtonRelease-1> {treeclick %W %x %y}
2303 global ctxbut
2304 bind $cflist $ctxbut {pop_flist_menu %W %X %Y %x %y}
2305 bind $ctext $ctxbut {pop_diff_menu %W %X %Y %x %y}
2306
2307 set maincursor [. cget -cursor]
2308 set textcursor [$ctext cget -cursor]
2309 set curtextcursor $textcursor
2310
2311 set rowctxmenu .rowctxmenu
2312 makemenu $rowctxmenu {
2313 {mc "Diff this -> selected" command {diffvssel 0}}
2314 {mc "Diff selected -> this" command {diffvssel 1}}
2315 {mc "Make patch" command mkpatch}
2316 {mc "Create tag" command mktag}
2317 {mc "Write commit to file" command writecommit}
2318 {mc "Create new branch" command mkbranch}
2319 {mc "Cherry-pick this commit" command cherrypick}
2320 {mc "Reset HEAD branch to here" command resethead}
2321 }
2322 $rowctxmenu configure -tearoff 0
2323
2324 set fakerowmenu .fakerowmenu
2325 makemenu $fakerowmenu {
2326 {mc "Diff this -> selected" command {diffvssel 0}}
2327 {mc "Diff selected -> this" command {diffvssel 1}}
2328 {mc "Make patch" command mkpatch}
2329 }
2330 $fakerowmenu configure -tearoff 0
2331
2332 set headctxmenu .headctxmenu
2333 makemenu $headctxmenu {
2334 {mc "Check out this branch" command cobranch}
2335 {mc "Remove this branch" command rmbranch}
2336 }
2337 $headctxmenu configure -tearoff 0
2338
2339 global flist_menu
2340 set flist_menu .flistctxmenu
2341 makemenu $flist_menu {
2342 {mc "Highlight this too" command {flist_hl 0}}
2343 {mc "Highlight this only" command {flist_hl 1}}
2344 {mc "External diff" command {external_diff}}
2345 {mc "Blame parent commit" command {external_blame 1}}
2346 }
2347 $flist_menu configure -tearoff 0
2348
2349 global diff_menu
2350 set diff_menu .diffctxmenu
2351 makemenu $diff_menu {
2352 {mc "Show origin of this line" command show_line_source}
2353 {mc "Run git gui blame on this line" command {external_blame_diff}}
2354 }
2355 $diff_menu configure -tearoff 0
2356}
2357
2358# Windows sends all mouse wheel events to the current focused window, not
2359# the one where the mouse hovers, so bind those events here and redirect
2360# to the correct window
2361proc windows_mousewheel_redirector {W X Y D} {
2362 global canv canv2 canv3
2363 set w [winfo containing -displayof $W $X $Y]
2364 if {$w ne ""} {
2365 set u [expr {$D < 0 ? 5 : -5}]
2366 if {$w == $canv || $w == $canv2 || $w == $canv3} {
2367 allcanvs yview scroll $u units
2368 } else {
2369 catch {
2370 $w yview scroll $u units
2371 }
2372 }
2373 }
2374}
2375
2376# Update row number label when selectedline changes
2377proc selectedline_change {n1 n2 op} {
2378 global selectedline rownumsel
2379
2380 if {$selectedline eq {}} {
2381 set rownumsel {}
2382 } else {
2383 set rownumsel [expr {$selectedline + 1}]
2384 }
2385}
2386
2387# mouse-2 makes all windows scan vertically, but only the one
2388# the cursor is in scans horizontally
2389proc canvscan {op w x y} {
2390 global canv canv2 canv3
2391 foreach c [list $canv $canv2 $canv3] {
2392 if {$c == $w} {
2393 $c scan $op $x $y
2394 } else {
2395 $c scan $op 0 $y
2396 }
2397 }
2398}
2399
2400proc scrollcanv {cscroll f0 f1} {
2401 $cscroll set $f0 $f1
2402 drawvisible
2403 flushhighlights
2404}
2405
2406# when we make a key binding for the toplevel, make sure
2407# it doesn't get triggered when that key is pressed in the
2408# find string entry widget.
2409proc bindkey {ev script} {
2410 global entries
2411 bind . $ev $script
2412 set escript [bind Entry $ev]
2413 if {$escript == {}} {
2414 set escript [bind Entry <Key>]
2415 }
2416 foreach e $entries {
2417 bind $e $ev "$escript; break"
2418 }
2419}
2420
2421# set the focus back to the toplevel for any click outside
2422# the entry widgets
2423proc click {w} {
2424 global ctext entries
2425 foreach e [concat $entries $ctext] {
2426 if {$w == $e} return
2427 }
2428 focus .
2429}
2430
2431# Adjust the progress bar for a change in requested extent or canvas size
2432proc adjustprogress {} {
2433 global progresscanv progressitem progresscoords
2434 global fprogitem fprogcoord lastprogupdate progupdatepending
2435 global rprogitem rprogcoord
2436
2437 set w [expr {[winfo width $progresscanv] - 4}]
2438 set x0 [expr {$w * [lindex $progresscoords 0]}]
2439 set x1 [expr {$w * [lindex $progresscoords 1]}]
2440 set h [winfo height $progresscanv]
2441 $progresscanv coords $progressitem $x0 0 $x1 $h
2442 $progresscanv coords $fprogitem 0 0 [expr {$w * $fprogcoord}] $h
2443 $progresscanv coords $rprogitem 0 0 [expr {$w * $rprogcoord}] $h
2444 set now [clock clicks -milliseconds]
2445 if {$now >= $lastprogupdate + 100} {
2446 set progupdatepending 0
2447 update
2448 } elseif {!$progupdatepending} {
2449 set progupdatepending 1
2450 after [expr {$lastprogupdate + 100 - $now}] doprogupdate
2451 }
2452}
2453
2454proc doprogupdate {} {
2455 global lastprogupdate progupdatepending
2456
2457 if {$progupdatepending} {
2458 set progupdatepending 0
2459 set lastprogupdate [clock clicks -milliseconds]
2460 update
2461 }
2462}
2463
2464proc savestuff {w} {
2465 global canv canv2 canv3 mainfont textfont uifont tabstop
2466 global stuffsaved findmergefiles maxgraphpct
2467 global maxwidth showneartags showlocalchanges
2468 global viewname viewfiles viewargs viewargscmd viewperm nextviewnum
2469 global cmitmode wrapcomment datetimeformat limitdiffs
2470 global colors bgcolor fgcolor diffcolors diffcontext selectbgcolor
2471 global autoselect extdifftool perfile_attrs markbgcolor
2472
2473 if {$stuffsaved} return
2474 if {![winfo viewable .]} return
2475 catch {
2476 set f [open "~/.gitk-new" w]
2477 puts $f [list set mainfont $mainfont]
2478 puts $f [list set textfont $textfont]
2479 puts $f [list set uifont $uifont]
2480 puts $f [list set tabstop $tabstop]
2481 puts $f [list set findmergefiles $findmergefiles]
2482 puts $f [list set maxgraphpct $maxgraphpct]
2483 puts $f [list set maxwidth $maxwidth]
2484 puts $f [list set cmitmode $cmitmode]
2485 puts $f [list set wrapcomment $wrapcomment]
2486 puts $f [list set autoselect $autoselect]
2487 puts $f [list set showneartags $showneartags]
2488 puts $f [list set showlocalchanges $showlocalchanges]
2489 puts $f [list set datetimeformat $datetimeformat]
2490 puts $f [list set limitdiffs $limitdiffs]
2491 puts $f [list set bgcolor $bgcolor]
2492 puts $f [list set fgcolor $fgcolor]
2493 puts $f [list set colors $colors]
2494 puts $f [list set diffcolors $diffcolors]
2495 puts $f [list set markbgcolor $markbgcolor]
2496 puts $f [list set diffcontext $diffcontext]
2497 puts $f [list set selectbgcolor $selectbgcolor]
2498 puts $f [list set extdifftool $extdifftool]
2499 puts $f [list set perfile_attrs $perfile_attrs]
2500
2501 puts $f "set geometry(main) [wm geometry .]"
2502 puts $f "set geometry(topwidth) [winfo width .tf]"
2503 puts $f "set geometry(topheight) [winfo height .tf]"
2504 puts $f "set geometry(pwsash0) \"[.tf.histframe.pwclist sash coord 0]\""
2505 puts $f "set geometry(pwsash1) \"[.tf.histframe.pwclist sash coord 1]\""
2506 puts $f "set geometry(botwidth) [winfo width .bleft]"
2507 puts $f "set geometry(botheight) [winfo height .bleft]"
2508
2509 puts -nonewline $f "set permviews {"
2510 for {set v 0} {$v < $nextviewnum} {incr v} {
2511 if {$viewperm($v)} {
2512 puts $f "{[list $viewname($v) $viewfiles($v) $viewargs($v) $viewargscmd($v)]}"
2513 }
2514 }
2515 puts $f "}"
2516 close $f
2517 file rename -force "~/.gitk-new" "~/.gitk"
2518 }
2519 set stuffsaved 1
2520}
2521
2522proc resizeclistpanes {win w} {
2523 global oldwidth
2524 if {[info exists oldwidth($win)]} {
2525 set s0 [$win sash coord 0]
2526 set s1 [$win sash coord 1]
2527 if {$w < 60} {
2528 set sash0 [expr {int($w/2 - 2)}]
2529 set sash1 [expr {int($w*5/6 - 2)}]
2530 } else {
2531 set factor [expr {1.0 * $w / $oldwidth($win)}]
2532 set sash0 [expr {int($factor * [lindex $s0 0])}]
2533 set sash1 [expr {int($factor * [lindex $s1 0])}]
2534 if {$sash0 < 30} {
2535 set sash0 30
2536 }
2537 if {$sash1 < $sash0 + 20} {
2538 set sash1 [expr {$sash0 + 20}]
2539 }
2540 if {$sash1 > $w - 10} {
2541 set sash1 [expr {$w - 10}]
2542 if {$sash0 > $sash1 - 20} {
2543 set sash0 [expr {$sash1 - 20}]
2544 }
2545 }
2546 }
2547 $win sash place 0 $sash0 [lindex $s0 1]
2548 $win sash place 1 $sash1 [lindex $s1 1]
2549 }
2550 set oldwidth($win) $w
2551}
2552
2553proc resizecdetpanes {win w} {
2554 global oldwidth
2555 if {[info exists oldwidth($win)]} {
2556 set s0 [$win sash coord 0]
2557 if {$w < 60} {
2558 set sash0 [expr {int($w*3/4 - 2)}]
2559 } else {
2560 set factor [expr {1.0 * $w / $oldwidth($win)}]
2561 set sash0 [expr {int($factor * [lindex $s0 0])}]
2562 if {$sash0 < 45} {
2563 set sash0 45
2564 }
2565 if {$sash0 > $w - 15} {
2566 set sash0 [expr {$w - 15}]
2567 }
2568 }
2569 $win sash place 0 $sash0 [lindex $s0 1]
2570 }
2571 set oldwidth($win) $w
2572}
2573
2574proc allcanvs args {
2575 global canv canv2 canv3
2576 eval $canv $args
2577 eval $canv2 $args
2578 eval $canv3 $args
2579}
2580
2581proc bindall {event action} {
2582 global canv canv2 canv3
2583 bind $canv $event $action
2584 bind $canv2 $event $action
2585 bind $canv3 $event $action
2586}
2587
2588proc about {} {
2589 global uifont
2590 set w .about
2591 if {[winfo exists $w]} {
2592 raise $w
2593 return
2594 }
2595 toplevel $w
2596 wm title $w [mc "About gitk"]
2597 make_transient $w .
2598 message $w.m -text [mc "
2599Gitk - a commit viewer for git
2600
2601Copyright © 2005-2008 Paul Mackerras
2602
2603Use and redistribute under the terms of the GNU General Public License"] \
2604 -justify center -aspect 400 -border 2 -bg white -relief groove
2605 pack $w.m -side top -fill x -padx 2 -pady 2
2606 button $w.ok -text [mc "Close"] -command "destroy $w" -default active
2607 pack $w.ok -side bottom
2608 bind $w <Visibility> "focus $w.ok"
2609 bind $w <Key-Escape> "destroy $w"
2610 bind $w <Key-Return> "destroy $w"
2611}
2612
2613proc keys {} {
2614 set w .keys
2615 if {[winfo exists $w]} {
2616 raise $w
2617 return
2618 }
2619 if {[tk windowingsystem] eq {aqua}} {
2620 set M1T Cmd
2621 } else {
2622 set M1T Ctrl
2623 }
2624 toplevel $w
2625 wm title $w [mc "Gitk key bindings"]
2626 make_transient $w .
2627 message $w.m -text "
2628[mc "Gitk key bindings:"]
2629
2630[mc "<%s-Q> Quit" $M1T]
2631[mc "<Home> Move to first commit"]
2632[mc "<End> Move to last commit"]
2633[mc "<Up>, p, i Move up one commit"]
2634[mc "<Down>, n, k Move down one commit"]
2635[mc "<Left>, z, j Go back in history list"]
2636[mc "<Right>, x, l Go forward in history list"]
2637[mc "<PageUp> Move up one page in commit list"]
2638[mc "<PageDown> Move down one page in commit list"]
2639[mc "<%s-Home> Scroll to top of commit list" $M1T]
2640[mc "<%s-End> Scroll to bottom of commit list" $M1T]
2641[mc "<%s-Up> Scroll commit list up one line" $M1T]
2642[mc "<%s-Down> Scroll commit list down one line" $M1T]
2643[mc "<%s-PageUp> Scroll commit list up one page" $M1T]
2644[mc "<%s-PageDown> Scroll commit list down one page" $M1T]
2645[mc "<Shift-Up> Find backwards (upwards, later commits)"]
2646[mc "<Shift-Down> Find forwards (downwards, earlier commits)"]
2647[mc "<Delete>, b Scroll diff view up one page"]
2648[mc "<Backspace> Scroll diff view up one page"]
2649[mc "<Space> Scroll diff view down one page"]
2650[mc "u Scroll diff view up 18 lines"]
2651[mc "d Scroll diff view down 18 lines"]
2652[mc "<%s-F> Find" $M1T]
2653[mc "<%s-G> Move to next find hit" $M1T]
2654[mc "<Return> Move to next find hit"]
2655[mc "/ Move to next find hit, or redo find"]
2656[mc "? Move to previous find hit"]
2657[mc "f Scroll diff view to next file"]
2658[mc "<%s-S> Search for next hit in diff view" $M1T]
2659[mc "<%s-R> Search for previous hit in diff view" $M1T]
2660[mc "<%s-KP+> Increase font size" $M1T]
2661[mc "<%s-plus> Increase font size" $M1T]
2662[mc "<%s-KP-> Decrease font size" $M1T]
2663[mc "<%s-minus> Decrease font size" $M1T]
2664[mc "<F5> Update"]
2665" \
2666 -justify left -bg white -border 2 -relief groove
2667 pack $w.m -side top -fill both -padx 2 -pady 2
2668 button $w.ok -text [mc "Close"] -command "destroy $w" -default active
2669 bind $w <Key-Escape> [list destroy $w]
2670 pack $w.ok -side bottom
2671 bind $w <Visibility> "focus $w.ok"
2672 bind $w <Key-Escape> "destroy $w"
2673 bind $w <Key-Return> "destroy $w"
2674}
2675
2676# Procedures for manipulating the file list window at the
2677# bottom right of the overall window.
2678
2679proc treeview {w l openlevs} {
2680 global treecontents treediropen treeheight treeparent treeindex
2681
2682 set ix 0
2683 set treeindex() 0
2684 set lev 0
2685 set prefix {}
2686 set prefixend -1
2687 set prefendstack {}
2688 set htstack {}
2689 set ht 0
2690 set treecontents() {}
2691 $w conf -state normal
2692 foreach f $l {
2693 while {[string range $f 0 $prefixend] ne $prefix} {
2694 if {$lev <= $openlevs} {
2695 $w mark set e:$treeindex($prefix) "end -1c"
2696 $w mark gravity e:$treeindex($prefix) left
2697 }
2698 set treeheight($prefix) $ht
2699 incr ht [lindex $htstack end]
2700 set htstack [lreplace $htstack end end]
2701 set prefixend [lindex $prefendstack end]
2702 set prefendstack [lreplace $prefendstack end end]
2703 set prefix [string range $prefix 0 $prefixend]
2704 incr lev -1
2705 }
2706 set tail [string range $f [expr {$prefixend+1}] end]
2707 while {[set slash [string first "/" $tail]] >= 0} {
2708 lappend htstack $ht
2709 set ht 0
2710 lappend prefendstack $prefixend
2711 incr prefixend [expr {$slash + 1}]
2712 set d [string range $tail 0 $slash]
2713 lappend treecontents($prefix) $d
2714 set oldprefix $prefix
2715 append prefix $d
2716 set treecontents($prefix) {}
2717 set treeindex($prefix) [incr ix]
2718 set treeparent($prefix) $oldprefix
2719 set tail [string range $tail [expr {$slash+1}] end]
2720 if {$lev <= $openlevs} {
2721 set ht 1
2722 set treediropen($prefix) [expr {$lev < $openlevs}]
2723 set bm [expr {$lev == $openlevs? "tri-rt": "tri-dn"}]
2724 $w mark set d:$ix "end -1c"
2725 $w mark gravity d:$ix left
2726 set str "\n"
2727 for {set i 0} {$i < $lev} {incr i} {append str "\t"}
2728 $w insert end $str
2729 $w image create end -align center -image $bm -padx 1 \
2730 -name a:$ix
2731 $w insert end $d [highlight_tag $prefix]
2732 $w mark set s:$ix "end -1c"
2733 $w mark gravity s:$ix left
2734 }
2735 incr lev
2736 }
2737 if {$tail ne {}} {
2738 if {$lev <= $openlevs} {
2739 incr ht
2740 set str "\n"
2741 for {set i 0} {$i < $lev} {incr i} {append str "\t"}
2742 $w insert end $str
2743 $w insert end $tail [highlight_tag $f]
2744 }
2745 lappend treecontents($prefix) $tail
2746 }
2747 }
2748 while {$htstack ne {}} {
2749 set treeheight($prefix) $ht
2750 incr ht [lindex $htstack end]
2751 set htstack [lreplace $htstack end end]
2752 set prefixend [lindex $prefendstack end]
2753 set prefendstack [lreplace $prefendstack end end]
2754 set prefix [string range $prefix 0 $prefixend]
2755 }
2756 $w conf -state disabled
2757}
2758
2759proc linetoelt {l} {
2760 global treeheight treecontents
2761
2762 set y 2
2763 set prefix {}
2764 while {1} {
2765 foreach e $treecontents($prefix) {
2766 if {$y == $l} {
2767 return "$prefix$e"
2768 }
2769 set n 1
2770 if {[string index $e end] eq "/"} {
2771 set n $treeheight($prefix$e)
2772 if {$y + $n > $l} {
2773 append prefix $e
2774 incr y
2775 break
2776 }
2777 }
2778 incr y $n
2779 }
2780 }
2781}
2782
2783proc highlight_tree {y prefix} {
2784 global treeheight treecontents cflist
2785
2786 foreach e $treecontents($prefix) {
2787 set path $prefix$e
2788 if {[highlight_tag $path] ne {}} {
2789 $cflist tag add bold $y.0 "$y.0 lineend"
2790 }
2791 incr y
2792 if {[string index $e end] eq "/" && $treeheight($path) > 1} {
2793 set y [highlight_tree $y $path]
2794 }
2795 }
2796 return $y
2797}
2798
2799proc treeclosedir {w dir} {
2800 global treediropen treeheight treeparent treeindex
2801
2802 set ix $treeindex($dir)
2803 $w conf -state normal
2804 $w delete s:$ix e:$ix
2805 set treediropen($dir) 0
2806 $w image configure a:$ix -image tri-rt
2807 $w conf -state disabled
2808 set n [expr {1 - $treeheight($dir)}]
2809 while {$dir ne {}} {
2810 incr treeheight($dir) $n
2811 set dir $treeparent($dir)
2812 }
2813}
2814
2815proc treeopendir {w dir} {
2816 global treediropen treeheight treeparent treecontents treeindex
2817
2818 set ix $treeindex($dir)
2819 $w conf -state normal
2820 $w image configure a:$ix -image tri-dn
2821 $w mark set e:$ix s:$ix
2822 $w mark gravity e:$ix right
2823 set lev 0
2824 set str "\n"
2825 set n [llength $treecontents($dir)]
2826 for {set x $dir} {$x ne {}} {set x $treeparent($x)} {
2827 incr lev
2828 append str "\t"
2829 incr treeheight($x) $n
2830 }
2831 foreach e $treecontents($dir) {
2832 set de $dir$e
2833 if {[string index $e end] eq "/"} {
2834 set iy $treeindex($de)
2835 $w mark set d:$iy e:$ix
2836 $w mark gravity d:$iy left
2837 $w insert e:$ix $str
2838 set treediropen($de) 0
2839 $w image create e:$ix -align center -image tri-rt -padx 1 \
2840 -name a:$iy
2841 $w insert e:$ix $e [highlight_tag $de]
2842 $w mark set s:$iy e:$ix
2843 $w mark gravity s:$iy left
2844 set treeheight($de) 1
2845 } else {
2846 $w insert e:$ix $str
2847 $w insert e:$ix $e [highlight_tag $de]
2848 }
2849 }
2850 $w mark gravity e:$ix right
2851 $w conf -state disabled
2852 set treediropen($dir) 1
2853 set top [lindex [split [$w index @0,0] .] 0]
2854 set ht [$w cget -height]
2855 set l [lindex [split [$w index s:$ix] .] 0]
2856 if {$l < $top} {
2857 $w yview $l.0
2858 } elseif {$l + $n + 1 > $top + $ht} {
2859 set top [expr {$l + $n + 2 - $ht}]
2860 if {$l < $top} {
2861 set top $l
2862 }
2863 $w yview $top.0
2864 }
2865}
2866
2867proc treeclick {w x y} {
2868 global treediropen cmitmode ctext cflist cflist_top
2869
2870 if {$cmitmode ne "tree"} return
2871 if {![info exists cflist_top]} return
2872 set l [lindex [split [$w index "@$x,$y"] "."] 0]
2873 $cflist tag remove highlight $cflist_top.0 "$cflist_top.0 lineend"
2874 $cflist tag add highlight $l.0 "$l.0 lineend"
2875 set cflist_top $l
2876 if {$l == 1} {
2877 $ctext yview 1.0
2878 return
2879 }
2880 set e [linetoelt $l]
2881 if {[string index $e end] ne "/"} {
2882 showfile $e
2883 } elseif {$treediropen($e)} {
2884 treeclosedir $w $e
2885 } else {
2886 treeopendir $w $e
2887 }
2888}
2889
2890proc setfilelist {id} {
2891 global treefilelist cflist jump_to_here
2892
2893 treeview $cflist $treefilelist($id) 0
2894 if {$jump_to_here ne {}} {
2895 set f [lindex $jump_to_here 0]
2896 if {[lsearch -exact $treefilelist($id) $f] >= 0} {
2897 showfile $f
2898 }
2899 }
2900}
2901
2902image create bitmap tri-rt -background black -foreground blue -data {
2903 #define tri-rt_width 13
2904 #define tri-rt_height 13
2905 static unsigned char tri-rt_bits[] = {
2906 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x30, 0x00, 0x70, 0x00, 0xf0, 0x00,
2907 0xf0, 0x01, 0xf0, 0x00, 0x70, 0x00, 0x30, 0x00, 0x10, 0x00, 0x00, 0x00,
2908 0x00, 0x00};
2909} -maskdata {
2910 #define tri-rt-mask_width 13
2911 #define tri-rt-mask_height 13
2912 static unsigned char tri-rt-mask_bits[] = {
2913 0x08, 0x00, 0x18, 0x00, 0x38, 0x00, 0x78, 0x00, 0xf8, 0x00, 0xf8, 0x01,
2914 0xf8, 0x03, 0xf8, 0x01, 0xf8, 0x00, 0x78, 0x00, 0x38, 0x00, 0x18, 0x00,
2915 0x08, 0x00};
2916}
2917image create bitmap tri-dn -background black -foreground blue -data {
2918 #define tri-dn_width 13
2919 #define tri-dn_height 13
2920 static unsigned char tri-dn_bits[] = {
2921 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xfc, 0x07, 0xf8, 0x03,
2922 0xf0, 0x01, 0xe0, 0x00, 0x40, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
2923 0x00, 0x00};
2924} -maskdata {
2925 #define tri-dn-mask_width 13
2926 #define tri-dn-mask_height 13
2927 static unsigned char tri-dn-mask_bits[] = {
2928 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0x1f, 0xfe, 0x0f, 0xfc, 0x07,
2929 0xf8, 0x03, 0xf0, 0x01, 0xe0, 0x00, 0x40, 0x00, 0x00, 0x00, 0x00, 0x00,
2930 0x00, 0x00};
2931}
2932
2933image create bitmap reficon-T -background black -foreground yellow -data {
2934 #define tagicon_width 13
2935 #define tagicon_height 9
2936 static unsigned char tagicon_bits[] = {
2937 0x00, 0x00, 0x00, 0x00, 0xf0, 0x07, 0xf8, 0x07,
2938 0xfc, 0x07, 0xf8, 0x07, 0xf0, 0x07, 0x00, 0x00, 0x00, 0x00};
2939} -maskdata {
2940 #define tagicon-mask_width 13
2941 #define tagicon-mask_height 9
2942 static unsigned char tagicon-mask_bits[] = {
2943 0x00, 0x00, 0xf0, 0x0f, 0xf8, 0x0f, 0xfc, 0x0f,
2944 0xfe, 0x0f, 0xfc, 0x0f, 0xf8, 0x0f, 0xf0, 0x0f, 0x00, 0x00};
2945}
2946set rectdata {
2947 #define headicon_width 13
2948 #define headicon_height 9
2949 static unsigned char headicon_bits[] = {
2950 0x00, 0x00, 0x00, 0x00, 0xf8, 0x07, 0xf8, 0x07,
2951 0xf8, 0x07, 0xf8, 0x07, 0xf8, 0x07, 0x00, 0x00, 0x00, 0x00};
2952}
2953set rectmask {
2954 #define headicon-mask_width 13
2955 #define headicon-mask_height 9
2956 static unsigned char headicon-mask_bits[] = {
2957 0x00, 0x00, 0xfc, 0x0f, 0xfc, 0x0f, 0xfc, 0x0f,
2958 0xfc, 0x0f, 0xfc, 0x0f, 0xfc, 0x0f, 0xfc, 0x0f, 0x00, 0x00};
2959}
2960image create bitmap reficon-H -background black -foreground green \
2961 -data $rectdata -maskdata $rectmask
2962image create bitmap reficon-o -background black -foreground "#ddddff" \
2963 -data $rectdata -maskdata $rectmask
2964
2965proc init_flist {first} {
2966 global cflist cflist_top difffilestart
2967
2968 $cflist conf -state normal
2969 $cflist delete 0.0 end
2970 if {$first ne {}} {
2971 $cflist insert end $first
2972 set cflist_top 1
2973 $cflist tag add highlight 1.0 "1.0 lineend"
2974 } else {
2975 catch {unset cflist_top}
2976 }
2977 $cflist conf -state disabled
2978 set difffilestart {}
2979}
2980
2981proc highlight_tag {f} {
2982 global highlight_paths
2983
2984 foreach p $highlight_paths {
2985 if {[string match $p $f]} {
2986 return "bold"
2987 }
2988 }
2989 return {}
2990}
2991
2992proc highlight_filelist {} {
2993 global cmitmode cflist
2994
2995 $cflist conf -state normal
2996 if {$cmitmode ne "tree"} {
2997 set end [lindex [split [$cflist index end] .] 0]
2998 for {set l 2} {$l < $end} {incr l} {
2999 set line [$cflist get $l.0 "$l.0 lineend"]
3000 if {[highlight_tag $line] ne {}} {
3001 $cflist tag add bold $l.0 "$l.0 lineend"
3002 }
3003 }
3004 } else {
3005 highlight_tree 2 {}
3006 }
3007 $cflist conf -state disabled
3008}
3009
3010proc unhighlight_filelist {} {
3011 global cflist
3012
3013 $cflist conf -state normal
3014 $cflist tag remove bold 1.0 end
3015 $cflist conf -state disabled
3016}
3017
3018proc add_flist {fl} {
3019 global cflist
3020
3021 $cflist conf -state normal
3022 foreach f $fl {
3023 $cflist insert end "\n"
3024 $cflist insert end $f [highlight_tag $f]
3025 }
3026 $cflist conf -state disabled
3027}
3028
3029proc sel_flist {w x y} {
3030 global ctext difffilestart cflist cflist_top cmitmode
3031
3032 if {$cmitmode eq "tree"} return
3033 if {![info exists cflist_top]} return
3034 set l [lindex [split [$w index "@$x,$y"] "."] 0]
3035 $cflist tag remove highlight $cflist_top.0 "$cflist_top.0 lineend"
3036 $cflist tag add highlight $l.0 "$l.0 lineend"
3037 set cflist_top $l
3038 if {$l == 1} {
3039 $ctext yview 1.0
3040 } else {
3041 catch {$ctext yview [lindex $difffilestart [expr {$l - 2}]]}
3042 }
3043}
3044
3045proc pop_flist_menu {w X Y x y} {
3046 global ctext cflist cmitmode flist_menu flist_menu_file
3047 global treediffs diffids
3048
3049 stopfinding
3050 set l [lindex [split [$w index "@$x,$y"] "."] 0]
3051 if {$l <= 1} return
3052 if {$cmitmode eq "tree"} {
3053 set e [linetoelt $l]
3054 if {[string index $e end] eq "/"} return
3055 } else {
3056 set e [lindex $treediffs($diffids) [expr {$l-2}]]
3057 }
3058 set flist_menu_file $e
3059 set xdiffstate "normal"
3060 if {$cmitmode eq "tree"} {
3061 set xdiffstate "disabled"
3062 }
3063 # Disable "External diff" item in tree mode
3064 $flist_menu entryconf 2 -state $xdiffstate
3065 tk_popup $flist_menu $X $Y
3066}
3067
3068proc find_ctext_fileinfo {line} {
3069 global ctext_file_names ctext_file_lines
3070
3071 set ok [bsearch $ctext_file_lines $line]
3072 set tline [lindex $ctext_file_lines $ok]
3073
3074 if {$ok >= [llength $ctext_file_lines] || $line < $tline} {
3075 return {}
3076 } else {
3077 return [list [lindex $ctext_file_names $ok] $tline]
3078 }
3079}
3080
3081proc pop_diff_menu {w X Y x y} {
3082 global ctext diff_menu flist_menu_file
3083 global diff_menu_txtpos diff_menu_line
3084 global diff_menu_filebase
3085
3086 set diff_menu_txtpos [split [$w index "@$x,$y"] "."]
3087 set diff_menu_line [lindex $diff_menu_txtpos 0]
3088 # don't pop up the menu on hunk-separator or file-separator lines
3089 if {[lsearch -glob [$ctext tag names $diff_menu_line.0] "*sep"] >= 0} {
3090 return
3091 }
3092 stopfinding
3093 set f [find_ctext_fileinfo $diff_menu_line]
3094 if {$f eq {}} return
3095 set flist_menu_file [lindex $f 0]
3096 set diff_menu_filebase [lindex $f 1]
3097 tk_popup $diff_menu $X $Y
3098}
3099
3100proc flist_hl {only} {
3101 global flist_menu_file findstring gdttype
3102
3103 set x [shellquote $flist_menu_file]
3104 if {$only || $findstring eq {} || $gdttype ne [mc "touching paths:"]} {
3105 set findstring $x
3106 } else {
3107 append findstring " " $x
3108 }
3109 set gdttype [mc "touching paths:"]
3110}
3111
3112proc save_file_from_commit {filename output what} {
3113 global nullfile
3114
3115 if {[catch {exec git show $filename -- > $output} err]} {
3116 if {[string match "fatal: bad revision *" $err]} {
3117 return $nullfile
3118 }
3119 error_popup "[mc "Error getting \"%s\" from %s:" $filename $what] $err"
3120 return {}
3121 }
3122 return $output
3123}
3124
3125proc external_diff_get_one_file {diffid filename diffdir} {
3126 global nullid nullid2 nullfile
3127 global gitdir
3128
3129 if {$diffid == $nullid} {
3130 set difffile [file join [file dirname $gitdir] $filename]
3131 if {[file exists $difffile]} {
3132 return $difffile
3133 }
3134 return $nullfile
3135 }
3136 if {$diffid == $nullid2} {
3137 set difffile [file join $diffdir "\[index\] [file tail $filename]"]
3138 return [save_file_from_commit :$filename $difffile index]
3139 }
3140 set difffile [file join $diffdir "\[$diffid\] [file tail $filename]"]
3141 return [save_file_from_commit $diffid:$filename $difffile \
3142 "revision $diffid"]
3143}
3144
3145proc external_diff {} {
3146 global gitktmpdir nullid nullid2
3147 global flist_menu_file
3148 global diffids
3149 global diffnum
3150 global gitdir extdifftool
3151
3152 if {[llength $diffids] == 1} {
3153 # no reference commit given
3154 set diffidto [lindex $diffids 0]
3155 if {$diffidto eq $nullid} {
3156 # diffing working copy with index
3157 set diffidfrom $nullid2
3158 } elseif {$diffidto eq $nullid2} {
3159 # diffing index with HEAD
3160 set diffidfrom "HEAD"
3161 } else {
3162 # use first parent commit
3163 global parentlist selectedline
3164 set diffidfrom [lindex $parentlist $selectedline 0]
3165 }
3166 } else {
3167 set diffidfrom [lindex $diffids 0]
3168 set diffidto [lindex $diffids 1]
3169 }
3170
3171 # make sure that several diffs wont collide
3172 if {![info exists gitktmpdir]} {
3173 set gitktmpdir [file join [file dirname $gitdir] \
3174 [format ".gitk-tmp.%s" [pid]]]
3175 if {[catch {file mkdir $gitktmpdir} err]} {
3176 error_popup "[mc "Error creating temporary directory %s:" $gitktmpdir] $err"
3177 unset gitktmpdir
3178 return
3179 }
3180 set diffnum 0
3181 }
3182 incr diffnum
3183 set diffdir [file join $gitktmpdir $diffnum]
3184 if {[catch {file mkdir $diffdir} err]} {
3185 error_popup "[mc "Error creating temporary directory %s:" $diffdir] $err"
3186 return
3187 }
3188
3189 # gather files to diff
3190 set difffromfile [external_diff_get_one_file $diffidfrom $flist_menu_file $diffdir]
3191 set difftofile [external_diff_get_one_file $diffidto $flist_menu_file $diffdir]
3192
3193 if {$difffromfile ne {} && $difftofile ne {}} {
3194 set cmd [concat | [shellsplit $extdifftool] \
3195 [list $difffromfile $difftofile]]
3196 if {[catch {set fl [open $cmd r]} err]} {
3197 file delete -force $diffdir
3198 error_popup "$extdifftool: [mc "command failed:"] $err"
3199 } else {
3200 fconfigure $fl -blocking 0
3201 filerun $fl [list delete_at_eof $fl $diffdir]
3202 }
3203 }
3204}
3205
3206proc find_hunk_blamespec {base line} {
3207 global ctext
3208
3209 # Find and parse the hunk header
3210 set s_lix [$ctext search -backwards -regexp ^@@ "$line.0 lineend" $base.0]
3211 if {$s_lix eq {}} return
3212
3213 set s_line [$ctext get $s_lix "$s_lix + 1 lines"]
3214 if {![regexp {^@@@*(( -\d+(,\d+)?)+) \+(\d+)(,\d+)? @@} $s_line \
3215 s_line old_specs osz osz1 new_line nsz]} {
3216 return
3217 }
3218
3219 # base lines for the parents
3220 set base_lines [list $new_line]
3221 foreach old_spec [lrange [split $old_specs " "] 1 end] {
3222 if {![regexp -- {-(\d+)(,\d+)?} $old_spec \
3223 old_spec old_line osz]} {
3224 return
3225 }
3226 lappend base_lines $old_line
3227 }
3228
3229 # Now scan the lines to determine offset within the hunk
3230 set max_parent [expr {[llength $base_lines]-2}]
3231 set dline 0
3232 set s_lno [lindex [split $s_lix "."] 0]
3233
3234 # Determine if the line is removed
3235 set chunk [$ctext get $line.0 "$line.1 + $max_parent chars"]
3236 if {[string match {[-+ ]*} $chunk]} {
3237 set removed_idx [string first "-" $chunk]
3238 # Choose a parent index
3239 if {$removed_idx >= 0} {
3240 set parent $removed_idx
3241 } else {
3242 set unchanged_idx [string first " " $chunk]
3243 if {$unchanged_idx >= 0} {
3244 set parent $unchanged_idx
3245 } else {
3246 # blame the current commit
3247 set parent -1
3248 }
3249 }
3250 # then count other lines that belong to it
3251 for {set i $line} {[incr i -1] > $s_lno} {} {
3252 set chunk [$ctext get $i.0 "$i.1 + $max_parent chars"]
3253 # Determine if the line is removed
3254 set removed_idx [string first "-" $chunk]
3255 if {$parent >= 0} {
3256 set code [string index $chunk $parent]
3257 if {$code eq "-" || ($removed_idx < 0 && $code ne "+")} {
3258 incr dline
3259 }
3260 } else {
3261 if {$removed_idx < 0} {
3262 incr dline
3263 }
3264 }
3265 }
3266 incr parent
3267 } else {
3268 set parent 0
3269 }
3270
3271 incr dline [lindex $base_lines $parent]
3272 return [list $parent $dline]
3273}
3274
3275proc external_blame_diff {} {
3276 global currentid cmitmode
3277 global diff_menu_txtpos diff_menu_line
3278 global diff_menu_filebase flist_menu_file
3279
3280 if {$cmitmode eq "tree"} {
3281 set parent_idx 0
3282 set line [expr {$diff_menu_line - $diff_menu_filebase}]
3283 } else {
3284 set hinfo [find_hunk_blamespec $diff_menu_filebase $diff_menu_line]
3285 if {$hinfo ne {}} {
3286 set parent_idx [lindex $hinfo 0]
3287 set line [lindex $hinfo 1]
3288 } else {
3289 set parent_idx 0
3290 set line 0
3291 }
3292 }
3293
3294 external_blame $parent_idx $line
3295}
3296
3297# Find the SHA1 ID of the blob for file $fname in the index
3298# at stage 0 or 2
3299proc index_sha1 {fname} {
3300 set f [open [list | git ls-files -s $fname] r]
3301 while {[gets $f line] >= 0} {
3302 set info [lindex [split $line "\t"] 0]
3303 set stage [lindex $info 2]
3304 if {$stage eq "0" || $stage eq "2"} {
3305 close $f
3306 return [lindex $info 1]
3307 }
3308 }
3309 close $f
3310 return {}
3311}
3312
3313proc external_blame {parent_idx {line {}}} {
3314 global flist_menu_file
3315 global nullid nullid2
3316 global parentlist selectedline currentid
3317
3318 if {$parent_idx > 0} {
3319 set base_commit [lindex $parentlist $selectedline [expr {$parent_idx-1}]]
3320 } else {
3321 set base_commit $currentid
3322 }
3323
3324 if {$base_commit eq {} || $base_commit eq $nullid || $base_commit eq $nullid2} {
3325 error_popup [mc "No such commit"]
3326 return
3327 }
3328
3329 set cmdline [list git gui blame]
3330 if {$line ne {} && $line > 1} {
3331 lappend cmdline "--line=$line"
3332 }
3333 lappend cmdline $base_commit $flist_menu_file
3334 if {[catch {eval exec $cmdline &} err]} {
3335 error_popup "[mc "git gui blame: command failed:"] $err"
3336 }
3337}
3338
3339proc show_line_source {} {
3340 global cmitmode currentid parents curview blamestuff blameinst
3341 global diff_menu_line diff_menu_filebase flist_menu_file
3342 global nullid nullid2 gitdir
3343
3344 set from_index {}
3345 if {$cmitmode eq "tree"} {
3346 set id $currentid
3347 set line [expr {$diff_menu_line - $diff_menu_filebase}]
3348 } else {
3349 set h [find_hunk_blamespec $diff_menu_filebase $diff_menu_line]
3350 if {$h eq {}} return
3351 set pi [lindex $h 0]
3352 if {$pi == 0} {
3353 mark_ctext_line $diff_menu_line
3354 return
3355 }
3356 incr pi -1
3357 if {$currentid eq $nullid} {
3358 if {$pi > 0} {
3359 # must be a merge in progress...
3360 if {[catch {
3361 # get the last line from .git/MERGE_HEAD
3362 set f [open [file join $gitdir MERGE_HEAD] r]
3363 set id [lindex [split [read $f] "\n"] end-1]
3364 close $f
3365 } err]} {
3366 error_popup [mc "Couldn't read merge head: %s" $err]
3367 return
3368 }
3369 } elseif {$parents($curview,$currentid) eq $nullid2} {
3370 # need to do the blame from the index
3371 if {[catch {
3372 set from_index [index_sha1 $flist_menu_file]
3373 } err]} {
3374 error_popup [mc "Error reading index: %s" $err]
3375 return
3376 }
3377 }
3378 } else {
3379 set id [lindex $parents($curview,$currentid) $pi]
3380 }
3381 set line [lindex $h 1]
3382 }
3383 set blameargs {}
3384 if {$from_index ne {}} {
3385 lappend blameargs | git cat-file blob $from_index
3386 }
3387 lappend blameargs | git blame -p -L$line,+1
3388 if {$from_index ne {}} {
3389 lappend blameargs --contents -
3390 } else {
3391 lappend blameargs $id
3392 }
3393 lappend blameargs -- $flist_menu_file
3394 if {[catch {
3395 set f [open $blameargs r]
3396 } err]} {
3397 error_popup [mc "Couldn't start git blame: %s" $err]
3398 return
3399 }
3400 fconfigure $f -blocking 0
3401 set i [reg_instance $f]
3402 set blamestuff($i) {}
3403 set blameinst $i
3404 filerun $f [list read_line_source $f $i]
3405}
3406
3407proc stopblaming {} {
3408 global blameinst
3409
3410 if {[info exists blameinst]} {
3411 stop_instance $blameinst
3412 unset blameinst
3413 }
3414}
3415
3416proc read_line_source {fd inst} {
3417 global blamestuff curview commfd blameinst nullid nullid2
3418
3419 while {[gets $fd line] >= 0} {
3420 lappend blamestuff($inst) $line
3421 }
3422 if {![eof $fd]} {
3423 return 1
3424 }
3425 unset commfd($inst)
3426 unset blameinst
3427 fconfigure $fd -blocking 1
3428 if {[catch {close $fd} err]} {
3429 error_popup [mc "Error running git blame: %s" $err]
3430 return 0
3431 }
3432
3433 set fname {}
3434 set line [split [lindex $blamestuff($inst) 0] " "]
3435 set id [lindex $line 0]
3436 set lnum [lindex $line 1]
3437 if {[string length $id] == 40 && [string is xdigit $id] &&
3438 [string is digit -strict $lnum]} {
3439 # look for "filename" line
3440 foreach l $blamestuff($inst) {
3441 if {[string match "filename *" $l]} {
3442 set fname [string range $l 9 end]
3443 break
3444 }
3445 }
3446 }
3447 if {$fname ne {}} {
3448 # all looks good, select it
3449 if {$id eq $nullid} {
3450 # blame uses all-zeroes to mean not committed,
3451 # which would mean a change in the index
3452 set id $nullid2
3453 }
3454 if {[commitinview $id $curview]} {
3455 selectline [rowofcommit $id] 1 [list $fname $lnum]
3456 } else {
3457 error_popup [mc "That line comes from commit %s, \
3458 which is not in this view" [shortids $id]]
3459 }
3460 } else {
3461 puts "oops couldn't parse git blame output"
3462 }
3463 return 0
3464}
3465
3466# delete $dir when we see eof on $f (presumably because the child has exited)
3467proc delete_at_eof {f dir} {
3468 while {[gets $f line] >= 0} {}
3469 if {[eof $f]} {
3470 if {[catch {close $f} err]} {
3471 error_popup "[mc "External diff viewer failed:"] $err"
3472 }
3473 file delete -force $dir
3474 return 0
3475 }
3476 return 1
3477}
3478
3479# Functions for adding and removing shell-type quoting
3480
3481proc shellquote {str} {
3482 if {![string match "*\['\"\\ \t]*" $str]} {
3483 return $str
3484 }
3485 if {![string match "*\['\"\\]*" $str]} {
3486 return "\"$str\""
3487 }
3488 if {![string match "*'*" $str]} {
3489 return "'$str'"
3490 }
3491 return "\"[string map {\" \\\" \\ \\\\} $str]\""
3492}
3493
3494proc shellarglist {l} {
3495 set str {}
3496 foreach a $l {
3497 if {$str ne {}} {
3498 append str " "
3499 }
3500 append str [shellquote $a]
3501 }
3502 return $str
3503}
3504
3505proc shelldequote {str} {
3506 set ret {}
3507 set used -1
3508 while {1} {
3509 incr used
3510 if {![regexp -start $used -indices "\['\"\\\\ \t]" $str first]} {
3511 append ret [string range $str $used end]
3512 set used [string length $str]
3513 break
3514 }
3515 set first [lindex $first 0]
3516 set ch [string index $str $first]
3517 if {$first > $used} {
3518 append ret [string range $str $used [expr {$first - 1}]]
3519 set used $first
3520 }
3521 if {$ch eq " " || $ch eq "\t"} break
3522 incr used
3523 if {$ch eq "'"} {
3524 set first [string first "'" $str $used]
3525 if {$first < 0} {
3526 error "unmatched single-quote"
3527 }
3528 append ret [string range $str $used [expr {$first - 1}]]
3529 set used $first
3530 continue
3531 }
3532 if {$ch eq "\\"} {
3533 if {$used >= [string length $str]} {
3534 error "trailing backslash"
3535 }
3536 append ret [string index $str $used]
3537 continue
3538 }
3539 # here ch == "\""
3540 while {1} {
3541 if {![regexp -start $used -indices "\[\"\\\\]" $str first]} {
3542 error "unmatched double-quote"
3543 }
3544 set first [lindex $first 0]
3545 set ch [string index $str $first]
3546 if {$first > $used} {
3547 append ret [string range $str $used [expr {$first - 1}]]
3548 set used $first
3549 }
3550 if {$ch eq "\""} break
3551 incr used
3552 append ret [string index $str $used]
3553 incr used
3554 }
3555 }
3556 return [list $used $ret]
3557}
3558
3559proc shellsplit {str} {
3560 set l {}
3561 while {1} {
3562 set str [string trimleft $str]
3563 if {$str eq {}} break
3564 set dq [shelldequote $str]
3565 set n [lindex $dq 0]
3566 set word [lindex $dq 1]
3567 set str [string range $str $n end]
3568 lappend l $word
3569 }
3570 return $l
3571}
3572
3573# Code to implement multiple views
3574
3575proc newview {ishighlight} {
3576 global nextviewnum newviewname newishighlight
3577 global revtreeargs viewargscmd newviewopts curview
3578
3579 set newishighlight $ishighlight
3580 set top .gitkview
3581 if {[winfo exists $top]} {
3582 raise $top
3583 return
3584 }
3585 set newviewname($nextviewnum) "[mc "View"] $nextviewnum"
3586 set newviewopts($nextviewnum,perm) 0
3587 set newviewopts($nextviewnum,cmd) $viewargscmd($curview)
3588 decode_view_opts $nextviewnum $revtreeargs
3589 vieweditor $top $nextviewnum [mc "Gitk view definition"]
3590}
3591
3592set known_view_options {
3593 {perm b . {} {mc "Remember this view"}}
3594 {args t50= + {} {mc "Commits to include (arguments to git log):"}}
3595 {all b * "--all" {mc "Use all refs"}}
3596 {dorder b . {"--date-order" "-d"} {mc "Strictly sort by date"}}
3597 {lright b . "--left-right" {mc "Mark branch sides"}}
3598 {since t15 + {"--since=*" "--after=*"} {mc "Since date:"}}
3599 {until t15 . {"--until=*" "--before=*"} {mc "Until date:"}}
3600 {limit t10 + "--max-count=*" {mc "Max count:"}}
3601 {skip t10 . "--skip=*" {mc "Skip:"}}
3602 {first b . "--first-parent" {mc "Limit to first parent"}}
3603 {cmd t50= + {} {mc "Command to generate more commits to include:"}}
3604 }
3605
3606proc encode_view_opts {n} {
3607 global known_view_options newviewopts
3608
3609 set rargs [list]
3610 foreach opt $known_view_options {
3611 set patterns [lindex $opt 3]
3612 if {$patterns eq {}} continue
3613 set pattern [lindex $patterns 0]
3614
3615 set val $newviewopts($n,[lindex $opt 0])
3616
3617 if {[lindex $opt 1] eq "b"} {
3618 if {$val} {
3619 lappend rargs $pattern
3620 }
3621 } else {
3622 set val [string trim $val]
3623 if {$val ne {}} {
3624 set pfix [string range $pattern 0 end-1]
3625 lappend rargs $pfix$val
3626 }
3627 }
3628 }
3629 return [concat $rargs [shellsplit $newviewopts($n,args)]]
3630}
3631
3632proc decode_view_opts {n view_args} {
3633 global known_view_options newviewopts
3634
3635 foreach opt $known_view_options {
3636 if {[lindex $opt 1] eq "b"} {
3637 set val 0
3638 } else {
3639 set val {}
3640 }
3641 set newviewopts($n,[lindex $opt 0]) $val
3642 }
3643 set oargs [list]
3644 foreach arg $view_args {
3645 if {[regexp -- {^-([0-9]+)$} $arg arg cnt]
3646 && ![info exists found(limit)]} {
3647 set newviewopts($n,limit) $cnt
3648 set found(limit) 1
3649 continue
3650 }
3651 catch { unset val }
3652 foreach opt $known_view_options {
3653 set id [lindex $opt 0]
3654 if {[info exists found($id)]} continue
3655 foreach pattern [lindex $opt 3] {
3656 if {![string match $pattern $arg]} continue
3657 if {[lindex $opt 1] ne "b"} {
3658 set size [string length $pattern]
3659 set val [string range $arg [expr {$size-1}] end]
3660 } else {
3661 set val 1
3662 }
3663 set newviewopts($n,$id) $val
3664 set found($id) 1
3665 break
3666 }
3667 if {[info exists val]} break
3668 }
3669 if {[info exists val]} continue
3670 lappend oargs $arg
3671 }
3672 set newviewopts($n,args) [shellarglist $oargs]
3673}
3674
3675proc edit_or_newview {} {
3676 global curview
3677
3678 if {$curview > 0} {
3679 editview
3680 } else {
3681 newview 0
3682 }
3683}
3684
3685proc editview {} {
3686 global curview
3687 global viewname viewperm newviewname newviewopts
3688 global viewargs viewargscmd
3689
3690 set top .gitkvedit-$curview
3691 if {[winfo exists $top]} {
3692 raise $top
3693 return
3694 }
3695 set newviewname($curview) $viewname($curview)
3696 set newviewopts($curview,perm) $viewperm($curview)
3697 set newviewopts($curview,cmd) $viewargscmd($curview)
3698 decode_view_opts $curview $viewargs($curview)
3699 vieweditor $top $curview "Gitk: edit view $viewname($curview)"
3700}
3701
3702proc vieweditor {top n title} {
3703 global newviewname newviewopts viewfiles bgcolor
3704 global known_view_options
3705
3706 toplevel $top
3707 wm title $top $title
3708 make_transient $top .
3709
3710 # View name
3711 frame $top.nfr
3712 label $top.nl -text [mc "Name"]
3713 entry $top.name -width 20 -textvariable newviewname($n)
3714 pack $top.nfr -in $top -fill x -pady 5 -padx 3
3715 pack $top.nl -in $top.nfr -side left -padx {0 30}
3716 pack $top.name -in $top.nfr -side left
3717
3718 # View options
3719 set cframe $top.nfr
3720 set cexpand 0
3721 set cnt 0
3722 foreach opt $known_view_options {
3723 set id [lindex $opt 0]
3724 set type [lindex $opt 1]
3725 set flags [lindex $opt 2]
3726 set title [eval [lindex $opt 4]]
3727 set lxpad 0
3728
3729 if {$flags eq "+" || $flags eq "*"} {
3730 set cframe $top.fr$cnt
3731 incr cnt
3732 frame $cframe
3733 pack $cframe -in $top -fill x -pady 3 -padx 3
3734 set cexpand [expr {$flags eq "*"}]
3735 } else {
3736 set lxpad 5
3737 }
3738
3739 if {$type eq "b"} {
3740 checkbutton $cframe.c_$id -text $title -variable newviewopts($n,$id)
3741 pack $cframe.c_$id -in $cframe -side left \
3742 -padx [list $lxpad 0] -expand $cexpand -anchor w
3743 } elseif {[regexp {^t(\d+)$} $type type sz]} {
3744 message $cframe.l_$id -aspect 1500 -text $title
3745 entry $cframe.e_$id -width $sz -background $bgcolor \
3746 -textvariable newviewopts($n,$id)
3747 pack $cframe.l_$id -in $cframe -side left -padx [list $lxpad 0]
3748 pack $cframe.e_$id -in $cframe -side left -expand 1 -fill x
3749 } elseif {[regexp {^t(\d+)=$} $type type sz]} {
3750 message $cframe.l_$id -aspect 1500 -text $title
3751 entry $cframe.e_$id -width $sz -background $bgcolor \
3752 -textvariable newviewopts($n,$id)
3753 pack $cframe.l_$id -in $cframe -side top -pady [list 3 0] -anchor w
3754 pack $cframe.e_$id -in $cframe -side top -fill x
3755 }
3756 }
3757
3758 # Path list
3759 message $top.l -aspect 1500 \
3760 -text [mc "Enter files and directories to include, one per line:"]
3761 pack $top.l -in $top -side top -pady [list 7 0] -anchor w -padx 3
3762 text $top.t -width 40 -height 5 -background $bgcolor -font uifont
3763 if {[info exists viewfiles($n)]} {
3764 foreach f $viewfiles($n) {
3765 $top.t insert end $f
3766 $top.t insert end "\n"
3767 }
3768 $top.t delete {end - 1c} end
3769 $top.t mark set insert 0.0
3770 }
3771 pack $top.t -in $top -side top -pady [list 0 5] -fill both -expand 1 -padx 3
3772 frame $top.buts
3773 button $top.buts.ok -text [mc "OK"] -command [list newviewok $top $n]
3774 button $top.buts.apply -text [mc "Apply (F5)"] -command [list newviewok $top $n 1]
3775 button $top.buts.can -text [mc "Cancel"] -command [list destroy $top]
3776 bind $top <Control-Return> [list newviewok $top $n]
3777 bind $top <F5> [list newviewok $top $n 1]
3778 bind $top <Escape> [list destroy $top]
3779 grid $top.buts.ok $top.buts.apply $top.buts.can
3780 grid columnconfigure $top.buts 0 -weight 1 -uniform a
3781 grid columnconfigure $top.buts 1 -weight 1 -uniform a
3782 grid columnconfigure $top.buts 2 -weight 1 -uniform a
3783 pack $top.buts -in $top -side top -fill x
3784 focus $top.t
3785}
3786
3787proc doviewmenu {m first cmd op argv} {
3788 set nmenu [$m index end]
3789 for {set i $first} {$i <= $nmenu} {incr i} {
3790 if {[$m entrycget $i -command] eq $cmd} {
3791 eval $m $op $i $argv
3792 break
3793 }
3794 }
3795}
3796
3797proc allviewmenus {n op args} {
3798 # global viewhlmenu
3799
3800 doviewmenu .bar.view 5 [list showview $n] $op $args
3801 # doviewmenu $viewhlmenu 1 [list addvhighlight $n] $op $args
3802}
3803
3804proc newviewok {top n {apply 0}} {
3805 global nextviewnum newviewperm newviewname newishighlight
3806 global viewname viewfiles viewperm selectedview curview
3807 global viewargs viewargscmd newviewopts viewhlmenu
3808
3809 if {[catch {
3810 set newargs [encode_view_opts $n]
3811 } err]} {
3812 error_popup "[mc "Error in commit selection arguments:"] $err" $top
3813 return
3814 }
3815 set files {}
3816 foreach f [split [$top.t get 0.0 end] "\n"] {
3817 set ft [string trim $f]
3818 if {$ft ne {}} {
3819 lappend files $ft
3820 }
3821 }
3822 if {![info exists viewfiles($n)]} {
3823 # creating a new view
3824 incr nextviewnum
3825 set viewname($n) $newviewname($n)
3826 set viewperm($n) $newviewopts($n,perm)
3827 set viewfiles($n) $files
3828 set viewargs($n) $newargs
3829 set viewargscmd($n) $newviewopts($n,cmd)
3830 addviewmenu $n
3831 if {!$newishighlight} {
3832 run showview $n
3833 } else {
3834 run addvhighlight $n
3835 }
3836 } else {
3837 # editing an existing view
3838 set viewperm($n) $newviewopts($n,perm)
3839 if {$newviewname($n) ne $viewname($n)} {
3840 set viewname($n) $newviewname($n)
3841 doviewmenu .bar.view 5 [list showview $n] \
3842 entryconf [list -label $viewname($n)]
3843 # doviewmenu $viewhlmenu 1 [list addvhighlight $n] \
3844 # entryconf [list -label $viewname($n) -value $viewname($n)]
3845 }
3846 if {$files ne $viewfiles($n) || $newargs ne $viewargs($n) || \
3847 $newviewopts($n,cmd) ne $viewargscmd($n)} {
3848 set viewfiles($n) $files
3849 set viewargs($n) $newargs
3850 set viewargscmd($n) $newviewopts($n,cmd)
3851 if {$curview == $n} {
3852 run reloadcommits
3853 }
3854 }
3855 }
3856 if {$apply} return
3857 catch {destroy $top}
3858}
3859
3860proc delview {} {
3861 global curview viewperm hlview selectedhlview
3862
3863 if {$curview == 0} return
3864 if {[info exists hlview] && $hlview == $curview} {
3865 set selectedhlview [mc "None"]
3866 unset hlview
3867 }
3868 allviewmenus $curview delete
3869 set viewperm($curview) 0
3870 showview 0
3871}
3872
3873proc addviewmenu {n} {
3874 global viewname viewhlmenu
3875
3876 .bar.view add radiobutton -label $viewname($n) \
3877 -command [list showview $n] -variable selectedview -value $n
3878 #$viewhlmenu add radiobutton -label $viewname($n) \
3879 # -command [list addvhighlight $n] -variable selectedhlview
3880}
3881
3882proc showview {n} {
3883 global curview cached_commitrow ordertok
3884 global displayorder parentlist rowidlist rowisopt rowfinal
3885 global colormap rowtextx nextcolor canvxmax
3886 global numcommits viewcomplete
3887 global selectedline currentid canv canvy0
3888 global treediffs
3889 global pending_select mainheadid
3890 global commitidx
3891 global selectedview
3892 global hlview selectedhlview commitinterest
3893
3894 if {$n == $curview} return
3895 set selid {}
3896 set ymax [lindex [$canv cget -scrollregion] 3]
3897 set span [$canv yview]
3898 set ytop [expr {[lindex $span 0] * $ymax}]
3899 set ybot [expr {[lindex $span 1] * $ymax}]
3900 set yscreen [expr {($ybot - $ytop) / 2}]
3901 if {$selectedline ne {}} {
3902 set selid $currentid
3903 set y [yc $selectedline]
3904 if {$ytop < $y && $y < $ybot} {
3905 set yscreen [expr {$y - $ytop}]
3906 }
3907 } elseif {[info exists pending_select]} {
3908 set selid $pending_select
3909 unset pending_select
3910 }
3911 unselectline
3912 normalline
3913 catch {unset treediffs}
3914 clear_display
3915 if {[info exists hlview] && $hlview == $n} {
3916 unset hlview
3917 set selectedhlview [mc "None"]
3918 }
3919 catch {unset commitinterest}
3920 catch {unset cached_commitrow}
3921 catch {unset ordertok}
3922
3923 set curview $n
3924 set selectedview $n
3925 .bar.view entryconf [mca "Edit view..."] -state [expr {$n == 0? "disabled": "normal"}]
3926 .bar.view entryconf [mca "Delete view"] -state [expr {$n == 0? "disabled": "normal"}]
3927
3928 run refill_reflist
3929 if {![info exists viewcomplete($n)]} {
3930 getcommits $selid
3931 return
3932 }
3933
3934 set displayorder {}
3935 set parentlist {}
3936 set rowidlist {}
3937 set rowisopt {}
3938 set rowfinal {}
3939 set numcommits $commitidx($n)
3940
3941 catch {unset colormap}
3942 catch {unset rowtextx}
3943 set nextcolor 0
3944 set canvxmax [$canv cget -width]
3945 set curview $n
3946 set row 0
3947 setcanvscroll
3948 set yf 0
3949 set row {}
3950 if {$selid ne {} && [commitinview $selid $n]} {
3951 set row [rowofcommit $selid]
3952 # try to get the selected row in the same position on the screen
3953 set ymax [lindex [$canv cget -scrollregion] 3]
3954 set ytop [expr {[yc $row] - $yscreen}]
3955 if {$ytop < 0} {
3956 set ytop 0
3957 }
3958 set yf [expr {$ytop * 1.0 / $ymax}]
3959 }
3960 allcanvs yview moveto $yf
3961 drawvisible
3962 if {$row ne {}} {
3963 selectline $row 0
3964 } elseif {!$viewcomplete($n)} {
3965 reset_pending_select $selid
3966 } else {
3967 reset_pending_select {}
3968
3969 if {[commitinview $pending_select $curview]} {
3970 selectline [rowofcommit $pending_select] 1
3971 } else {
3972 set row [first_real_row]
3973 if {$row < $numcommits} {
3974 selectline $row 0
3975 }
3976 }
3977 }
3978 if {!$viewcomplete($n)} {
3979 if {$numcommits == 0} {
3980 show_status [mc "Reading commits..."]
3981 }
3982 } elseif {$numcommits == 0} {
3983 show_status [mc "No commits selected"]
3984 }
3985}
3986
3987# Stuff relating to the highlighting facility
3988
3989proc ishighlighted {id} {
3990 global vhighlights fhighlights nhighlights rhighlights
3991
3992 if {[info exists nhighlights($id)] && $nhighlights($id) > 0} {
3993 return $nhighlights($id)
3994 }
3995 if {[info exists vhighlights($id)] && $vhighlights($id) > 0} {
3996 return $vhighlights($id)
3997 }
3998 if {[info exists fhighlights($id)] && $fhighlights($id) > 0} {
3999 return $fhighlights($id)
4000 }
4001 if {[info exists rhighlights($id)] && $rhighlights($id) > 0} {
4002 return $rhighlights($id)
4003 }
4004 return 0
4005}
4006
4007proc bolden {id font} {
4008 global canv linehtag currentid boldids need_redisplay
4009
4010 # need_redisplay = 1 means the display is stale and about to be redrawn
4011 if {$need_redisplay} return
4012 lappend boldids $id
4013 $canv itemconf $linehtag($id) -font $font
4014 if {[info exists currentid] && $id eq $currentid} {
4015 $canv delete secsel
4016 set t [eval $canv create rect [$canv bbox $linehtag($id)] \
4017 -outline {{}} -tags secsel \
4018 -fill [$canv cget -selectbackground]]
4019 $canv lower $t
4020 }
4021}
4022
4023proc bolden_name {id font} {
4024 global canv2 linentag currentid boldnameids need_redisplay
4025
4026 if {$need_redisplay} return
4027 lappend boldnameids $id
4028 $canv2 itemconf $linentag($id) -font $font
4029 if {[info exists currentid] && $id eq $currentid} {
4030 $canv2 delete secsel
4031 set t [eval $canv2 create rect [$canv2 bbox $linentag($id)] \
4032 -outline {{}} -tags secsel \
4033 -fill [$canv2 cget -selectbackground]]
4034 $canv2 lower $t
4035 }
4036}
4037
4038proc unbolden {} {
4039 global boldids
4040
4041 set stillbold {}
4042 foreach id $boldids {
4043 if {![ishighlighted $id]} {
4044 bolden $id mainfont
4045 } else {
4046 lappend stillbold $id
4047 }
4048 }
4049 set boldids $stillbold
4050}
4051
4052proc addvhighlight {n} {
4053 global hlview viewcomplete curview vhl_done commitidx
4054
4055 if {[info exists hlview]} {
4056 delvhighlight
4057 }
4058 set hlview $n
4059 if {$n != $curview && ![info exists viewcomplete($n)]} {
4060 start_rev_list $n
4061 }
4062 set vhl_done $commitidx($hlview)
4063 if {$vhl_done > 0} {
4064 drawvisible
4065 }
4066}
4067
4068proc delvhighlight {} {
4069 global hlview vhighlights
4070
4071 if {![info exists hlview]} return
4072 unset hlview
4073 catch {unset vhighlights}
4074 unbolden
4075}
4076
4077proc vhighlightmore {} {
4078 global hlview vhl_done commitidx vhighlights curview
4079
4080 set max $commitidx($hlview)
4081 set vr [visiblerows]
4082 set r0 [lindex $vr 0]
4083 set r1 [lindex $vr 1]
4084 for {set i $vhl_done} {$i < $max} {incr i} {
4085 set id [commitonrow $i $hlview]
4086 if {[commitinview $id $curview]} {
4087 set row [rowofcommit $id]
4088 if {$r0 <= $row && $row <= $r1} {
4089 if {![highlighted $row]} {
4090 bolden $id mainfontbold
4091 }
4092 set vhighlights($id) 1
4093 }
4094 }
4095 }
4096 set vhl_done $max
4097 return 0
4098}
4099
4100proc askvhighlight {row id} {
4101 global hlview vhighlights iddrawn
4102
4103 if {[commitinview $id $hlview]} {
4104 if {[info exists iddrawn($id)] && ![ishighlighted $id]} {
4105 bolden $id mainfontbold
4106 }
4107 set vhighlights($id) 1
4108 } else {
4109 set vhighlights($id) 0
4110 }
4111}
4112
4113proc hfiles_change {} {
4114 global highlight_files filehighlight fhighlights fh_serial
4115 global highlight_paths gdttype
4116
4117 if {[info exists filehighlight]} {
4118 # delete previous highlights
4119 catch {close $filehighlight}
4120 unset filehighlight
4121 catch {unset fhighlights}
4122 unbolden
4123 unhighlight_filelist
4124 }
4125 set highlight_paths {}
4126 after cancel do_file_hl $fh_serial
4127 incr fh_serial
4128 if {$highlight_files ne {}} {
4129 after 300 do_file_hl $fh_serial
4130 }
4131}
4132
4133proc gdttype_change {name ix op} {
4134 global gdttype highlight_files findstring findpattern
4135
4136 stopfinding
4137 if {$findstring ne {}} {
4138 if {$gdttype eq [mc "containing:"]} {
4139 if {$highlight_files ne {}} {
4140 set highlight_files {}
4141 hfiles_change
4142 }
4143 findcom_change
4144 } else {
4145 if {$findpattern ne {}} {
4146 set findpattern {}
4147 findcom_change
4148 }
4149 set highlight_files $findstring
4150 hfiles_change
4151 }
4152 drawvisible
4153 }
4154 # enable/disable findtype/findloc menus too
4155}
4156
4157proc find_change {name ix op} {
4158 global gdttype findstring highlight_files
4159
4160 stopfinding
4161 if {$gdttype eq [mc "containing:"]} {
4162 findcom_change
4163 } else {
4164 if {$highlight_files ne $findstring} {
4165 set highlight_files $findstring
4166 hfiles_change
4167 }
4168 }
4169 drawvisible
4170}
4171
4172proc findcom_change args {
4173 global nhighlights boldnameids
4174 global findpattern findtype findstring gdttype
4175
4176 stopfinding
4177 # delete previous highlights, if any
4178 foreach id $boldnameids {
4179 bolden_name $id mainfont
4180 }
4181 set boldnameids {}
4182 catch {unset nhighlights}
4183 unbolden
4184 unmarkmatches
4185 if {$gdttype ne [mc "containing:"] || $findstring eq {}} {
4186 set findpattern {}
4187 } elseif {$findtype eq [mc "Regexp"]} {
4188 set findpattern $findstring
4189 } else {
4190 set e [string map {"*" "\\*" "?" "\\?" "\[" "\\\[" "\\" "\\\\"} \
4191 $findstring]
4192 set findpattern "*$e*"
4193 }
4194}
4195
4196proc makepatterns {l} {
4197 set ret {}
4198 foreach e $l {
4199 set ee [string map {"*" "\\*" "?" "\\?" "\[" "\\\[" "\\" "\\\\"} $e]
4200 if {[string index $ee end] eq "/"} {
4201 lappend ret "$ee*"
4202 } else {
4203 lappend ret $ee
4204 lappend ret "$ee/*"
4205 }
4206 }
4207 return $ret
4208}
4209
4210proc do_file_hl {serial} {
4211 global highlight_files filehighlight highlight_paths gdttype fhl_list
4212
4213 if {$gdttype eq [mc "touching paths:"]} {
4214 if {[catch {set paths [shellsplit $highlight_files]}]} return
4215 set highlight_paths [makepatterns $paths]
4216 highlight_filelist
4217 set gdtargs [concat -- $paths]
4218 } elseif {$gdttype eq [mc "adding/removing string:"]} {
4219 set gdtargs [list "-S$highlight_files"]
4220 } else {
4221 # must be "containing:", i.e. we're searching commit info
4222 return
4223 }
4224 set cmd [concat | git diff-tree -r -s --stdin $gdtargs]
4225 set filehighlight [open $cmd r+]
4226 fconfigure $filehighlight -blocking 0
4227 filerun $filehighlight readfhighlight
4228 set fhl_list {}
4229 drawvisible
4230 flushhighlights
4231}
4232
4233proc flushhighlights {} {
4234 global filehighlight fhl_list
4235
4236 if {[info exists filehighlight]} {
4237 lappend fhl_list {}
4238 puts $filehighlight ""
4239 flush $filehighlight
4240 }
4241}
4242
4243proc askfilehighlight {row id} {
4244 global filehighlight fhighlights fhl_list
4245
4246 lappend fhl_list $id
4247 set fhighlights($id) -1
4248 puts $filehighlight $id
4249}
4250
4251proc readfhighlight {} {
4252 global filehighlight fhighlights curview iddrawn
4253 global fhl_list find_dirn
4254
4255 if {![info exists filehighlight]} {
4256 return 0
4257 }
4258 set nr 0
4259 while {[incr nr] <= 100 && [gets $filehighlight line] >= 0} {
4260 set line [string trim $line]
4261 set i [lsearch -exact $fhl_list $line]
4262 if {$i < 0} continue
4263 for {set j 0} {$j < $i} {incr j} {
4264 set id [lindex $fhl_list $j]
4265 set fhighlights($id) 0
4266 }
4267 set fhl_list [lrange $fhl_list [expr {$i+1}] end]
4268 if {$line eq {}} continue
4269 if {![commitinview $line $curview]} continue
4270 if {[info exists iddrawn($line)] && ![ishighlighted $line]} {
4271 bolden $line mainfontbold
4272 }
4273 set fhighlights($line) 1
4274 }
4275 if {[eof $filehighlight]} {
4276 # strange...
4277 puts "oops, git diff-tree died"
4278 catch {close $filehighlight}
4279 unset filehighlight
4280 return 0
4281 }
4282 if {[info exists find_dirn]} {
4283 run findmore
4284 }
4285 return 1
4286}
4287
4288proc doesmatch {f} {
4289 global findtype findpattern
4290
4291 if {$findtype eq [mc "Regexp"]} {
4292 return [regexp $findpattern $f]
4293 } elseif {$findtype eq [mc "IgnCase"]} {
4294 return [string match -nocase $findpattern $f]
4295 } else {
4296 return [string match $findpattern $f]
4297 }
4298}
4299
4300proc askfindhighlight {row id} {
4301 global nhighlights commitinfo iddrawn
4302 global findloc
4303 global markingmatches
4304
4305 if {![info exists commitinfo($id)]} {
4306 getcommit $id
4307 }
4308 set info $commitinfo($id)
4309 set isbold 0
4310 set fldtypes [list [mc Headline] [mc Author] [mc Date] [mc Committer] [mc CDate] [mc Comments]]
4311 foreach f $info ty $fldtypes {
4312 if {($findloc eq [mc "All fields"] || $findloc eq $ty) &&
4313 [doesmatch $f]} {
4314 if {$ty eq [mc "Author"]} {
4315 set isbold 2
4316 break
4317 }
4318 set isbold 1
4319 }
4320 }
4321 if {$isbold && [info exists iddrawn($id)]} {
4322 if {![ishighlighted $id]} {
4323 bolden $id mainfontbold
4324 if {$isbold > 1} {
4325 bolden_name $id mainfontbold
4326 }
4327 }
4328 if {$markingmatches} {
4329 markrowmatches $row $id
4330 }
4331 }
4332 set nhighlights($id) $isbold
4333}
4334
4335proc markrowmatches {row id} {
4336 global canv canv2 linehtag linentag commitinfo findloc
4337
4338 set headline [lindex $commitinfo($id) 0]
4339 set author [lindex $commitinfo($id) 1]
4340 $canv delete match$row
4341 $canv2 delete match$row
4342 if {$findloc eq [mc "All fields"] || $findloc eq [mc "Headline"]} {
4343 set m [findmatches $headline]
4344 if {$m ne {}} {
4345 markmatches $canv $row $headline $linehtag($id) $m \
4346 [$canv itemcget $linehtag($id) -font] $row
4347 }
4348 }
4349 if {$findloc eq [mc "All fields"] || $findloc eq [mc "Author"]} {
4350 set m [findmatches $author]
4351 if {$m ne {}} {
4352 markmatches $canv2 $row $author $linentag($id) $m \
4353 [$canv2 itemcget $linentag($id) -font] $row
4354 }
4355 }
4356}
4357
4358proc vrel_change {name ix op} {
4359 global highlight_related
4360
4361 rhighlight_none
4362 if {$highlight_related ne [mc "None"]} {
4363 run drawvisible
4364 }
4365}
4366
4367# prepare for testing whether commits are descendents or ancestors of a
4368proc rhighlight_sel {a} {
4369 global descendent desc_todo ancestor anc_todo
4370 global highlight_related
4371
4372 catch {unset descendent}
4373 set desc_todo [list $a]
4374 catch {unset ancestor}
4375 set anc_todo [list $a]
4376 if {$highlight_related ne [mc "None"]} {
4377 rhighlight_none
4378 run drawvisible
4379 }
4380}
4381
4382proc rhighlight_none {} {
4383 global rhighlights
4384
4385 catch {unset rhighlights}
4386 unbolden
4387}
4388
4389proc is_descendent {a} {
4390 global curview children descendent desc_todo
4391
4392 set v $curview
4393 set la [rowofcommit $a]
4394 set todo $desc_todo
4395 set leftover {}
4396 set done 0
4397 for {set i 0} {$i < [llength $todo]} {incr i} {
4398 set do [lindex $todo $i]
4399 if {[rowofcommit $do] < $la} {
4400 lappend leftover $do
4401 continue
4402 }
4403 foreach nk $children($v,$do) {
4404 if {![info exists descendent($nk)]} {
4405 set descendent($nk) 1
4406 lappend todo $nk
4407 if {$nk eq $a} {
4408 set done 1
4409 }
4410 }
4411 }
4412 if {$done} {
4413 set desc_todo [concat $leftover [lrange $todo [expr {$i+1}] end]]
4414 return
4415 }
4416 }
4417 set descendent($a) 0
4418 set desc_todo $leftover
4419}
4420
4421proc is_ancestor {a} {
4422 global curview parents ancestor anc_todo
4423
4424 set v $curview
4425 set la [rowofcommit $a]
4426 set todo $anc_todo
4427 set leftover {}
4428 set done 0
4429 for {set i 0} {$i < [llength $todo]} {incr i} {
4430 set do [lindex $todo $i]
4431 if {![commitinview $do $v] || [rowofcommit $do] > $la} {
4432 lappend leftover $do
4433 continue
4434 }
4435 foreach np $parents($v,$do) {
4436 if {![info exists ancestor($np)]} {
4437 set ancestor($np) 1
4438 lappend todo $np
4439 if {$np eq $a} {
4440 set done 1
4441 }
4442 }
4443 }
4444 if {$done} {
4445 set anc_todo [concat $leftover [lrange $todo [expr {$i+1}] end]]
4446 return
4447 }
4448 }
4449 set ancestor($a) 0
4450 set anc_todo $leftover
4451}
4452
4453proc askrelhighlight {row id} {
4454 global descendent highlight_related iddrawn rhighlights
4455 global selectedline ancestor
4456
4457 if {$selectedline eq {}} return
4458 set isbold 0
4459 if {$highlight_related eq [mc "Descendant"] ||
4460 $highlight_related eq [mc "Not descendant"]} {
4461 if {![info exists descendent($id)]} {
4462 is_descendent $id
4463 }
4464 if {$descendent($id) == ($highlight_related eq [mc "Descendant"])} {
4465 set isbold 1
4466 }
4467 } elseif {$highlight_related eq [mc "Ancestor"] ||
4468 $highlight_related eq [mc "Not ancestor"]} {
4469 if {![info exists ancestor($id)]} {
4470 is_ancestor $id
4471 }
4472 if {$ancestor($id) == ($highlight_related eq [mc "Ancestor"])} {
4473 set isbold 1
4474 }
4475 }
4476 if {[info exists iddrawn($id)]} {
4477 if {$isbold && ![ishighlighted $id]} {
4478 bolden $id mainfontbold
4479 }
4480 }
4481 set rhighlights($id) $isbold
4482}
4483
4484# Graph layout functions
4485
4486proc shortids {ids} {
4487 set res {}
4488 foreach id $ids {
4489 if {[llength $id] > 1} {
4490 lappend res [shortids $id]
4491 } elseif {[regexp {^[0-9a-f]{40}$} $id]} {
4492 lappend res [string range $id 0 7]
4493 } else {
4494 lappend res $id
4495 }
4496 }
4497 return $res
4498}
4499
4500proc ntimes {n o} {
4501 set ret {}
4502 set o [list $o]
4503 for {set mask 1} {$mask <= $n} {incr mask $mask} {
4504 if {($n & $mask) != 0} {
4505 set ret [concat $ret $o]
4506 }
4507 set o [concat $o $o]
4508 }
4509 return $ret
4510}
4511
4512proc ordertoken {id} {
4513 global ordertok curview varcid varcstart varctok curview parents children
4514 global nullid nullid2
4515
4516 if {[info exists ordertok($id)]} {
4517 return $ordertok($id)
4518 }
4519 set origid $id
4520 set todo {}
4521 while {1} {
4522 if {[info exists varcid($curview,$id)]} {
4523 set a $varcid($curview,$id)
4524 set p [lindex $varcstart($curview) $a]
4525 } else {
4526 set p [lindex $children($curview,$id) 0]
4527 }
4528 if {[info exists ordertok($p)]} {
4529 set tok $ordertok($p)
4530 break
4531 }
4532 set id [first_real_child $curview,$p]
4533 if {$id eq {}} {
4534 # it's a root
4535 set tok [lindex $varctok($curview) $varcid($curview,$p)]
4536 break
4537 }
4538 if {[llength $parents($curview,$id)] == 1} {
4539 lappend todo [list $p {}]
4540 } else {
4541 set j [lsearch -exact $parents($curview,$id) $p]
4542 if {$j < 0} {
4543 puts "oops didn't find [shortids $p] in parents of [shortids $id]"
4544 }
4545 lappend todo [list $p [strrep $j]]
4546 }
4547 }
4548 for {set i [llength $todo]} {[incr i -1] >= 0} {} {
4549 set p [lindex $todo $i 0]
4550 append tok [lindex $todo $i 1]
4551 set ordertok($p) $tok
4552 }
4553 set ordertok($origid) $tok
4554 return $tok
4555}
4556
4557# Work out where id should go in idlist so that order-token
4558# values increase from left to right
4559proc idcol {idlist id {i 0}} {
4560 set t [ordertoken $id]
4561 if {$i < 0} {
4562 set i 0
4563 }
4564 if {$i >= [llength $idlist] || $t < [ordertoken [lindex $idlist $i]]} {
4565 if {$i > [llength $idlist]} {
4566 set i [llength $idlist]
4567 }
4568 while {[incr i -1] >= 0 && $t < [ordertoken [lindex $idlist $i]]} {}
4569 incr i
4570 } else {
4571 if {$t > [ordertoken [lindex $idlist $i]]} {
4572 while {[incr i] < [llength $idlist] &&
4573 $t >= [ordertoken [lindex $idlist $i]]} {}
4574 }
4575 }
4576 return $i
4577}
4578
4579proc initlayout {} {
4580 global rowidlist rowisopt rowfinal displayorder parentlist
4581 global numcommits canvxmax canv
4582 global nextcolor
4583 global colormap rowtextx
4584
4585 set numcommits 0
4586 set displayorder {}
4587 set parentlist {}
4588 set nextcolor 0
4589 set rowidlist {}
4590 set rowisopt {}
4591 set rowfinal {}
4592 set canvxmax [$canv cget -width]
4593 catch {unset colormap}
4594 catch {unset rowtextx}
4595 setcanvscroll
4596}
4597
4598proc setcanvscroll {} {
4599 global canv canv2 canv3 numcommits linespc canvxmax canvy0
4600 global lastscrollset lastscrollrows
4601
4602 set ymax [expr {$canvy0 + ($numcommits - 0.5) * $linespc + 2}]
4603 $canv conf -scrollregion [list 0 0 $canvxmax $ymax]
4604 $canv2 conf -scrollregion [list 0 0 0 $ymax]
4605 $canv3 conf -scrollregion [list 0 0 0 $ymax]
4606 set lastscrollset [clock clicks -milliseconds]
4607 set lastscrollrows $numcommits
4608}
4609
4610proc visiblerows {} {
4611 global canv numcommits linespc
4612
4613 set ymax [lindex [$canv cget -scrollregion] 3]
4614 if {$ymax eq {} || $ymax == 0} return
4615 set f [$canv yview]
4616 set y0 [expr {int([lindex $f 0] * $ymax)}]
4617 set r0 [expr {int(($y0 - 3) / $linespc) - 1}]
4618 if {$r0 < 0} {
4619 set r0 0
4620 }
4621 set y1 [expr {int([lindex $f 1] * $ymax)}]
4622 set r1 [expr {int(($y1 - 3) / $linespc) + 1}]
4623 if {$r1 >= $numcommits} {
4624 set r1 [expr {$numcommits - 1}]
4625 }
4626 return [list $r0 $r1]
4627}
4628
4629proc layoutmore {} {
4630 global commitidx viewcomplete curview
4631 global numcommits pending_select curview
4632 global lastscrollset lastscrollrows
4633
4634 if {$lastscrollrows < 100 || $viewcomplete($curview) ||
4635 [clock clicks -milliseconds] - $lastscrollset > 500} {
4636 setcanvscroll
4637 }
4638 if {[info exists pending_select] &&
4639 [commitinview $pending_select $curview]} {
4640 update
4641 selectline [rowofcommit $pending_select] 1
4642 }
4643 drawvisible
4644}
4645
4646proc doshowlocalchanges {} {
4647 global curview mainheadid
4648
4649 if {$mainheadid eq {}} return
4650 if {[commitinview $mainheadid $curview]} {
4651 dodiffindex
4652 } else {
4653 interestedin $mainheadid dodiffindex
4654 }
4655}
4656
4657proc dohidelocalchanges {} {
4658 global nullid nullid2 lserial curview
4659
4660 if {[commitinview $nullid $curview]} {
4661 removefakerow $nullid
4662 }
4663 if {[commitinview $nullid2 $curview]} {
4664 removefakerow $nullid2
4665 }
4666 incr lserial
4667}
4668
4669# spawn off a process to do git diff-index --cached HEAD
4670proc dodiffindex {} {
4671 global lserial showlocalchanges
4672 global isworktree
4673
4674 if {!$showlocalchanges || !$isworktree} return
4675 incr lserial
4676 set fd [open "|git diff-index --cached HEAD" r]
4677 fconfigure $fd -blocking 0
4678 set i [reg_instance $fd]
4679 filerun $fd [list readdiffindex $fd $lserial $i]
4680}
4681
4682proc readdiffindex {fd serial inst} {
4683 global mainheadid nullid nullid2 curview commitinfo commitdata lserial
4684
4685 set isdiff 1
4686 if {[gets $fd line] < 0} {
4687 if {![eof $fd]} {
4688 return 1
4689 }
4690 set isdiff 0
4691 }
4692 # we only need to see one line and we don't really care what it says...
4693 stop_instance $inst
4694
4695 if {$serial != $lserial} {
4696 return 0
4697 }
4698
4699 # now see if there are any local changes not checked in to the index
4700 set fd [open "|git diff-files" r]
4701 fconfigure $fd -blocking 0
4702 set i [reg_instance $fd]
4703 filerun $fd [list readdifffiles $fd $serial $i]
4704
4705 if {$isdiff && ![commitinview $nullid2 $curview]} {
4706 # add the line for the changes in the index to the graph
4707 set hl [mc "Local changes checked in to index but not committed"]
4708 set commitinfo($nullid2) [list $hl {} {} {} {} " $hl\n"]
4709 set commitdata($nullid2) "\n $hl\n"
4710 if {[commitinview $nullid $curview]} {
4711 removefakerow $nullid
4712 }
4713 insertfakerow $nullid2 $mainheadid
4714 } elseif {!$isdiff && [commitinview $nullid2 $curview]} {
4715 removefakerow $nullid2
4716 }
4717 return 0
4718}
4719
4720proc readdifffiles {fd serial inst} {
4721 global mainheadid nullid nullid2 curview
4722 global commitinfo commitdata lserial
4723
4724 set isdiff 1
4725 if {[gets $fd line] < 0} {
4726 if {![eof $fd]} {
4727 return 1
4728 }
4729 set isdiff 0
4730 }
4731 # we only need to see one line and we don't really care what it says...
4732 stop_instance $inst
4733
4734 if {$serial != $lserial} {
4735 return 0
4736 }
4737
4738 if {$isdiff && ![commitinview $nullid $curview]} {
4739 # add the line for the local diff to the graph
4740 set hl [mc "Local uncommitted changes, not checked in to index"]
4741 set commitinfo($nullid) [list $hl {} {} {} {} " $hl\n"]
4742 set commitdata($nullid) "\n $hl\n"
4743 if {[commitinview $nullid2 $curview]} {
4744 set p $nullid2
4745 } else {
4746 set p $mainheadid
4747 }
4748 insertfakerow $nullid $p
4749 } elseif {!$isdiff && [commitinview $nullid $curview]} {
4750 removefakerow $nullid
4751 }
4752 return 0
4753}
4754
4755proc nextuse {id row} {
4756 global curview children
4757
4758 if {[info exists children($curview,$id)]} {
4759 foreach kid $children($curview,$id) {
4760 if {![commitinview $kid $curview]} {
4761 return -1
4762 }
4763 if {[rowofcommit $kid] > $row} {
4764 return [rowofcommit $kid]
4765 }
4766 }
4767 }
4768 if {[commitinview $id $curview]} {
4769 return [rowofcommit $id]
4770 }
4771 return -1
4772}
4773
4774proc prevuse {id row} {
4775 global curview children
4776
4777 set ret -1
4778 if {[info exists children($curview,$id)]} {
4779 foreach kid $children($curview,$id) {
4780 if {![commitinview $kid $curview]} break
4781 if {[rowofcommit $kid] < $row} {
4782 set ret [rowofcommit $kid]
4783 }
4784 }
4785 }
4786 return $ret
4787}
4788
4789proc make_idlist {row} {
4790 global displayorder parentlist uparrowlen downarrowlen mingaplen
4791 global commitidx curview children
4792
4793 set r [expr {$row - $mingaplen - $downarrowlen - 1}]
4794 if {$r < 0} {
4795 set r 0
4796 }
4797 set ra [expr {$row - $downarrowlen}]
4798 if {$ra < 0} {
4799 set ra 0
4800 }
4801 set rb [expr {$row + $uparrowlen}]
4802 if {$rb > $commitidx($curview)} {
4803 set rb $commitidx($curview)
4804 }
4805 make_disporder $r [expr {$rb + 1}]
4806 set ids {}
4807 for {} {$r < $ra} {incr r} {
4808 set nextid [lindex $displayorder [expr {$r + 1}]]
4809 foreach p [lindex $parentlist $r] {
4810 if {$p eq $nextid} continue
4811 set rn [nextuse $p $r]
4812 if {$rn >= $row &&
4813 $rn <= $r + $downarrowlen + $mingaplen + $uparrowlen} {
4814 lappend ids [list [ordertoken $p] $p]
4815 }
4816 }
4817 }
4818 for {} {$r < $row} {incr r} {
4819 set nextid [lindex $displayorder [expr {$r + 1}]]
4820 foreach p [lindex $parentlist $r] {
4821 if {$p eq $nextid} continue
4822 set rn [nextuse $p $r]
4823 if {$rn < 0 || $rn >= $row} {
4824 lappend ids [list [ordertoken $p] $p]
4825 }
4826 }
4827 }
4828 set id [lindex $displayorder $row]
4829 lappend ids [list [ordertoken $id] $id]
4830 while {$r < $rb} {
4831 foreach p [lindex $parentlist $r] {
4832 set firstkid [lindex $children($curview,$p) 0]
4833 if {[rowofcommit $firstkid] < $row} {
4834 lappend ids [list [ordertoken $p] $p]
4835 }
4836 }
4837 incr r
4838 set id [lindex $displayorder $r]
4839 if {$id ne {}} {
4840 set firstkid [lindex $children($curview,$id) 0]
4841 if {$firstkid ne {} && [rowofcommit $firstkid] < $row} {
4842 lappend ids [list [ordertoken $id] $id]
4843 }
4844 }
4845 }
4846 set idlist {}
4847 foreach idx [lsort -unique $ids] {
4848 lappend idlist [lindex $idx 1]
4849 }
4850 return $idlist
4851}
4852
4853proc rowsequal {a b} {
4854 while {[set i [lsearch -exact $a {}]] >= 0} {
4855 set a [lreplace $a $i $i]
4856 }
4857 while {[set i [lsearch -exact $b {}]] >= 0} {
4858 set b [lreplace $b $i $i]
4859 }
4860 return [expr {$a eq $b}]
4861}
4862
4863proc makeupline {id row rend col} {
4864 global rowidlist uparrowlen downarrowlen mingaplen
4865
4866 for {set r $rend} {1} {set r $rstart} {
4867 set rstart [prevuse $id $r]
4868 if {$rstart < 0} return
4869 if {$rstart < $row} break
4870 }
4871 if {$rstart + $uparrowlen + $mingaplen + $downarrowlen < $rend} {
4872 set rstart [expr {$rend - $uparrowlen - 1}]
4873 }
4874 for {set r $rstart} {[incr r] <= $row} {} {
4875 set idlist [lindex $rowidlist $r]
4876 if {$idlist ne {} && [lsearch -exact $idlist $id] < 0} {
4877 set col [idcol $idlist $id $col]
4878 lset rowidlist $r [linsert $idlist $col $id]
4879 changedrow $r
4880 }
4881 }
4882}
4883
4884proc layoutrows {row endrow} {
4885 global rowidlist rowisopt rowfinal displayorder
4886 global uparrowlen downarrowlen maxwidth mingaplen
4887 global children parentlist
4888 global commitidx viewcomplete curview
4889
4890 make_disporder [expr {$row - 1}] [expr {$endrow + $uparrowlen}]
4891 set idlist {}
4892 if {$row > 0} {
4893 set rm1 [expr {$row - 1}]
4894 foreach id [lindex $rowidlist $rm1] {
4895 if {$id ne {}} {
4896 lappend idlist $id
4897 }
4898 }
4899 set final [lindex $rowfinal $rm1]
4900 }
4901 for {} {$row < $endrow} {incr row} {
4902 set rm1 [expr {$row - 1}]
4903 if {$rm1 < 0 || $idlist eq {}} {
4904 set idlist [make_idlist $row]
4905 set final 1
4906 } else {
4907 set id [lindex $displayorder $rm1]
4908 set col [lsearch -exact $idlist $id]
4909 set idlist [lreplace $idlist $col $col]
4910 foreach p [lindex $parentlist $rm1] {
4911 if {[lsearch -exact $idlist $p] < 0} {
4912 set col [idcol $idlist $p $col]
4913 set idlist [linsert $idlist $col $p]
4914 # if not the first child, we have to insert a line going up
4915 if {$id ne [lindex $children($curview,$p) 0]} {
4916 makeupline $p $rm1 $row $col
4917 }
4918 }
4919 }
4920 set id [lindex $displayorder $row]
4921 if {$row > $downarrowlen} {
4922 set termrow [expr {$row - $downarrowlen - 1}]
4923 foreach p [lindex $parentlist $termrow] {
4924 set i [lsearch -exact $idlist $p]
4925 if {$i < 0} continue
4926 set nr [nextuse $p $termrow]
4927 if {$nr < 0 || $nr >= $row + $mingaplen + $uparrowlen} {
4928 set idlist [lreplace $idlist $i $i]
4929 }
4930 }
4931 }
4932 set col [lsearch -exact $idlist $id]
4933 if {$col < 0} {
4934 set col [idcol $idlist $id]
4935 set idlist [linsert $idlist $col $id]
4936 if {$children($curview,$id) ne {}} {
4937 makeupline $id $rm1 $row $col
4938 }
4939 }
4940 set r [expr {$row + $uparrowlen - 1}]
4941 if {$r < $commitidx($curview)} {
4942 set x $col
4943 foreach p [lindex $parentlist $r] {
4944 if {[lsearch -exact $idlist $p] >= 0} continue
4945 set fk [lindex $children($curview,$p) 0]
4946 if {[rowofcommit $fk] < $row} {
4947 set x [idcol $idlist $p $x]
4948 set idlist [linsert $idlist $x $p]
4949 }
4950 }
4951 if {[incr r] < $commitidx($curview)} {
4952 set p [lindex $displayorder $r]
4953 if {[lsearch -exact $idlist $p] < 0} {
4954 set fk [lindex $children($curview,$p) 0]
4955 if {$fk ne {} && [rowofcommit $fk] < $row} {
4956 set x [idcol $idlist $p $x]
4957 set idlist [linsert $idlist $x $p]
4958 }
4959 }
4960 }
4961 }
4962 }
4963 if {$final && !$viewcomplete($curview) &&
4964 $row + $uparrowlen + $mingaplen + $downarrowlen
4965 >= $commitidx($curview)} {
4966 set final 0
4967 }
4968 set l [llength $rowidlist]
4969 if {$row == $l} {
4970 lappend rowidlist $idlist
4971 lappend rowisopt 0
4972 lappend rowfinal $final
4973 } elseif {$row < $l} {
4974 if {![rowsequal $idlist [lindex $rowidlist $row]]} {
4975 lset rowidlist $row $idlist
4976 changedrow $row
4977 }
4978 lset rowfinal $row $final
4979 } else {
4980 set pad [ntimes [expr {$row - $l}] {}]
4981 set rowidlist [concat $rowidlist $pad]
4982 lappend rowidlist $idlist
4983 set rowfinal [concat $rowfinal $pad]
4984 lappend rowfinal $final
4985 set rowisopt [concat $rowisopt [ntimes [expr {$row - $l + 1}] 0]]
4986 }
4987 }
4988 return $row
4989}
4990
4991proc changedrow {row} {
4992 global displayorder iddrawn rowisopt need_redisplay
4993
4994 set l [llength $rowisopt]
4995 if {$row < $l} {
4996 lset rowisopt $row 0
4997 if {$row + 1 < $l} {
4998 lset rowisopt [expr {$row + 1}] 0
4999 if {$row + 2 < $l} {
5000 lset rowisopt [expr {$row + 2}] 0
5001 }
5002 }
5003 }
5004 set id [lindex $displayorder $row]
5005 if {[info exists iddrawn($id)]} {
5006 set need_redisplay 1
5007 }
5008}
5009
5010proc insert_pad {row col npad} {
5011 global rowidlist
5012
5013 set pad [ntimes $npad {}]
5014 set idlist [lindex $rowidlist $row]
5015 set bef [lrange $idlist 0 [expr {$col - 1}]]
5016 set aft [lrange $idlist $col end]
5017 set i [lsearch -exact $aft {}]
5018 if {$i > 0} {
5019 set aft [lreplace $aft $i $i]
5020 }
5021 lset rowidlist $row [concat $bef $pad $aft]
5022 changedrow $row
5023}
5024
5025proc optimize_rows {row col endrow} {
5026 global rowidlist rowisopt displayorder curview children
5027
5028 if {$row < 1} {
5029 set row 1
5030 }
5031 for {} {$row < $endrow} {incr row; set col 0} {
5032 if {[lindex $rowisopt $row]} continue
5033 set haspad 0
5034 set y0 [expr {$row - 1}]
5035 set ym [expr {$row - 2}]
5036 set idlist [lindex $rowidlist $row]
5037 set previdlist [lindex $rowidlist $y0]
5038 if {$idlist eq {} || $previdlist eq {}} continue
5039 if {$ym >= 0} {
5040 set pprevidlist [lindex $rowidlist $ym]
5041 if {$pprevidlist eq {}} continue
5042 } else {
5043 set pprevidlist {}
5044 }
5045 set x0 -1
5046 set xm -1
5047 for {} {$col < [llength $idlist]} {incr col} {
5048 set id [lindex $idlist $col]
5049 if {[lindex $previdlist $col] eq $id} continue
5050 if {$id eq {}} {
5051 set haspad 1
5052 continue
5053 }
5054 set x0 [lsearch -exact $previdlist $id]
5055 if {$x0 < 0} continue
5056 set z [expr {$x0 - $col}]
5057 set isarrow 0
5058 set z0 {}
5059 if {$ym >= 0} {
5060 set xm [lsearch -exact $pprevidlist $id]
5061 if {$xm >= 0} {
5062 set z0 [expr {$xm - $x0}]
5063 }
5064 }
5065 if {$z0 eq {}} {
5066 # if row y0 is the first child of $id then it's not an arrow
5067 if {[lindex $children($curview,$id) 0] ne
5068 [lindex $displayorder $y0]} {
5069 set isarrow 1
5070 }
5071 }
5072 if {!$isarrow && $id ne [lindex $displayorder $row] &&
5073 [lsearch -exact [lindex $rowidlist [expr {$row+1}]] $id] < 0} {
5074 set isarrow 1
5075 }
5076 # Looking at lines from this row to the previous row,
5077 # make them go straight up if they end in an arrow on
5078 # the previous row; otherwise make them go straight up
5079 # or at 45 degrees.
5080 if {$z < -1 || ($z < 0 && $isarrow)} {
5081 # Line currently goes left too much;
5082 # insert pads in the previous row, then optimize it
5083 set npad [expr {-1 - $z + $isarrow}]
5084 insert_pad $y0 $x0 $npad
5085 if {$y0 > 0} {
5086 optimize_rows $y0 $x0 $row
5087 }
5088 set previdlist [lindex $rowidlist $y0]
5089 set x0 [lsearch -exact $previdlist $id]
5090 set z [expr {$x0 - $col}]
5091 if {$z0 ne {}} {
5092 set pprevidlist [lindex $rowidlist $ym]
5093 set xm [lsearch -exact $pprevidlist $id]
5094 set z0 [expr {$xm - $x0}]
5095 }
5096 } elseif {$z > 1 || ($z > 0 && $isarrow)} {
5097 # Line currently goes right too much;
5098 # insert pads in this line
5099 set npad [expr {$z - 1 + $isarrow}]
5100 insert_pad $row $col $npad
5101 set idlist [lindex $rowidlist $row]
5102 incr col $npad
5103 set z [expr {$x0 - $col}]
5104 set haspad 1
5105 }
5106 if {$z0 eq {} && !$isarrow && $ym >= 0} {
5107 # this line links to its first child on row $row-2
5108 set id [lindex $displayorder $ym]
5109 set xc [lsearch -exact $pprevidlist $id]
5110 if {$xc >= 0} {
5111 set z0 [expr {$xc - $x0}]
5112 }
5113 }
5114 # avoid lines jigging left then immediately right
5115 if {$z0 ne {} && $z < 0 && $z0 > 0} {
5116 insert_pad $y0 $x0 1
5117 incr x0
5118 optimize_rows $y0 $x0 $row
5119 set previdlist [lindex $rowidlist $y0]
5120 }
5121 }
5122 if {!$haspad} {
5123 # Find the first column that doesn't have a line going right
5124 for {set col [llength $idlist]} {[incr col -1] >= 0} {} {
5125 set id [lindex $idlist $col]
5126 if {$id eq {}} break
5127 set x0 [lsearch -exact $previdlist $id]
5128 if {$x0 < 0} {
5129 # check if this is the link to the first child
5130 set kid [lindex $displayorder $y0]
5131 if {[lindex $children($curview,$id) 0] eq $kid} {
5132 # it is, work out offset to child
5133 set x0 [lsearch -exact $previdlist $kid]
5134 }
5135 }
5136 if {$x0 <= $col} break
5137 }
5138 # Insert a pad at that column as long as it has a line and
5139 # isn't the last column
5140 if {$x0 >= 0 && [incr col] < [llength $idlist]} {
5141 set idlist [linsert $idlist $col {}]
5142 lset rowidlist $row $idlist
5143 changedrow $row
5144 }
5145 }
5146 }
5147}
5148
5149proc xc {row col} {
5150 global canvx0 linespc
5151 return [expr {$canvx0 + $col * $linespc}]
5152}
5153
5154proc yc {row} {
5155 global canvy0 linespc
5156 return [expr {$canvy0 + $row * $linespc}]
5157}
5158
5159proc linewidth {id} {
5160 global thickerline lthickness
5161
5162 set wid $lthickness
5163 if {[info exists thickerline] && $id eq $thickerline} {
5164 set wid [expr {2 * $lthickness}]
5165 }
5166 return $wid
5167}
5168
5169proc rowranges {id} {
5170 global curview children uparrowlen downarrowlen
5171 global rowidlist
5172
5173 set kids $children($curview,$id)
5174 if {$kids eq {}} {
5175 return {}
5176 }
5177 set ret {}
5178 lappend kids $id
5179 foreach child $kids {
5180 if {![commitinview $child $curview]} break
5181 set row [rowofcommit $child]
5182 if {![info exists prev]} {
5183 lappend ret [expr {$row + 1}]
5184 } else {
5185 if {$row <= $prevrow} {
5186 puts "oops children of [shortids $id] out of order [shortids $child] $row <= [shortids $prev] $prevrow"
5187 }
5188 # see if the line extends the whole way from prevrow to row
5189 if {$row > $prevrow + $uparrowlen + $downarrowlen &&
5190 [lsearch -exact [lindex $rowidlist \
5191 [expr {int(($row + $prevrow) / 2)}]] $id] < 0} {
5192 # it doesn't, see where it ends
5193 set r [expr {$prevrow + $downarrowlen}]
5194 if {[lsearch -exact [lindex $rowidlist $r] $id] < 0} {
5195 while {[incr r -1] > $prevrow &&
5196 [lsearch -exact [lindex $rowidlist $r] $id] < 0} {}
5197 } else {
5198 while {[incr r] <= $row &&
5199 [lsearch -exact [lindex $rowidlist $r] $id] >= 0} {}
5200 incr r -1
5201 }
5202 lappend ret $r
5203 # see where it starts up again
5204 set r [expr {$row - $uparrowlen}]
5205 if {[lsearch -exact [lindex $rowidlist $r] $id] < 0} {
5206 while {[incr r] < $row &&
5207 [lsearch -exact [lindex $rowidlist $r] $id] < 0} {}
5208 } else {
5209 while {[incr r -1] >= $prevrow &&
5210 [lsearch -exact [lindex $rowidlist $r] $id] >= 0} {}
5211 incr r
5212 }
5213 lappend ret $r
5214 }
5215 }
5216 if {$child eq $id} {
5217 lappend ret $row
5218 }
5219 set prev $child
5220 set prevrow $row
5221 }
5222 return $ret
5223}
5224
5225proc drawlineseg {id row endrow arrowlow} {
5226 global rowidlist displayorder iddrawn linesegs
5227 global canv colormap linespc curview maxlinelen parentlist
5228
5229 set cols [list [lsearch -exact [lindex $rowidlist $row] $id]]
5230 set le [expr {$row + 1}]
5231 set arrowhigh 1
5232 while {1} {
5233 set c [lsearch -exact [lindex $rowidlist $le] $id]
5234 if {$c < 0} {
5235 incr le -1
5236 break
5237 }
5238 lappend cols $c
5239 set x [lindex $displayorder $le]
5240 if {$x eq $id} {
5241 set arrowhigh 0
5242 break
5243 }
5244 if {[info exists iddrawn($x)] || $le == $endrow} {
5245 set c [lsearch -exact [lindex $rowidlist [expr {$le+1}]] $id]
5246 if {$c >= 0} {
5247 lappend cols $c
5248 set arrowhigh 0
5249 }
5250 break
5251 }
5252 incr le
5253 }
5254 if {$le <= $row} {
5255 return $row
5256 }
5257
5258 set lines {}
5259 set i 0
5260 set joinhigh 0
5261 if {[info exists linesegs($id)]} {
5262 set lines $linesegs($id)
5263 foreach li $lines {
5264 set r0 [lindex $li 0]
5265 if {$r0 > $row} {
5266 if {$r0 == $le && [lindex $li 1] - $row <= $maxlinelen} {
5267 set joinhigh 1
5268 }
5269 break
5270 }
5271 incr i
5272 }
5273 }
5274 set joinlow 0
5275 if {$i > 0} {
5276 set li [lindex $lines [expr {$i-1}]]
5277 set r1 [lindex $li 1]
5278 if {$r1 == $row && $le - [lindex $li 0] <= $maxlinelen} {
5279 set joinlow 1
5280 }
5281 }
5282
5283 set x [lindex $cols [expr {$le - $row}]]
5284 set xp [lindex $cols [expr {$le - 1 - $row}]]
5285 set dir [expr {$xp - $x}]
5286 if {$joinhigh} {
5287 set ith [lindex $lines $i 2]
5288 set coords [$canv coords $ith]
5289 set ah [$canv itemcget $ith -arrow]
5290 set arrowhigh [expr {$ah eq "first" || $ah eq "both"}]
5291 set x2 [lindex $cols [expr {$le + 1 - $row}]]
5292 if {$x2 ne {} && $x - $x2 == $dir} {
5293 set coords [lrange $coords 0 end-2]
5294 }
5295 } else {
5296 set coords [list [xc $le $x] [yc $le]]
5297 }
5298 if {$joinlow} {
5299 set itl [lindex $lines [expr {$i-1}] 2]
5300 set al [$canv itemcget $itl -arrow]
5301 set arrowlow [expr {$al eq "last" || $al eq "both"}]
5302 } elseif {$arrowlow} {
5303 if {[lsearch -exact [lindex $rowidlist [expr {$row-1}]] $id] >= 0 ||
5304 [lsearch -exact [lindex $parentlist [expr {$row-1}]] $id] >= 0} {
5305 set arrowlow 0
5306 }
5307 }
5308 set arrow [lindex {none first last both} [expr {$arrowhigh + 2*$arrowlow}]]
5309 for {set y $le} {[incr y -1] > $row} {} {
5310 set x $xp
5311 set xp [lindex $cols [expr {$y - 1 - $row}]]
5312 set ndir [expr {$xp - $x}]
5313 if {$dir != $ndir || $xp < 0} {
5314 lappend coords [xc $y $x] [yc $y]
5315 }
5316 set dir $ndir
5317 }
5318 if {!$joinlow} {
5319 if {$xp < 0} {
5320 # join parent line to first child
5321 set ch [lindex $displayorder $row]
5322 set xc [lsearch -exact [lindex $rowidlist $row] $ch]
5323 if {$xc < 0} {
5324 puts "oops: drawlineseg: child $ch not on row $row"
5325 } elseif {$xc != $x} {
5326 if {($arrowhigh && $le == $row + 1) || $dir == 0} {
5327 set d [expr {int(0.5 * $linespc)}]
5328 set x1 [xc $row $x]
5329 if {$xc < $x} {
5330 set x2 [expr {$x1 - $d}]
5331 } else {
5332 set x2 [expr {$x1 + $d}]
5333 }
5334 set y2 [yc $row]
5335 set y1 [expr {$y2 + $d}]
5336 lappend coords $x1 $y1 $x2 $y2
5337 } elseif {$xc < $x - 1} {
5338 lappend coords [xc $row [expr {$x-1}]] [yc $row]
5339 } elseif {$xc > $x + 1} {
5340 lappend coords [xc $row [expr {$x+1}]] [yc $row]
5341 }
5342 set x $xc
5343 }
5344 lappend coords [xc $row $x] [yc $row]
5345 } else {
5346 set xn [xc $row $xp]
5347 set yn [yc $row]
5348 lappend coords $xn $yn
5349 }
5350 if {!$joinhigh} {
5351 assigncolor $id
5352 set t [$canv create line $coords -width [linewidth $id] \
5353 -fill $colormap($id) -tags lines.$id -arrow $arrow]
5354 $canv lower $t
5355 bindline $t $id
5356 set lines [linsert $lines $i [list $row $le $t]]
5357 } else {
5358 $canv coords $ith $coords
5359 if {$arrow ne $ah} {
5360 $canv itemconf $ith -arrow $arrow
5361 }
5362 lset lines $i 0 $row
5363 }
5364 } else {
5365 set xo [lsearch -exact [lindex $rowidlist [expr {$row - 1}]] $id]
5366 set ndir [expr {$xo - $xp}]
5367 set clow [$canv coords $itl]
5368 if {$dir == $ndir} {
5369 set clow [lrange $clow 2 end]
5370 }
5371 set coords [concat $coords $clow]
5372 if {!$joinhigh} {
5373 lset lines [expr {$i-1}] 1 $le
5374 } else {
5375 # coalesce two pieces
5376 $canv delete $ith
5377 set b [lindex $lines [expr {$i-1}] 0]
5378 set e [lindex $lines $i 1]
5379 set lines [lreplace $lines [expr {$i-1}] $i [list $b $e $itl]]
5380 }
5381 $canv coords $itl $coords
5382 if {$arrow ne $al} {
5383 $canv itemconf $itl -arrow $arrow
5384 }
5385 }
5386
5387 set linesegs($id) $lines
5388 return $le
5389}
5390
5391proc drawparentlinks {id row} {
5392 global rowidlist canv colormap curview parentlist
5393 global idpos linespc
5394
5395 set rowids [lindex $rowidlist $row]
5396 set col [lsearch -exact $rowids $id]
5397 if {$col < 0} return
5398 set olds [lindex $parentlist $row]
5399 set row2 [expr {$row + 1}]
5400 set x [xc $row $col]
5401 set y [yc $row]
5402 set y2 [yc $row2]
5403 set d [expr {int(0.5 * $linespc)}]
5404 set ymid [expr {$y + $d}]
5405 set ids [lindex $rowidlist $row2]
5406 # rmx = right-most X coord used
5407 set rmx 0
5408 foreach p $olds {
5409 set i [lsearch -exact $ids $p]
5410 if {$i < 0} {
5411 puts "oops, parent $p of $id not in list"
5412 continue
5413 }
5414 set x2 [xc $row2 $i]
5415 if {$x2 > $rmx} {
5416 set rmx $x2
5417 }
5418 set j [lsearch -exact $rowids $p]
5419 if {$j < 0} {
5420 # drawlineseg will do this one for us
5421 continue
5422 }
5423 assigncolor $p
5424 # should handle duplicated parents here...
5425 set coords [list $x $y]
5426 if {$i != $col} {
5427 # if attaching to a vertical segment, draw a smaller
5428 # slant for visual distinctness
5429 if {$i == $j} {
5430 if {$i < $col} {
5431 lappend coords [expr {$x2 + $d}] $y $x2 $ymid
5432 } else {
5433 lappend coords [expr {$x2 - $d}] $y $x2 $ymid
5434 }
5435 } elseif {$i < $col && $i < $j} {
5436 # segment slants towards us already
5437 lappend coords [xc $row $j] $y
5438 } else {
5439 if {$i < $col - 1} {
5440 lappend coords [expr {$x2 + $linespc}] $y
5441 } elseif {$i > $col + 1} {
5442 lappend coords [expr {$x2 - $linespc}] $y
5443 }
5444 lappend coords $x2 $y2
5445 }
5446 } else {
5447 lappend coords $x2 $y2
5448 }
5449 set t [$canv create line $coords -width [linewidth $p] \
5450 -fill $colormap($p) -tags lines.$p]
5451 $canv lower $t
5452 bindline $t $p
5453 }
5454 if {$rmx > [lindex $idpos($id) 1]} {
5455 lset idpos($id) 1 $rmx
5456 redrawtags $id
5457 }
5458}
5459
5460proc drawlines {id} {
5461 global canv
5462
5463 $canv itemconf lines.$id -width [linewidth $id]
5464}
5465
5466proc drawcmittext {id row col} {
5467 global linespc canv canv2 canv3 fgcolor curview
5468 global cmitlisted commitinfo rowidlist parentlist
5469 global rowtextx idpos idtags idheads idotherrefs
5470 global linehtag linentag linedtag selectedline
5471 global canvxmax boldids boldnameids fgcolor
5472 global mainheadid nullid nullid2 circleitem circlecolors ctxbut
5473
5474 # listed is 0 for boundary, 1 for normal, 2 for negative, 3 for left, 4 for right
5475 set listed $cmitlisted($curview,$id)
5476 if {$id eq $nullid} {
5477 set ofill red
5478 } elseif {$id eq $nullid2} {
5479 set ofill green
5480 } elseif {$id eq $mainheadid} {
5481 set ofill yellow
5482 } else {
5483 set ofill [lindex $circlecolors $listed]
5484 }
5485 set x [xc $row $col]
5486 set y [yc $row]
5487 set orad [expr {$linespc / 3}]
5488 if {$listed <= 2} {
5489 set t [$canv create oval [expr {$x - $orad}] [expr {$y - $orad}] \
5490 [expr {$x + $orad - 1}] [expr {$y + $orad - 1}] \
5491 -fill $ofill -outline $fgcolor -width 1 -tags circle]
5492 } elseif {$listed == 3} {
5493 # triangle pointing left for left-side commits
5494 set t [$canv create polygon \
5495 [expr {$x - $orad}] $y \
5496 [expr {$x + $orad - 1}] [expr {$y - $orad}] \
5497 [expr {$x + $orad - 1}] [expr {$y + $orad - 1}] \
5498 -fill $ofill -outline $fgcolor -width 1 -tags circle]
5499 } else {
5500 # triangle pointing right for right-side commits
5501 set t [$canv create polygon \
5502 [expr {$x + $orad - 1}] $y \
5503 [expr {$x - $orad}] [expr {$y - $orad}] \
5504 [expr {$x - $orad}] [expr {$y + $orad - 1}] \
5505 -fill $ofill -outline $fgcolor -width 1 -tags circle]
5506 }
5507 set circleitem($row) $t
5508 $canv raise $t
5509 $canv bind $t <1> {selcanvline {} %x %y}
5510 set rmx [llength [lindex $rowidlist $row]]
5511 set olds [lindex $parentlist $row]
5512 if {$olds ne {}} {
5513 set nextids [lindex $rowidlist [expr {$row + 1}]]
5514 foreach p $olds {
5515 set i [lsearch -exact $nextids $p]
5516 if {$i > $rmx} {
5517 set rmx $i
5518 }
5519 }
5520 }
5521 set xt [xc $row $rmx]
5522 set rowtextx($row) $xt
5523 set idpos($id) [list $x $xt $y]
5524 if {[info exists idtags($id)] || [info exists idheads($id)]
5525 || [info exists idotherrefs($id)]} {
5526 set xt [drawtags $id $x $xt $y]
5527 }
5528 set headline [lindex $commitinfo($id) 0]
5529 set name [lindex $commitinfo($id) 1]
5530 set date [lindex $commitinfo($id) 2]
5531 set date [formatdate $date]
5532 set font mainfont
5533 set nfont mainfont
5534 set isbold [ishighlighted $id]
5535 if {$isbold > 0} {
5536 lappend boldids $id
5537 set font mainfontbold
5538 if {$isbold > 1} {
5539 lappend boldnameids $id
5540 set nfont mainfontbold
5541 }
5542 }
5543 set linehtag($id) [$canv create text $xt $y -anchor w -fill $fgcolor \
5544 -text $headline -font $font -tags text]
5545 $canv bind $linehtag($id) $ctxbut "rowmenu %X %Y $id"
5546 set linentag($id) [$canv2 create text 3 $y -anchor w -fill $fgcolor \
5547 -text $name -font $nfont -tags text]
5548 set linedtag($id) [$canv3 create text 3 $y -anchor w -fill $fgcolor \
5549 -text $date -font mainfont -tags text]
5550 if {$selectedline == $row} {
5551 make_secsel $id
5552 }
5553 set xr [expr {$xt + [font measure $font $headline]}]
5554 if {$xr > $canvxmax} {
5555 set canvxmax $xr
5556 setcanvscroll
5557 }
5558}
5559
5560proc drawcmitrow {row} {
5561 global displayorder rowidlist nrows_drawn
5562 global iddrawn markingmatches
5563 global commitinfo numcommits
5564 global filehighlight fhighlights findpattern nhighlights
5565 global hlview vhighlights
5566 global highlight_related rhighlights
5567
5568 if {$row >= $numcommits} return
5569
5570 set id [lindex $displayorder $row]
5571 if {[info exists hlview] && ![info exists vhighlights($id)]} {
5572 askvhighlight $row $id
5573 }
5574 if {[info exists filehighlight] && ![info exists fhighlights($id)]} {
5575 askfilehighlight $row $id
5576 }
5577 if {$findpattern ne {} && ![info exists nhighlights($id)]} {
5578 askfindhighlight $row $id
5579 }
5580 if {$highlight_related ne [mc "None"] && ![info exists rhighlights($id)]} {
5581 askrelhighlight $row $id
5582 }
5583 if {![info exists iddrawn($id)]} {
5584 set col [lsearch -exact [lindex $rowidlist $row] $id]
5585 if {$col < 0} {
5586 puts "oops, row $row id $id not in list"
5587 return
5588 }
5589 if {![info exists commitinfo($id)]} {
5590 getcommit $id
5591 }
5592 assigncolor $id
5593 drawcmittext $id $row $col
5594 set iddrawn($id) 1
5595 incr nrows_drawn
5596 }
5597 if {$markingmatches} {
5598 markrowmatches $row $id
5599 }
5600}
5601
5602proc drawcommits {row {endrow {}}} {
5603 global numcommits iddrawn displayorder curview need_redisplay
5604 global parentlist rowidlist rowfinal uparrowlen downarrowlen nrows_drawn
5605
5606 if {$row < 0} {
5607 set row 0
5608 }
5609 if {$endrow eq {}} {
5610 set endrow $row
5611 }
5612 if {$endrow >= $numcommits} {
5613 set endrow [expr {$numcommits - 1}]
5614 }
5615
5616 set rl1 [expr {$row - $downarrowlen - 3}]
5617 if {$rl1 < 0} {
5618 set rl1 0
5619 }
5620 set ro1 [expr {$row - 3}]
5621 if {$ro1 < 0} {
5622 set ro1 0
5623 }
5624 set r2 [expr {$endrow + $uparrowlen + 3}]
5625 if {$r2 > $numcommits} {
5626 set r2 $numcommits
5627 }
5628 for {set r $rl1} {$r < $r2} {incr r} {
5629 if {[lindex $rowidlist $r] ne {} && [lindex $rowfinal $r]} {
5630 if {$rl1 < $r} {
5631 layoutrows $rl1 $r
5632 }
5633 set rl1 [expr {$r + 1}]
5634 }
5635 }
5636 if {$rl1 < $r} {
5637 layoutrows $rl1 $r
5638 }
5639 optimize_rows $ro1 0 $r2
5640 if {$need_redisplay || $nrows_drawn > 2000} {
5641 clear_display
5642 drawvisible
5643 }
5644
5645 # make the lines join to already-drawn rows either side
5646 set r [expr {$row - 1}]
5647 if {$r < 0 || ![info exists iddrawn([lindex $displayorder $r])]} {
5648 set r $row
5649 }
5650 set er [expr {$endrow + 1}]
5651 if {$er >= $numcommits ||
5652 ![info exists iddrawn([lindex $displayorder $er])]} {
5653 set er $endrow
5654 }
5655 for {} {$r <= $er} {incr r} {
5656 set id [lindex $displayorder $r]
5657 set wasdrawn [info exists iddrawn($id)]
5658 drawcmitrow $r
5659 if {$r == $er} break
5660 set nextid [lindex $displayorder [expr {$r + 1}]]
5661 if {$wasdrawn && [info exists iddrawn($nextid)]} continue
5662 drawparentlinks $id $r
5663
5664 set rowids [lindex $rowidlist $r]
5665 foreach lid $rowids {
5666 if {$lid eq {}} continue
5667 if {[info exists lineend($lid)] && $lineend($lid) > $r} continue
5668 if {$lid eq $id} {
5669 # see if this is the first child of any of its parents
5670 foreach p [lindex $parentlist $r] {
5671 if {[lsearch -exact $rowids $p] < 0} {
5672 # make this line extend up to the child
5673 set lineend($p) [drawlineseg $p $r $er 0]
5674 }
5675 }
5676 } else {
5677 set lineend($lid) [drawlineseg $lid $r $er 1]
5678 }
5679 }
5680 }
5681}
5682
5683proc undolayout {row} {
5684 global uparrowlen mingaplen downarrowlen
5685 global rowidlist rowisopt rowfinal need_redisplay
5686
5687 set r [expr {$row - ($uparrowlen + $mingaplen + $downarrowlen)}]
5688 if {$r < 0} {
5689 set r 0
5690 }
5691 if {[llength $rowidlist] > $r} {
5692 incr r -1
5693 set rowidlist [lrange $rowidlist 0 $r]
5694 set rowfinal [lrange $rowfinal 0 $r]
5695 set rowisopt [lrange $rowisopt 0 $r]
5696 set need_redisplay 1
5697 run drawvisible
5698 }
5699}
5700
5701proc drawvisible {} {
5702 global canv linespc curview vrowmod selectedline targetrow targetid
5703 global need_redisplay cscroll numcommits
5704
5705 set fs [$canv yview]
5706 set ymax [lindex [$canv cget -scrollregion] 3]
5707 if {$ymax eq {} || $ymax == 0 || $numcommits == 0} return
5708 set f0 [lindex $fs 0]
5709 set f1 [lindex $fs 1]
5710 set y0 [expr {int($f0 * $ymax)}]
5711 set y1 [expr {int($f1 * $ymax)}]
5712
5713 if {[info exists targetid]} {
5714 if {[commitinview $targetid $curview]} {
5715 set r [rowofcommit $targetid]
5716 if {$r != $targetrow} {
5717 # Fix up the scrollregion and change the scrolling position
5718 # now that our target row has moved.
5719 set diff [expr {($r - $targetrow) * $linespc}]
5720 set targetrow $r
5721 setcanvscroll
5722 set ymax [lindex [$canv cget -scrollregion] 3]
5723 incr y0 $diff
5724 incr y1 $diff
5725 set f0 [expr {$y0 / $ymax}]
5726 set f1 [expr {$y1 / $ymax}]
5727 allcanvs yview moveto $f0
5728 $cscroll set $f0 $f1
5729 set need_redisplay 1
5730 }
5731 } else {
5732 unset targetid
5733 }
5734 }
5735
5736 set row [expr {int(($y0 - 3) / $linespc) - 1}]
5737 set endrow [expr {int(($y1 - 3) / $linespc) + 1}]
5738 if {$endrow >= $vrowmod($curview)} {
5739 update_arcrows $curview
5740 }
5741 if {$selectedline ne {} &&
5742 $row <= $selectedline && $selectedline <= $endrow} {
5743 set targetrow $selectedline
5744 } elseif {[info exists targetid]} {
5745 set targetrow [expr {int(($row + $endrow) / 2)}]
5746 }
5747 if {[info exists targetrow]} {
5748 if {$targetrow >= $numcommits} {
5749 set targetrow [expr {$numcommits - 1}]
5750 }
5751 set targetid [commitonrow $targetrow]
5752 }
5753 drawcommits $row $endrow
5754}
5755
5756proc clear_display {} {
5757 global iddrawn linesegs need_redisplay nrows_drawn
5758 global vhighlights fhighlights nhighlights rhighlights
5759 global linehtag linentag linedtag boldids boldnameids
5760
5761 allcanvs delete all
5762 catch {unset iddrawn}
5763 catch {unset linesegs}
5764 catch {unset linehtag}
5765 catch {unset linentag}
5766 catch {unset linedtag}
5767 set boldids {}
5768 set boldnameids {}
5769 catch {unset vhighlights}
5770 catch {unset fhighlights}
5771 catch {unset nhighlights}
5772 catch {unset rhighlights}
5773 set need_redisplay 0
5774 set nrows_drawn 0
5775}
5776
5777proc findcrossings {id} {
5778 global rowidlist parentlist numcommits displayorder
5779
5780 set cross {}
5781 set ccross {}
5782 foreach {s e} [rowranges $id] {
5783 if {$e >= $numcommits} {
5784 set e [expr {$numcommits - 1}]
5785 }
5786 if {$e <= $s} continue
5787 for {set row $e} {[incr row -1] >= $s} {} {
5788 set x [lsearch -exact [lindex $rowidlist $row] $id]
5789 if {$x < 0} break
5790 set olds [lindex $parentlist $row]
5791 set kid [lindex $displayorder $row]
5792 set kidx [lsearch -exact [lindex $rowidlist $row] $kid]
5793 if {$kidx < 0} continue
5794 set nextrow [lindex $rowidlist [expr {$row + 1}]]
5795 foreach p $olds {
5796 set px [lsearch -exact $nextrow $p]
5797 if {$px < 0} continue
5798 if {($kidx < $x && $x < $px) || ($px < $x && $x < $kidx)} {
5799 if {[lsearch -exact $ccross $p] >= 0} continue
5800 if {$x == $px + ($kidx < $px? -1: 1)} {
5801 lappend ccross $p
5802 } elseif {[lsearch -exact $cross $p] < 0} {
5803 lappend cross $p
5804 }
5805 }
5806 }
5807 }
5808 }
5809 return [concat $ccross {{}} $cross]
5810}
5811
5812proc assigncolor {id} {
5813 global colormap colors nextcolor
5814 global parents children children curview
5815
5816 if {[info exists colormap($id)]} return
5817 set ncolors [llength $colors]
5818 if {[info exists children($curview,$id)]} {
5819 set kids $children($curview,$id)
5820 } else {
5821 set kids {}
5822 }
5823 if {[llength $kids] == 1} {
5824 set child [lindex $kids 0]
5825 if {[info exists colormap($child)]
5826 && [llength $parents($curview,$child)] == 1} {
5827 set colormap($id) $colormap($child)
5828 return
5829 }
5830 }
5831 set badcolors {}
5832 set origbad {}
5833 foreach x [findcrossings $id] {
5834 if {$x eq {}} {
5835 # delimiter between corner crossings and other crossings
5836 if {[llength $badcolors] >= $ncolors - 1} break
5837 set origbad $badcolors
5838 }
5839 if {[info exists colormap($x)]
5840 && [lsearch -exact $badcolors $colormap($x)] < 0} {
5841 lappend badcolors $colormap($x)
5842 }
5843 }
5844 if {[llength $badcolors] >= $ncolors} {
5845 set badcolors $origbad
5846 }
5847 set origbad $badcolors
5848 if {[llength $badcolors] < $ncolors - 1} {
5849 foreach child $kids {
5850 if {[info exists colormap($child)]
5851 && [lsearch -exact $badcolors $colormap($child)] < 0} {
5852 lappend badcolors $colormap($child)
5853 }
5854 foreach p $parents($curview,$child) {
5855 if {[info exists colormap($p)]
5856 && [lsearch -exact $badcolors $colormap($p)] < 0} {
5857 lappend badcolors $colormap($p)
5858 }
5859 }
5860 }
5861 if {[llength $badcolors] >= $ncolors} {
5862 set badcolors $origbad
5863 }
5864 }
5865 for {set i 0} {$i <= $ncolors} {incr i} {
5866 set c [lindex $colors $nextcolor]
5867 if {[incr nextcolor] >= $ncolors} {
5868 set nextcolor 0
5869 }
5870 if {[lsearch -exact $badcolors $c]} break
5871 }
5872 set colormap($id) $c
5873}
5874
5875proc bindline {t id} {
5876 global canv
5877
5878 $canv bind $t <Enter> "lineenter %x %y $id"
5879 $canv bind $t <Motion> "linemotion %x %y $id"
5880 $canv bind $t <Leave> "lineleave $id"
5881 $canv bind $t <Button-1> "lineclick %x %y $id 1"
5882}
5883
5884proc drawtags {id x xt y1} {
5885 global idtags idheads idotherrefs mainhead
5886 global linespc lthickness
5887 global canv rowtextx curview fgcolor bgcolor ctxbut
5888
5889 set marks {}
5890 set ntags 0
5891 set nheads 0
5892 if {[info exists idtags($id)]} {
5893 set marks $idtags($id)
5894 set ntags [llength $marks]
5895 }
5896 if {[info exists idheads($id)]} {
5897 set marks [concat $marks $idheads($id)]
5898 set nheads [llength $idheads($id)]
5899 }
5900 if {[info exists idotherrefs($id)]} {
5901 set marks [concat $marks $idotherrefs($id)]
5902 }
5903 if {$marks eq {}} {
5904 return $xt
5905 }
5906
5907 set delta [expr {int(0.5 * ($linespc - $lthickness))}]
5908 set yt [expr {$y1 - 0.5 * $linespc}]
5909 set yb [expr {$yt + $linespc - 1}]
5910 set xvals {}
5911 set wvals {}
5912 set i -1
5913 foreach tag $marks {
5914 incr i
5915 if {$i >= $ntags && $i < $ntags + $nheads && $tag eq $mainhead} {
5916 set wid [font measure mainfontbold $tag]
5917 } else {
5918 set wid [font measure mainfont $tag]
5919 }
5920 lappend xvals $xt
5921 lappend wvals $wid
5922 set xt [expr {$xt + $delta + $wid + $lthickness + $linespc}]
5923 }
5924 set t [$canv create line $x $y1 [lindex $xvals end] $y1 \
5925 -width $lthickness -fill black -tags tag.$id]
5926 $canv lower $t
5927 foreach tag $marks x $xvals wid $wvals {
5928 set xl [expr {$x + $delta}]
5929 set xr [expr {$x + $delta + $wid + $lthickness}]
5930 set font mainfont
5931 if {[incr ntags -1] >= 0} {
5932 # draw a tag
5933 set t [$canv create polygon $x [expr {$yt + $delta}] $xl $yt \
5934 $xr $yt $xr $yb $xl $yb $x [expr {$yb - $delta}] \
5935 -width 1 -outline black -fill yellow -tags tag.$id]
5936 $canv bind $t <1> [list showtag $tag 1]
5937 set rowtextx([rowofcommit $id]) [expr {$xr + $linespc}]
5938 } else {
5939 # draw a head or other ref
5940 if {[incr nheads -1] >= 0} {
5941 set col green
5942 if {$tag eq $mainhead} {
5943 set font mainfontbold
5944 }
5945 } else {
5946 set col "#ddddff"
5947 }
5948 set xl [expr {$xl - $delta/2}]
5949 $canv create polygon $x $yt $xr $yt $xr $yb $x $yb \
5950 -width 1 -outline black -fill $col -tags tag.$id
5951 if {[regexp {^(remotes/.*/|remotes/)} $tag match remoteprefix]} {
5952 set rwid [font measure mainfont $remoteprefix]
5953 set xi [expr {$x + 1}]
5954 set yti [expr {$yt + 1}]
5955 set xri [expr {$x + $rwid}]
5956 $canv create polygon $xi $yti $xri $yti $xri $yb $xi $yb \
5957 -width 0 -fill "#ffddaa" -tags tag.$id
5958 }
5959 }
5960 set t [$canv create text $xl $y1 -anchor w -text $tag -fill $fgcolor \
5961 -font $font -tags [list tag.$id text]]
5962 if {$ntags >= 0} {
5963 $canv bind $t <1> [list showtag $tag 1]
5964 } elseif {$nheads >= 0} {
5965 $canv bind $t $ctxbut [list headmenu %X %Y $id $tag]
5966 }
5967 }
5968 return $xt
5969}
5970
5971proc xcoord {i level ln} {
5972 global canvx0 xspc1 xspc2
5973
5974 set x [expr {$canvx0 + $i * $xspc1($ln)}]
5975 if {$i > 0 && $i == $level} {
5976 set x [expr {$x + 0.5 * ($xspc2 - $xspc1($ln))}]
5977 } elseif {$i > $level} {
5978 set x [expr {$x + $xspc2 - $xspc1($ln)}]
5979 }
5980 return $x
5981}
5982
5983proc show_status {msg} {
5984 global canv fgcolor
5985
5986 clear_display
5987 $canv create text 3 3 -anchor nw -text $msg -font mainfont \
5988 -tags text -fill $fgcolor
5989}
5990
5991# Don't change the text pane cursor if it is currently the hand cursor,
5992# showing that we are over a sha1 ID link.
5993proc settextcursor {c} {
5994 global ctext curtextcursor
5995
5996 if {[$ctext cget -cursor] == $curtextcursor} {
5997 $ctext config -cursor $c
5998 }
5999 set curtextcursor $c
6000}
6001
6002proc nowbusy {what {name {}}} {
6003 global isbusy busyname statusw
6004
6005 if {[array names isbusy] eq {}} {
6006 . config -cursor watch
6007 settextcursor watch
6008 }
6009 set isbusy($what) 1
6010 set busyname($what) $name
6011 if {$name ne {}} {
6012 $statusw conf -text $name
6013 }
6014}
6015
6016proc notbusy {what} {
6017 global isbusy maincursor textcursor busyname statusw
6018
6019 catch {
6020 unset isbusy($what)
6021 if {$busyname($what) ne {} &&
6022 [$statusw cget -text] eq $busyname($what)} {
6023 $statusw conf -text {}
6024 }
6025 }
6026 if {[array names isbusy] eq {}} {
6027 . config -cursor $maincursor
6028 settextcursor $textcursor
6029 }
6030}
6031
6032proc findmatches {f} {
6033 global findtype findstring
6034 if {$findtype == [mc "Regexp"]} {
6035 set matches [regexp -indices -all -inline $findstring $f]
6036 } else {
6037 set fs $findstring
6038 if {$findtype == [mc "IgnCase"]} {
6039 set f [string tolower $f]
6040 set fs [string tolower $fs]
6041 }
6042 set matches {}
6043 set i 0
6044 set l [string length $fs]
6045 while {[set j [string first $fs $f $i]] >= 0} {
6046 lappend matches [list $j [expr {$j+$l-1}]]
6047 set i [expr {$j + $l}]
6048 }
6049 }
6050 return $matches
6051}
6052
6053proc dofind {{dirn 1} {wrap 1}} {
6054 global findstring findstartline findcurline selectedline numcommits
6055 global gdttype filehighlight fh_serial find_dirn findallowwrap
6056
6057 if {[info exists find_dirn]} {
6058 if {$find_dirn == $dirn} return
6059 stopfinding
6060 }
6061 focus .
6062 if {$findstring eq {} || $numcommits == 0} return
6063 if {$selectedline eq {}} {
6064 set findstartline [lindex [visiblerows] [expr {$dirn < 0}]]
6065 } else {
6066 set findstartline $selectedline
6067 }
6068 set findcurline $findstartline
6069 nowbusy finding [mc "Searching"]
6070 if {$gdttype ne [mc "containing:"] && ![info exists filehighlight]} {
6071 after cancel do_file_hl $fh_serial
6072 do_file_hl $fh_serial
6073 }
6074 set find_dirn $dirn
6075 set findallowwrap $wrap
6076 run findmore
6077}
6078
6079proc stopfinding {} {
6080 global find_dirn findcurline fprogcoord
6081
6082 if {[info exists find_dirn]} {
6083 unset find_dirn
6084 unset findcurline
6085 notbusy finding
6086 set fprogcoord 0
6087 adjustprogress
6088 }
6089 stopblaming
6090}
6091
6092proc findmore {} {
6093 global commitdata commitinfo numcommits findpattern findloc
6094 global findstartline findcurline findallowwrap
6095 global find_dirn gdttype fhighlights fprogcoord
6096 global curview varcorder vrownum varccommits vrowmod
6097
6098 if {![info exists find_dirn]} {
6099 return 0
6100 }
6101 set fldtypes [list [mc "Headline"] [mc "Author"] [mc "Date"] [mc "Committer"] [mc "CDate"] [mc "Comments"]]
6102 set l $findcurline
6103 set moretodo 0
6104 if {$find_dirn > 0} {
6105 incr l
6106 if {$l >= $numcommits} {
6107 set l 0
6108 }
6109 if {$l <= $findstartline} {
6110 set lim [expr {$findstartline + 1}]
6111 } else {
6112 set lim $numcommits
6113 set moretodo $findallowwrap
6114 }
6115 } else {
6116 if {$l == 0} {
6117 set l $numcommits
6118 }
6119 incr l -1
6120 if {$l >= $findstartline} {
6121 set lim [expr {$findstartline - 1}]
6122 } else {
6123 set lim -1
6124 set moretodo $findallowwrap
6125 }
6126 }
6127 set n [expr {($lim - $l) * $find_dirn}]
6128 if {$n > 500} {
6129 set n 500
6130 set moretodo 1
6131 }
6132 if {$l + ($find_dirn > 0? $n: 1) > $vrowmod($curview)} {
6133 update_arcrows $curview
6134 }
6135 set found 0
6136 set domore 1
6137 set ai [bsearch $vrownum($curview) $l]
6138 set a [lindex $varcorder($curview) $ai]
6139 set arow [lindex $vrownum($curview) $ai]
6140 set ids [lindex $varccommits($curview,$a)]
6141 set arowend [expr {$arow + [llength $ids]}]
6142 if {$gdttype eq [mc "containing:"]} {
6143 for {} {$n > 0} {incr n -1; incr l $find_dirn} {
6144 if {$l < $arow || $l >= $arowend} {
6145 incr ai $find_dirn
6146 set a [lindex $varcorder($curview) $ai]
6147 set arow [lindex $vrownum($curview) $ai]
6148 set ids [lindex $varccommits($curview,$a)]
6149 set arowend [expr {$arow + [llength $ids]}]
6150 }
6151 set id [lindex $ids [expr {$l - $arow}]]
6152 # shouldn't happen unless git log doesn't give all the commits...
6153 if {![info exists commitdata($id)] ||
6154 ![doesmatch $commitdata($id)]} {
6155 continue
6156 }
6157 if {![info exists commitinfo($id)]} {
6158 getcommit $id
6159 }
6160 set info $commitinfo($id)
6161 foreach f $info ty $fldtypes {
6162 if {($findloc eq [mc "All fields"] || $findloc eq $ty) &&
6163 [doesmatch $f]} {
6164 set found 1
6165 break
6166 }
6167 }
6168 if {$found} break
6169 }
6170 } else {
6171 for {} {$n > 0} {incr n -1; incr l $find_dirn} {
6172 if {$l < $arow || $l >= $arowend} {
6173 incr ai $find_dirn
6174 set a [lindex $varcorder($curview) $ai]
6175 set arow [lindex $vrownum($curview) $ai]
6176 set ids [lindex $varccommits($curview,$a)]
6177 set arowend [expr {$arow + [llength $ids]}]
6178 }
6179 set id [lindex $ids [expr {$l - $arow}]]
6180 if {![info exists fhighlights($id)]} {
6181 # this sets fhighlights($id) to -1
6182 askfilehighlight $l $id
6183 }
6184 if {$fhighlights($id) > 0} {
6185 set found $domore
6186 break
6187 }
6188 if {$fhighlights($id) < 0} {
6189 if {$domore} {
6190 set domore 0
6191 set findcurline [expr {$l - $find_dirn}]
6192 }
6193 }
6194 }
6195 }
6196 if {$found || ($domore && !$moretodo)} {
6197 unset findcurline
6198 unset find_dirn
6199 notbusy finding
6200 set fprogcoord 0
6201 adjustprogress
6202 if {$found} {
6203 findselectline $l
6204 } else {
6205 bell
6206 }
6207 return 0
6208 }
6209 if {!$domore} {
6210 flushhighlights
6211 } else {
6212 set findcurline [expr {$l - $find_dirn}]
6213 }
6214 set n [expr {($findcurline - $findstartline) * $find_dirn - 1}]
6215 if {$n < 0} {
6216 incr n $numcommits
6217 }
6218 set fprogcoord [expr {$n * 1.0 / $numcommits}]
6219 adjustprogress
6220 return $domore
6221}
6222
6223proc findselectline {l} {
6224 global findloc commentend ctext findcurline markingmatches gdttype
6225
6226 set markingmatches 1
6227 set findcurline $l
6228 selectline $l 1
6229 if {$findloc == [mc "All fields"] || $findloc == [mc "Comments"]} {
6230 # highlight the matches in the comments
6231 set f [$ctext get 1.0 $commentend]
6232 set matches [findmatches $f]
6233 foreach match $matches {
6234 set start [lindex $match 0]
6235 set end [expr {[lindex $match 1] + 1}]
6236 $ctext tag add found "1.0 + $start c" "1.0 + $end c"
6237 }
6238 }
6239 drawvisible
6240}
6241
6242# mark the bits of a headline or author that match a find string
6243proc markmatches {canv l str tag matches font row} {
6244 global selectedline
6245
6246 set bbox [$canv bbox $tag]
6247 set x0 [lindex $bbox 0]
6248 set y0 [lindex $bbox 1]
6249 set y1 [lindex $bbox 3]
6250 foreach match $matches {
6251 set start [lindex $match 0]
6252 set end [lindex $match 1]
6253 if {$start > $end} continue
6254 set xoff [font measure $font [string range $str 0 [expr {$start-1}]]]
6255 set xlen [font measure $font [string range $str 0 [expr {$end}]]]
6256 set t [$canv create rect [expr {$x0+$xoff}] $y0 \
6257 [expr {$x0+$xlen+2}] $y1 \
6258 -outline {} -tags [list match$l matches] -fill yellow]
6259 $canv lower $t
6260 if {$row == $selectedline} {
6261 $canv raise $t secsel
6262 }
6263 }
6264}
6265
6266proc unmarkmatches {} {
6267 global markingmatches
6268
6269 allcanvs delete matches
6270 set markingmatches 0
6271 stopfinding
6272}
6273
6274proc selcanvline {w x y} {
6275 global canv canvy0 ctext linespc
6276 global rowtextx
6277 set ymax [lindex [$canv cget -scrollregion] 3]
6278 if {$ymax == {}} return
6279 set yfrac [lindex [$canv yview] 0]
6280 set y [expr {$y + $yfrac * $ymax}]
6281 set l [expr {int(($y - $canvy0) / $linespc + 0.5)}]
6282 if {$l < 0} {
6283 set l 0
6284 }
6285 if {$w eq $canv} {
6286 set xmax [lindex [$canv cget -scrollregion] 2]
6287 set xleft [expr {[lindex [$canv xview] 0] * $xmax}]
6288 if {![info exists rowtextx($l)] || $xleft + $x < $rowtextx($l)} return
6289 }
6290 unmarkmatches
6291 selectline $l 1
6292}
6293
6294proc commit_descriptor {p} {
6295 global commitinfo
6296 if {![info exists commitinfo($p)]} {
6297 getcommit $p
6298 }
6299 set l "..."
6300 if {[llength $commitinfo($p)] > 1} {
6301 set l [lindex $commitinfo($p) 0]
6302 }
6303 return "$p ($l)\n"
6304}
6305
6306# append some text to the ctext widget, and make any SHA1 ID
6307# that we know about be a clickable link.
6308proc appendwithlinks {text tags} {
6309 global ctext linknum curview
6310
6311 set start [$ctext index "end - 1c"]
6312 $ctext insert end $text $tags
6313 set links [regexp -indices -all -inline {\m[0-9a-f]{6,40}\M} $text]
6314 foreach l $links {
6315 set s [lindex $l 0]
6316 set e [lindex $l 1]
6317 set linkid [string range $text $s $e]
6318 incr e
6319 $ctext tag delete link$linknum
6320 $ctext tag add link$linknum "$start + $s c" "$start + $e c"
6321 setlink $linkid link$linknum
6322 incr linknum
6323 }
6324}
6325
6326proc setlink {id lk} {
6327 global curview ctext pendinglinks
6328
6329 set known 0
6330 if {[string length $id] < 40} {
6331 set matches [longid $id]
6332 if {[llength $matches] > 0} {
6333 if {[llength $matches] > 1} return
6334 set known 1
6335 set id [lindex $matches 0]
6336 }
6337 } else {
6338 set known [commitinview $id $curview]
6339 }
6340 if {$known} {
6341 $ctext tag conf $lk -foreground blue -underline 1
6342 $ctext tag bind $lk <1> [list selbyid $id]
6343 $ctext tag bind $lk <Enter> {linkcursor %W 1}
6344 $ctext tag bind $lk <Leave> {linkcursor %W -1}
6345 } else {
6346 lappend pendinglinks($id) $lk
6347 interestedin $id {makelink %P}
6348 }
6349}
6350
6351proc makelink {id} {
6352 global pendinglinks
6353
6354 if {![info exists pendinglinks($id)]} return
6355 foreach lk $pendinglinks($id) {
6356 setlink $id $lk
6357 }
6358 unset pendinglinks($id)
6359}
6360
6361proc linkcursor {w inc} {
6362 global linkentercount curtextcursor
6363
6364 if {[incr linkentercount $inc] > 0} {
6365 $w configure -cursor hand2
6366 } else {
6367 $w configure -cursor $curtextcursor
6368 if {$linkentercount < 0} {
6369 set linkentercount 0
6370 }
6371 }
6372}
6373
6374proc viewnextline {dir} {
6375 global canv linespc
6376
6377 $canv delete hover
6378 set ymax [lindex [$canv cget -scrollregion] 3]
6379 set wnow [$canv yview]
6380 set wtop [expr {[lindex $wnow 0] * $ymax}]
6381 set newtop [expr {$wtop + $dir * $linespc}]
6382 if {$newtop < 0} {
6383 set newtop 0
6384 } elseif {$newtop > $ymax} {
6385 set newtop $ymax
6386 }
6387 allcanvs yview moveto [expr {$newtop * 1.0 / $ymax}]
6388}
6389
6390# add a list of tag or branch names at position pos
6391# returns the number of names inserted
6392proc appendrefs {pos ids var} {
6393 global ctext linknum curview $var maxrefs
6394
6395 if {[catch {$ctext index $pos}]} {
6396 return 0
6397 }
6398 $ctext conf -state normal
6399 $ctext delete $pos "$pos lineend"
6400 set tags {}
6401 foreach id $ids {
6402 foreach tag [set $var\($id\)] {
6403 lappend tags [list $tag $id]
6404 }
6405 }
6406 if {[llength $tags] > $maxrefs} {
6407 $ctext insert $pos "many ([llength $tags])"
6408 } else {
6409 set tags [lsort -index 0 -decreasing $tags]
6410 set sep {}
6411 foreach ti $tags {
6412 set id [lindex $ti 1]
6413 set lk link$linknum
6414 incr linknum
6415 $ctext tag delete $lk
6416 $ctext insert $pos $sep
6417 $ctext insert $pos [lindex $ti 0] $lk
6418 setlink $id $lk
6419 set sep ", "
6420 }
6421 }
6422 $ctext conf -state disabled
6423 return [llength $tags]
6424}
6425
6426# called when we have finished computing the nearby tags
6427proc dispneartags {delay} {
6428 global selectedline currentid showneartags tagphase
6429
6430 if {$selectedline eq {} || !$showneartags} return
6431 after cancel dispnexttag
6432 if {$delay} {
6433 after 200 dispnexttag
6434 set tagphase -1
6435 } else {
6436 after idle dispnexttag
6437 set tagphase 0
6438 }
6439}
6440
6441proc dispnexttag {} {
6442 global selectedline currentid showneartags tagphase ctext
6443
6444 if {$selectedline eq {} || !$showneartags} return
6445 switch -- $tagphase {
6446 0 {
6447 set dtags [desctags $currentid]
6448 if {$dtags ne {}} {
6449 appendrefs precedes $dtags idtags
6450 }
6451 }
6452 1 {
6453 set atags [anctags $currentid]
6454 if {$atags ne {}} {
6455 appendrefs follows $atags idtags
6456 }
6457 }
6458 2 {
6459 set dheads [descheads $currentid]
6460 if {$dheads ne {}} {
6461 if {[appendrefs branch $dheads idheads] > 1
6462 && [$ctext get "branch -3c"] eq "h"} {
6463 # turn "Branch" into "Branches"
6464 $ctext conf -state normal
6465 $ctext insert "branch -2c" "es"
6466 $ctext conf -state disabled
6467 }
6468 }
6469 }
6470 }
6471 if {[incr tagphase] <= 2} {
6472 after idle dispnexttag
6473 }
6474}
6475
6476proc make_secsel {id} {
6477 global linehtag linentag linedtag canv canv2 canv3
6478
6479 if {![info exists linehtag($id)]} return
6480 $canv delete secsel
6481 set t [eval $canv create rect [$canv bbox $linehtag($id)] -outline {{}} \
6482 -tags secsel -fill [$canv cget -selectbackground]]
6483 $canv lower $t
6484 $canv2 delete secsel
6485 set t [eval $canv2 create rect [$canv2 bbox $linentag($id)] -outline {{}} \
6486 -tags secsel -fill [$canv2 cget -selectbackground]]
6487 $canv2 lower $t
6488 $canv3 delete secsel
6489 set t [eval $canv3 create rect [$canv3 bbox $linedtag($id)] -outline {{}} \
6490 -tags secsel -fill [$canv3 cget -selectbackground]]
6491 $canv3 lower $t
6492}
6493
6494proc selectline {l isnew {desired_loc {}}} {
6495 global canv ctext commitinfo selectedline
6496 global canvy0 linespc parents children curview
6497 global currentid sha1entry
6498 global commentend idtags linknum
6499 global mergemax numcommits pending_select
6500 global cmitmode showneartags allcommits
6501 global targetrow targetid lastscrollrows
6502 global autoselect jump_to_here
6503
6504 catch {unset pending_select}
6505 $canv delete hover
6506 normalline
6507 unsel_reflist
6508 stopfinding
6509 if {$l < 0 || $l >= $numcommits} return
6510 set id [commitonrow $l]
6511 set targetid $id
6512 set targetrow $l
6513 set selectedline $l
6514 set currentid $id
6515 if {$lastscrollrows < $numcommits} {
6516 setcanvscroll
6517 }
6518
6519 set y [expr {$canvy0 + $l * $linespc}]
6520 set ymax [lindex [$canv cget -scrollregion] 3]
6521 set ytop [expr {$y - $linespc - 1}]
6522 set ybot [expr {$y + $linespc + 1}]
6523 set wnow [$canv yview]
6524 set wtop [expr {[lindex $wnow 0] * $ymax}]
6525 set wbot [expr {[lindex $wnow 1] * $ymax}]
6526 set wh [expr {$wbot - $wtop}]
6527 set newtop $wtop
6528 if {$ytop < $wtop} {
6529 if {$ybot < $wtop} {
6530 set newtop [expr {$y - $wh / 2.0}]
6531 } else {
6532 set newtop $ytop
6533 if {$newtop > $wtop - $linespc} {
6534 set newtop [expr {$wtop - $linespc}]
6535 }
6536 }
6537 } elseif {$ybot > $wbot} {
6538 if {$ytop > $wbot} {
6539 set newtop [expr {$y - $wh / 2.0}]
6540 } else {
6541 set newtop [expr {$ybot - $wh}]
6542 if {$newtop < $wtop + $linespc} {
6543 set newtop [expr {$wtop + $linespc}]
6544 }
6545 }
6546 }
6547 if {$newtop != $wtop} {
6548 if {$newtop < 0} {
6549 set newtop 0
6550 }
6551 allcanvs yview moveto [expr {$newtop * 1.0 / $ymax}]
6552 drawvisible
6553 }
6554
6555 make_secsel $id
6556
6557 if {$isnew} {
6558 addtohistory [list selbyid $id]
6559 }
6560
6561 $sha1entry delete 0 end
6562 $sha1entry insert 0 $id
6563 if {$autoselect} {
6564 $sha1entry selection from 0
6565 $sha1entry selection to end
6566 }
6567 rhighlight_sel $id
6568
6569 $ctext conf -state normal
6570 clear_ctext
6571 set linknum 0
6572 if {![info exists commitinfo($id)]} {
6573 getcommit $id
6574 }
6575 set info $commitinfo($id)
6576 set date [formatdate [lindex $info 2]]
6577 $ctext insert end "[mc "Author"]: [lindex $info 1] $date\n"
6578 set date [formatdate [lindex $info 4]]
6579 $ctext insert end "[mc "Committer"]: [lindex $info 3] $date\n"
6580 if {[info exists idtags($id)]} {
6581 $ctext insert end [mc "Tags:"]
6582 foreach tag $idtags($id) {
6583 $ctext insert end " $tag"
6584 }
6585 $ctext insert end "\n"
6586 }
6587
6588 set headers {}
6589 set olds $parents($curview,$id)
6590 if {[llength $olds] > 1} {
6591 set np 0
6592 foreach p $olds {
6593 if {$np >= $mergemax} {
6594 set tag mmax
6595 } else {
6596 set tag m$np
6597 }
6598 $ctext insert end "[mc "Parent"]: " $tag
6599 appendwithlinks [commit_descriptor $p] {}
6600 incr np
6601 }
6602 } else {
6603 foreach p $olds {
6604 append headers "[mc "Parent"]: [commit_descriptor $p]"
6605 }
6606 }
6607
6608 foreach c $children($curview,$id) {
6609 append headers "[mc "Child"]: [commit_descriptor $c]"
6610 }
6611
6612 # make anything that looks like a SHA1 ID be a clickable link
6613 appendwithlinks $headers {}
6614 if {$showneartags} {
6615 if {![info exists allcommits]} {
6616 getallcommits
6617 }
6618 $ctext insert end "[mc "Branch"]: "
6619 $ctext mark set branch "end -1c"
6620 $ctext mark gravity branch left
6621 $ctext insert end "\n[mc "Follows"]: "
6622 $ctext mark set follows "end -1c"
6623 $ctext mark gravity follows left
6624 $ctext insert end "\n[mc "Precedes"]: "
6625 $ctext mark set precedes "end -1c"
6626 $ctext mark gravity precedes left
6627 $ctext insert end "\n"
6628 dispneartags 1
6629 }
6630 $ctext insert end "\n"
6631 set comment [lindex $info 5]
6632 if {[string first "\r" $comment] >= 0} {
6633 set comment [string map {"\r" "\n "} $comment]
6634 }
6635 appendwithlinks $comment {comment}
6636
6637 $ctext tag remove found 1.0 end
6638 $ctext conf -state disabled
6639 set commentend [$ctext index "end - 1c"]
6640
6641 set jump_to_here $desired_loc
6642 init_flist [mc "Comments"]
6643 if {$cmitmode eq "tree"} {
6644 gettree $id
6645 } elseif {[llength $olds] <= 1} {
6646 startdiff $id
6647 } else {
6648 mergediff $id
6649 }
6650}
6651
6652proc selfirstline {} {
6653 unmarkmatches
6654 selectline 0 1
6655}
6656
6657proc sellastline {} {
6658 global numcommits
6659 unmarkmatches
6660 set l [expr {$numcommits - 1}]
6661 selectline $l 1
6662}
6663
6664proc selnextline {dir} {
6665 global selectedline
6666 focus .
6667 if {$selectedline eq {}} return
6668 set l [expr {$selectedline + $dir}]
6669 unmarkmatches
6670 selectline $l 1
6671}
6672
6673proc selnextpage {dir} {
6674 global canv linespc selectedline numcommits
6675
6676 set lpp [expr {([winfo height $canv] - 2) / $linespc}]
6677 if {$lpp < 1} {
6678 set lpp 1
6679 }
6680 allcanvs yview scroll [expr {$dir * $lpp}] units
6681 drawvisible
6682 if {$selectedline eq {}} return
6683 set l [expr {$selectedline + $dir * $lpp}]
6684 if {$l < 0} {
6685 set l 0
6686 } elseif {$l >= $numcommits} {
6687 set l [expr $numcommits - 1]
6688 }
6689 unmarkmatches
6690 selectline $l 1
6691}
6692
6693proc unselectline {} {
6694 global selectedline currentid
6695
6696 set selectedline {}
6697 catch {unset currentid}
6698 allcanvs delete secsel
6699 rhighlight_none
6700}
6701
6702proc reselectline {} {
6703 global selectedline
6704
6705 if {$selectedline ne {}} {
6706 selectline $selectedline 0
6707 }
6708}
6709
6710proc addtohistory {cmd} {
6711 global history historyindex curview
6712
6713 set elt [list $curview $cmd]
6714 if {$historyindex > 0
6715 && [lindex $history [expr {$historyindex - 1}]] == $elt} {
6716 return
6717 }
6718
6719 if {$historyindex < [llength $history]} {
6720 set history [lreplace $history $historyindex end $elt]
6721 } else {
6722 lappend history $elt
6723 }
6724 incr historyindex
6725 if {$historyindex > 1} {
6726 .tf.bar.leftbut conf -state normal
6727 } else {
6728 .tf.bar.leftbut conf -state disabled
6729 }
6730 .tf.bar.rightbut conf -state disabled
6731}
6732
6733proc godo {elt} {
6734 global curview
6735
6736 set view [lindex $elt 0]
6737 set cmd [lindex $elt 1]
6738 if {$curview != $view} {
6739 showview $view
6740 }
6741 eval $cmd
6742}
6743
6744proc goback {} {
6745 global history historyindex
6746 focus .
6747
6748 if {$historyindex > 1} {
6749 incr historyindex -1
6750 godo [lindex $history [expr {$historyindex - 1}]]
6751 .tf.bar.rightbut conf -state normal
6752 }
6753 if {$historyindex <= 1} {
6754 .tf.bar.leftbut conf -state disabled
6755 }
6756}
6757
6758proc goforw {} {
6759 global history historyindex
6760 focus .
6761
6762 if {$historyindex < [llength $history]} {
6763 set cmd [lindex $history $historyindex]
6764 incr historyindex
6765 godo $cmd
6766 .tf.bar.leftbut conf -state normal
6767 }
6768 if {$historyindex >= [llength $history]} {
6769 .tf.bar.rightbut conf -state disabled
6770 }
6771}
6772
6773proc gettree {id} {
6774 global treefilelist treeidlist diffids diffmergeid treepending
6775 global nullid nullid2
6776
6777 set diffids $id
6778 catch {unset diffmergeid}
6779 if {![info exists treefilelist($id)]} {
6780 if {![info exists treepending]} {
6781 if {$id eq $nullid} {
6782 set cmd [list | git ls-files]
6783 } elseif {$id eq $nullid2} {
6784 set cmd [list | git ls-files --stage -t]
6785 } else {
6786 set cmd [list | git ls-tree -r $id]
6787 }
6788 if {[catch {set gtf [open $cmd r]}]} {
6789 return
6790 }
6791 set treepending $id
6792 set treefilelist($id) {}
6793 set treeidlist($id) {}
6794 fconfigure $gtf -blocking 0 -encoding binary
6795 filerun $gtf [list gettreeline $gtf $id]
6796 }
6797 } else {
6798 setfilelist $id
6799 }
6800}
6801
6802proc gettreeline {gtf id} {
6803 global treefilelist treeidlist treepending cmitmode diffids nullid nullid2
6804
6805 set nl 0
6806 while {[incr nl] <= 1000 && [gets $gtf line] >= 0} {
6807 if {$diffids eq $nullid} {
6808 set fname $line
6809 } else {
6810 set i [string first "\t" $line]
6811 if {$i < 0} continue
6812 set fname [string range $line [expr {$i+1}] end]
6813 set line [string range $line 0 [expr {$i-1}]]
6814 if {$diffids ne $nullid2 && [lindex $line 1] ne "blob"} continue
6815 set sha1 [lindex $line 2]
6816 lappend treeidlist($id) $sha1
6817 }
6818 if {[string index $fname 0] eq "\""} {
6819 set fname [lindex $fname 0]
6820 }
6821 set fname [encoding convertfrom $fname]
6822 lappend treefilelist($id) $fname
6823 }
6824 if {![eof $gtf]} {
6825 return [expr {$nl >= 1000? 2: 1}]
6826 }
6827 close $gtf
6828 unset treepending
6829 if {$cmitmode ne "tree"} {
6830 if {![info exists diffmergeid]} {
6831 gettreediffs $diffids
6832 }
6833 } elseif {$id ne $diffids} {
6834 gettree $diffids
6835 } else {
6836 setfilelist $id
6837 }
6838 return 0
6839}
6840
6841proc showfile {f} {
6842 global treefilelist treeidlist diffids nullid nullid2
6843 global ctext_file_names ctext_file_lines
6844 global ctext commentend
6845
6846 set i [lsearch -exact $treefilelist($diffids) $f]
6847 if {$i < 0} {
6848 puts "oops, $f not in list for id $diffids"
6849 return
6850 }
6851 if {$diffids eq $nullid} {
6852 if {[catch {set bf [open $f r]} err]} {
6853 puts "oops, can't read $f: $err"
6854 return
6855 }
6856 } else {
6857 set blob [lindex $treeidlist($diffids) $i]
6858 if {[catch {set bf [open [concat | git cat-file blob $blob] r]} err]} {
6859 puts "oops, error reading blob $blob: $err"
6860 return
6861 }
6862 }
6863 fconfigure $bf -blocking 0 -encoding [get_path_encoding $f]
6864 filerun $bf [list getblobline $bf $diffids]
6865 $ctext config -state normal
6866 clear_ctext $commentend
6867 lappend ctext_file_names $f
6868 lappend ctext_file_lines [lindex [split $commentend "."] 0]
6869 $ctext insert end "\n"
6870 $ctext insert end "$f\n" filesep
6871 $ctext config -state disabled
6872 $ctext yview $commentend
6873 settabs 0
6874}
6875
6876proc getblobline {bf id} {
6877 global diffids cmitmode ctext
6878
6879 if {$id ne $diffids || $cmitmode ne "tree"} {
6880 catch {close $bf}
6881 return 0
6882 }
6883 $ctext config -state normal
6884 set nl 0
6885 while {[incr nl] <= 1000 && [gets $bf line] >= 0} {
6886 $ctext insert end "$line\n"
6887 }
6888 if {[eof $bf]} {
6889 global jump_to_here ctext_file_names commentend
6890
6891 # delete last newline
6892 $ctext delete "end - 2c" "end - 1c"
6893 close $bf
6894 if {$jump_to_here ne {} &&
6895 [lindex $jump_to_here 0] eq [lindex $ctext_file_names 0]} {
6896 set lnum [expr {[lindex $jump_to_here 1] +
6897 [lindex [split $commentend .] 0]}]
6898 mark_ctext_line $lnum
6899 }
6900 return 0
6901 }
6902 $ctext config -state disabled
6903 return [expr {$nl >= 1000? 2: 1}]
6904}
6905
6906proc mark_ctext_line {lnum} {
6907 global ctext markbgcolor
6908
6909 $ctext tag delete omark
6910 $ctext tag add omark $lnum.0 "$lnum.0 + 1 line"
6911 $ctext tag conf omark -background $markbgcolor
6912 $ctext see $lnum.0
6913}
6914
6915proc mergediff {id} {
6916 global diffmergeid
6917 global diffids treediffs
6918 global parents curview
6919
6920 set diffmergeid $id
6921 set diffids $id
6922 set treediffs($id) {}
6923 set np [llength $parents($curview,$id)]
6924 settabs $np
6925 getblobdiffs $id
6926}
6927
6928proc startdiff {ids} {
6929 global treediffs diffids treepending diffmergeid nullid nullid2
6930
6931 settabs 1
6932 set diffids $ids
6933 catch {unset diffmergeid}
6934 if {![info exists treediffs($ids)] ||
6935 [lsearch -exact $ids $nullid] >= 0 ||
6936 [lsearch -exact $ids $nullid2] >= 0} {
6937 if {![info exists treepending]} {
6938 gettreediffs $ids
6939 }
6940 } else {
6941 addtocflist $ids
6942 }
6943}
6944
6945proc path_filter {filter name} {
6946 foreach p $filter {
6947 set l [string length $p]
6948 if {[string index $p end] eq "/"} {
6949 if {[string compare -length $l $p $name] == 0} {
6950 return 1
6951 }
6952 } else {
6953 if {[string compare -length $l $p $name] == 0 &&
6954 ([string length $name] == $l ||
6955 [string index $name $l] eq "/")} {
6956 return 1
6957 }
6958 }
6959 }
6960 return 0
6961}
6962
6963proc addtocflist {ids} {
6964 global treediffs
6965
6966 add_flist $treediffs($ids)
6967 getblobdiffs $ids
6968}
6969
6970proc diffcmd {ids flags} {
6971 global nullid nullid2
6972
6973 set i [lsearch -exact $ids $nullid]
6974 set j [lsearch -exact $ids $nullid2]
6975 if {$i >= 0} {
6976 if {[llength $ids] > 1 && $j < 0} {
6977 # comparing working directory with some specific revision
6978 set cmd [concat | git diff-index $flags]
6979 if {$i == 0} {
6980 lappend cmd -R [lindex $ids 1]
6981 } else {
6982 lappend cmd [lindex $ids 0]
6983 }
6984 } else {
6985 # comparing working directory with index
6986 set cmd [concat | git diff-files $flags]
6987 if {$j == 1} {
6988 lappend cmd -R
6989 }
6990 }
6991 } elseif {$j >= 0} {
6992 set cmd [concat | git diff-index --cached $flags]
6993 if {[llength $ids] > 1} {
6994 # comparing index with specific revision
6995 if {$i == 0} {
6996 lappend cmd -R [lindex $ids 1]
6997 } else {
6998 lappend cmd [lindex $ids 0]
6999 }
7000 } else {
7001 # comparing index with HEAD
7002 lappend cmd HEAD
7003 }
7004 } else {
7005 set cmd [concat | git diff-tree -r $flags $ids]
7006 }
7007 return $cmd
7008}
7009
7010proc gettreediffs {ids} {
7011 global treediff treepending
7012
7013 if {[catch {set gdtf [open [diffcmd $ids {--no-commit-id}] r]}]} return
7014
7015 set treepending $ids
7016 set treediff {}
7017 fconfigure $gdtf -blocking 0 -encoding binary
7018 filerun $gdtf [list gettreediffline $gdtf $ids]
7019}
7020
7021proc gettreediffline {gdtf ids} {
7022 global treediff treediffs treepending diffids diffmergeid
7023 global cmitmode vfilelimit curview limitdiffs perfile_attrs
7024
7025 set nr 0
7026 set sublist {}
7027 set max 1000
7028 if {$perfile_attrs} {
7029 # cache_gitattr is slow, and even slower on win32 where we
7030 # have to invoke it for only about 30 paths at a time
7031 set max 500
7032 if {[tk windowingsystem] == "win32"} {
7033 set max 120
7034 }
7035 }
7036 while {[incr nr] <= $max && [gets $gdtf line] >= 0} {
7037 set i [string first "\t" $line]
7038 if {$i >= 0} {
7039 set file [string range $line [expr {$i+1}] end]
7040 if {[string index $file 0] eq "\""} {
7041 set file [lindex $file 0]
7042 }
7043 set file [encoding convertfrom $file]
7044 if {$file ne [lindex $treediff end]} {
7045 lappend treediff $file
7046 lappend sublist $file
7047 }
7048 }
7049 }
7050 if {$perfile_attrs} {
7051 cache_gitattr encoding $sublist
7052 }
7053 if {![eof $gdtf]} {
7054 return [expr {$nr >= $max? 2: 1}]
7055 }
7056 close $gdtf
7057 if {$limitdiffs && $vfilelimit($curview) ne {}} {
7058 set flist {}
7059 foreach f $treediff {
7060 if {[path_filter $vfilelimit($curview) $f]} {
7061 lappend flist $f
7062 }
7063 }
7064 set treediffs($ids) $flist
7065 } else {
7066 set treediffs($ids) $treediff
7067 }
7068 unset treepending
7069 if {$cmitmode eq "tree"} {
7070 gettree $diffids
7071 } elseif {$ids != $diffids} {
7072 if {![info exists diffmergeid]} {
7073 gettreediffs $diffids
7074 }
7075 } else {
7076 addtocflist $ids
7077 }
7078 return 0
7079}
7080
7081# empty string or positive integer
7082proc diffcontextvalidate {v} {
7083 return [regexp {^(|[1-9][0-9]*)$} $v]
7084}
7085
7086proc diffcontextchange {n1 n2 op} {
7087 global diffcontextstring diffcontext
7088
7089 if {[string is integer -strict $diffcontextstring]} {
7090 if {$diffcontextstring > 0} {
7091 set diffcontext $diffcontextstring
7092 reselectline
7093 }
7094 }
7095}
7096
7097proc changeignorespace {} {
7098 reselectline
7099}
7100
7101proc getblobdiffs {ids} {
7102 global blobdifffd diffids env
7103 global diffinhdr treediffs
7104 global diffcontext
7105 global ignorespace
7106 global limitdiffs vfilelimit curview
7107 global diffencoding targetline diffnparents
7108
7109 set cmd [diffcmd $ids "-p -C --cc --no-commit-id -U$diffcontext"]
7110 if {$ignorespace} {
7111 append cmd " -w"
7112 }
7113 if {$limitdiffs && $vfilelimit($curview) ne {}} {
7114 set cmd [concat $cmd -- $vfilelimit($curview)]
7115 }
7116 if {[catch {set bdf [open $cmd r]} err]} {
7117 error_popup [mc "Error getting diffs: %s" $err]
7118 return
7119 }
7120 set targetline {}
7121 set diffnparents 0
7122 set diffinhdr 0
7123 set diffencoding [get_path_encoding {}]
7124 fconfigure $bdf -blocking 0 -encoding binary
7125 set blobdifffd($ids) $bdf
7126 filerun $bdf [list getblobdiffline $bdf $diffids]
7127}
7128
7129proc setinlist {var i val} {
7130 global $var
7131
7132 while {[llength [set $var]] < $i} {
7133 lappend $var {}
7134 }
7135 if {[llength [set $var]] == $i} {
7136 lappend $var $val
7137 } else {
7138 lset $var $i $val
7139 }
7140}
7141
7142proc makediffhdr {fname ids} {
7143 global ctext curdiffstart treediffs diffencoding
7144 global ctext_file_names jump_to_here targetline diffline
7145
7146 set fname [encoding convertfrom $fname]
7147 set diffencoding [get_path_encoding $fname]
7148 set i [lsearch -exact $treediffs($ids) $fname]
7149 if {$i >= 0} {
7150 setinlist difffilestart $i $curdiffstart
7151 }
7152 lset ctext_file_names end $fname
7153 set l [expr {(78 - [string length $fname]) / 2}]
7154 set pad [string range "----------------------------------------" 1 $l]
7155 $ctext insert $curdiffstart "$pad $fname $pad" filesep
7156 set targetline {}
7157 if {$jump_to_here ne {} && [lindex $jump_to_here 0] eq $fname} {
7158 set targetline [lindex $jump_to_here 1]
7159 }
7160 set diffline 0
7161}
7162
7163proc getblobdiffline {bdf ids} {
7164 global diffids blobdifffd ctext curdiffstart
7165 global diffnexthead diffnextnote difffilestart
7166 global ctext_file_names ctext_file_lines
7167 global diffinhdr treediffs mergemax diffnparents
7168 global diffencoding jump_to_here targetline diffline
7169
7170 set nr 0
7171 $ctext conf -state normal
7172 while {[incr nr] <= 1000 && [gets $bdf line] >= 0} {
7173 if {$ids != $diffids || $bdf != $blobdifffd($ids)} {
7174 close $bdf
7175 return 0
7176 }
7177 if {![string compare -length 5 "diff " $line]} {
7178 if {![regexp {^diff (--cc|--git) } $line m type]} {
7179 set line [encoding convertfrom $line]
7180 $ctext insert end "$line\n" hunksep
7181 continue
7182 }
7183 # start of a new file
7184 set diffinhdr 1
7185 $ctext insert end "\n"
7186 set curdiffstart [$ctext index "end - 1c"]
7187 lappend ctext_file_names ""
7188 lappend ctext_file_lines [lindex [split $curdiffstart "."] 0]
7189 $ctext insert end "\n" filesep
7190
7191 if {$type eq "--cc"} {
7192 # start of a new file in a merge diff
7193 set fname [string range $line 10 end]
7194 if {[lsearch -exact $treediffs($ids) $fname] < 0} {
7195 lappend treediffs($ids) $fname
7196 add_flist [list $fname]
7197 }
7198
7199 } else {
7200 set line [string range $line 11 end]
7201 # If the name hasn't changed the length will be odd,
7202 # the middle char will be a space, and the two bits either
7203 # side will be a/name and b/name, or "a/name" and "b/name".
7204 # If the name has changed we'll get "rename from" and
7205 # "rename to" or "copy from" and "copy to" lines following
7206 # this, and we'll use them to get the filenames.
7207 # This complexity is necessary because spaces in the
7208 # filename(s) don't get escaped.
7209 set l [string length $line]
7210 set i [expr {$l / 2}]
7211 if {!(($l & 1) && [string index $line $i] eq " " &&
7212 [string range $line 2 [expr {$i - 1}]] eq \
7213 [string range $line [expr {$i + 3}] end])} {
7214 continue
7215 }
7216 # unescape if quoted and chop off the a/ from the front
7217 if {[string index $line 0] eq "\""} {
7218 set fname [string range [lindex $line 0] 2 end]
7219 } else {
7220 set fname [string range $line 2 [expr {$i - 1}]]
7221 }
7222 }
7223 makediffhdr $fname $ids
7224
7225 } elseif {![string compare -length 16 "* Unmerged path " $line]} {
7226 set fname [encoding convertfrom [string range $line 16 end]]
7227 $ctext insert end "\n"
7228 set curdiffstart [$ctext index "end - 1c"]
7229 lappend ctext_file_names $fname
7230 lappend ctext_file_lines [lindex [split $curdiffstart "."] 0]
7231 $ctext insert end "$line\n" filesep
7232 set i [lsearch -exact $treediffs($ids) $fname]
7233 if {$i >= 0} {
7234 setinlist difffilestart $i $curdiffstart
7235 }
7236
7237 } elseif {![string compare -length 2 "@@" $line]} {
7238 regexp {^@@+} $line ats
7239 set line [encoding convertfrom $diffencoding $line]
7240 $ctext insert end "$line\n" hunksep
7241 if {[regexp { \+(\d+),\d+ @@} $line m nl]} {
7242 set diffline $nl
7243 }
7244 set diffnparents [expr {[string length $ats] - 1}]
7245 set diffinhdr 0
7246
7247 } elseif {$diffinhdr} {
7248 if {![string compare -length 12 "rename from " $line]} {
7249 set fname [string range $line [expr 6 + [string first " from " $line] ] end]
7250 if {[string index $fname 0] eq "\""} {
7251 set fname [lindex $fname 0]
7252 }
7253 set fname [encoding convertfrom $fname]
7254 set i [lsearch -exact $treediffs($ids) $fname]
7255 if {$i >= 0} {
7256 setinlist difffilestart $i $curdiffstart
7257 }
7258 } elseif {![string compare -length 10 $line "rename to "] ||
7259 ![string compare -length 8 $line "copy to "]} {
7260 set fname [string range $line [expr 4 + [string first " to " $line] ] end]
7261 if {[string index $fname 0] eq "\""} {
7262 set fname [lindex $fname 0]
7263 }
7264 makediffhdr $fname $ids
7265 } elseif {[string compare -length 3 $line "---"] == 0} {
7266 # do nothing
7267 continue
7268 } elseif {[string compare -length 3 $line "+++"] == 0} {
7269 set diffinhdr 0
7270 continue
7271 }
7272 $ctext insert end "$line\n" filesep
7273
7274 } else {
7275 set line [encoding convertfrom $diffencoding $line]
7276 # parse the prefix - one ' ', '-' or '+' for each parent
7277 set prefix [string range $line 0 [expr {$diffnparents - 1}]]
7278 set tag [expr {$diffnparents > 1? "m": "d"}]
7279 if {[string trim $prefix " -+"] eq {}} {
7280 # prefix only has " ", "-" and "+" in it: normal diff line
7281 set num [string first "-" $prefix]
7282 if {$num >= 0} {
7283 # removed line, first parent with line is $num
7284 if {$num >= $mergemax} {
7285 set num "max"
7286 }
7287 $ctext insert end "$line\n" $tag$num
7288 } else {
7289 set tags {}
7290 if {[string first "+" $prefix] >= 0} {
7291 # added line
7292 lappend tags ${tag}result
7293 if {$diffnparents > 1} {
7294 set num [string first " " $prefix]
7295 if {$num >= 0} {
7296 if {$num >= $mergemax} {
7297 set num "max"
7298 }
7299 lappend tags m$num
7300 }
7301 }
7302 }
7303 if {$targetline ne {}} {
7304 if {$diffline == $targetline} {
7305 set seehere [$ctext index "end - 1 chars"]
7306 set targetline {}
7307 } else {
7308 incr diffline
7309 }
7310 }
7311 $ctext insert end "$line\n" $tags
7312 }
7313 } else {
7314 # "\ No newline at end of file",
7315 # or something else we don't recognize
7316 $ctext insert end "$line\n" hunksep
7317 }
7318 }
7319 }
7320 if {[info exists seehere]} {
7321 mark_ctext_line [lindex [split $seehere .] 0]
7322 }
7323 $ctext conf -state disabled
7324 if {[eof $bdf]} {
7325 close $bdf
7326 return 0
7327 }
7328 return [expr {$nr >= 1000? 2: 1}]
7329}
7330
7331proc changediffdisp {} {
7332 global ctext diffelide
7333
7334 $ctext tag conf d0 -elide [lindex $diffelide 0]
7335 $ctext tag conf dresult -elide [lindex $diffelide 1]
7336}
7337
7338proc highlightfile {loc cline} {
7339 global ctext cflist cflist_top
7340
7341 $ctext yview $loc
7342 $cflist tag remove highlight $cflist_top.0 "$cflist_top.0 lineend"
7343 $cflist tag add highlight $cline.0 "$cline.0 lineend"
7344 $cflist see $cline.0
7345 set cflist_top $cline
7346}
7347
7348proc prevfile {} {
7349 global difffilestart ctext cmitmode
7350
7351 if {$cmitmode eq "tree"} return
7352 set prev 0.0
7353 set prevline 1
7354 set here [$ctext index @0,0]
7355 foreach loc $difffilestart {
7356 if {[$ctext compare $loc >= $here]} {
7357 highlightfile $prev $prevline
7358 return
7359 }
7360 set prev $loc
7361 incr prevline
7362 }
7363 highlightfile $prev $prevline
7364}
7365
7366proc nextfile {} {
7367 global difffilestart ctext cmitmode
7368
7369 if {$cmitmode eq "tree"} return
7370 set here [$ctext index @0,0]
7371 set line 1
7372 foreach loc $difffilestart {
7373 incr line
7374 if {[$ctext compare $loc > $here]} {
7375 highlightfile $loc $line
7376 return
7377 }
7378 }
7379}
7380
7381proc clear_ctext {{first 1.0}} {
7382 global ctext smarktop smarkbot
7383 global ctext_file_names ctext_file_lines
7384 global pendinglinks
7385
7386 set l [lindex [split $first .] 0]
7387 if {![info exists smarktop] || [$ctext compare $first < $smarktop.0]} {
7388 set smarktop $l
7389 }
7390 if {![info exists smarkbot] || [$ctext compare $first < $smarkbot.0]} {
7391 set smarkbot $l
7392 }
7393 $ctext delete $first end
7394 if {$first eq "1.0"} {
7395 catch {unset pendinglinks}
7396 }
7397 set ctext_file_names {}
7398 set ctext_file_lines {}
7399}
7400
7401proc settabs {{firstab {}}} {
7402 global firsttabstop tabstop ctext have_tk85
7403
7404 if {$firstab ne {} && $have_tk85} {
7405 set firsttabstop $firstab
7406 }
7407 set w [font measure textfont "0"]
7408 if {$firsttabstop != 0} {
7409 $ctext conf -tabs [list [expr {($firsttabstop + $tabstop) * $w}] \
7410 [expr {($firsttabstop + 2 * $tabstop) * $w}]]
7411 } elseif {$have_tk85 || $tabstop != 8} {
7412 $ctext conf -tabs [expr {$tabstop * $w}]
7413 } else {
7414 $ctext conf -tabs {}
7415 }
7416}
7417
7418proc incrsearch {name ix op} {
7419 global ctext searchstring searchdirn
7420
7421 $ctext tag remove found 1.0 end
7422 if {[catch {$ctext index anchor}]} {
7423 # no anchor set, use start of selection, or of visible area
7424 set sel [$ctext tag ranges sel]
7425 if {$sel ne {}} {
7426 $ctext mark set anchor [lindex $sel 0]
7427 } elseif {$searchdirn eq "-forwards"} {
7428 $ctext mark set anchor @0,0
7429 } else {
7430 $ctext mark set anchor @0,[winfo height $ctext]
7431 }
7432 }
7433 if {$searchstring ne {}} {
7434 set here [$ctext search $searchdirn -- $searchstring anchor]
7435 if {$here ne {}} {
7436 $ctext see $here
7437 }
7438 searchmarkvisible 1
7439 }
7440}
7441
7442proc dosearch {} {
7443 global sstring ctext searchstring searchdirn
7444
7445 focus $sstring
7446 $sstring icursor end
7447 set searchdirn -forwards
7448 if {$searchstring ne {}} {
7449 set sel [$ctext tag ranges sel]
7450 if {$sel ne {}} {
7451 set start "[lindex $sel 0] + 1c"
7452 } elseif {[catch {set start [$ctext index anchor]}]} {
7453 set start "@0,0"
7454 }
7455 set match [$ctext search -count mlen -- $searchstring $start]
7456 $ctext tag remove sel 1.0 end
7457 if {$match eq {}} {
7458 bell
7459 return
7460 }
7461 $ctext see $match
7462 set mend "$match + $mlen c"
7463 $ctext tag add sel $match $mend
7464 $ctext mark unset anchor
7465 }
7466}
7467
7468proc dosearchback {} {
7469 global sstring ctext searchstring searchdirn
7470
7471 focus $sstring
7472 $sstring icursor end
7473 set searchdirn -backwards
7474 if {$searchstring ne {}} {
7475 set sel [$ctext tag ranges sel]
7476 if {$sel ne {}} {
7477 set start [lindex $sel 0]
7478 } elseif {[catch {set start [$ctext index anchor]}]} {
7479 set start @0,[winfo height $ctext]
7480 }
7481 set match [$ctext search -backwards -count ml -- $searchstring $start]
7482 $ctext tag remove sel 1.0 end
7483 if {$match eq {}} {
7484 bell
7485 return
7486 }
7487 $ctext see $match
7488 set mend "$match + $ml c"
7489 $ctext tag add sel $match $mend
7490 $ctext mark unset anchor
7491 }
7492}
7493
7494proc searchmark {first last} {
7495 global ctext searchstring
7496
7497 set mend $first.0
7498 while {1} {
7499 set match [$ctext search -count mlen -- $searchstring $mend $last.end]
7500 if {$match eq {}} break
7501 set mend "$match + $mlen c"
7502 $ctext tag add found $match $mend
7503 }
7504}
7505
7506proc searchmarkvisible {doall} {
7507 global ctext smarktop smarkbot
7508
7509 set topline [lindex [split [$ctext index @0,0] .] 0]
7510 set botline [lindex [split [$ctext index @0,[winfo height $ctext]] .] 0]
7511 if {$doall || $botline < $smarktop || $topline > $smarkbot} {
7512 # no overlap with previous
7513 searchmark $topline $botline
7514 set smarktop $topline
7515 set smarkbot $botline
7516 } else {
7517 if {$topline < $smarktop} {
7518 searchmark $topline [expr {$smarktop-1}]
7519 set smarktop $topline
7520 }
7521 if {$botline > $smarkbot} {
7522 searchmark [expr {$smarkbot+1}] $botline
7523 set smarkbot $botline
7524 }
7525 }
7526}
7527
7528proc scrolltext {f0 f1} {
7529 global searchstring
7530
7531 .bleft.bottom.sb set $f0 $f1
7532 if {$searchstring ne {}} {
7533 searchmarkvisible 0
7534 }
7535}
7536
7537proc setcoords {} {
7538 global linespc charspc canvx0 canvy0
7539 global xspc1 xspc2 lthickness
7540
7541 set linespc [font metrics mainfont -linespace]
7542 set charspc [font measure mainfont "m"]
7543 set canvy0 [expr {int(3 + 0.5 * $linespc)}]
7544 set canvx0 [expr {int(3 + 0.5 * $linespc)}]
7545 set lthickness [expr {int($linespc / 9) + 1}]
7546 set xspc1(0) $linespc
7547 set xspc2 $linespc
7548}
7549
7550proc redisplay {} {
7551 global canv
7552 global selectedline
7553
7554 set ymax [lindex [$canv cget -scrollregion] 3]
7555 if {$ymax eq {} || $ymax == 0} return
7556 set span [$canv yview]
7557 clear_display
7558 setcanvscroll
7559 allcanvs yview moveto [lindex $span 0]
7560 drawvisible
7561 if {$selectedline ne {}} {
7562 selectline $selectedline 0
7563 allcanvs yview moveto [lindex $span 0]
7564 }
7565}
7566
7567proc parsefont {f n} {
7568 global fontattr
7569
7570 set fontattr($f,family) [lindex $n 0]
7571 set s [lindex $n 1]
7572 if {$s eq {} || $s == 0} {
7573 set s 10
7574 } elseif {$s < 0} {
7575 set s [expr {int(-$s / [winfo fpixels . 1p] + 0.5)}]
7576 }
7577 set fontattr($f,size) $s
7578 set fontattr($f,weight) normal
7579 set fontattr($f,slant) roman
7580 foreach style [lrange $n 2 end] {
7581 switch -- $style {
7582 "normal" -
7583 "bold" {set fontattr($f,weight) $style}
7584 "roman" -
7585 "italic" {set fontattr($f,slant) $style}
7586 }
7587 }
7588}
7589
7590proc fontflags {f {isbold 0}} {
7591 global fontattr
7592
7593 return [list -family $fontattr($f,family) -size $fontattr($f,size) \
7594 -weight [expr {$isbold? "bold": $fontattr($f,weight)}] \
7595 -slant $fontattr($f,slant)]
7596}
7597
7598proc fontname {f} {
7599 global fontattr
7600
7601 set n [list $fontattr($f,family) $fontattr($f,size)]
7602 if {$fontattr($f,weight) eq "bold"} {
7603 lappend n "bold"
7604 }
7605 if {$fontattr($f,slant) eq "italic"} {
7606 lappend n "italic"
7607 }
7608 return $n
7609}
7610
7611proc incrfont {inc} {
7612 global mainfont textfont ctext canv cflist showrefstop
7613 global stopped entries fontattr
7614
7615 unmarkmatches
7616 set s $fontattr(mainfont,size)
7617 incr s $inc
7618 if {$s < 1} {
7619 set s 1
7620 }
7621 set fontattr(mainfont,size) $s
7622 font config mainfont -size $s
7623 font config mainfontbold -size $s
7624 set mainfont [fontname mainfont]
7625 set s $fontattr(textfont,size)
7626 incr s $inc
7627 if {$s < 1} {
7628 set s 1
7629 }
7630 set fontattr(textfont,size) $s
7631 font config textfont -size $s
7632 font config textfontbold -size $s
7633 set textfont [fontname textfont]
7634 setcoords
7635 settabs
7636 redisplay
7637}
7638
7639proc clearsha1 {} {
7640 global sha1entry sha1string
7641 if {[string length $sha1string] == 40} {
7642 $sha1entry delete 0 end
7643 }
7644}
7645
7646proc sha1change {n1 n2 op} {
7647 global sha1string currentid sha1but
7648 if {$sha1string == {}
7649 || ([info exists currentid] && $sha1string == $currentid)} {
7650 set state disabled
7651 } else {
7652 set state normal
7653 }
7654 if {[$sha1but cget -state] == $state} return
7655 if {$state == "normal"} {
7656 $sha1but conf -state normal -relief raised -text "[mc "Goto:"] "
7657 } else {
7658 $sha1but conf -state disabled -relief flat -text "[mc "SHA1 ID:"] "
7659 }
7660}
7661
7662proc gotocommit {} {
7663 global sha1string tagids headids curview varcid
7664
7665 if {$sha1string == {}
7666 || ([info exists currentid] && $sha1string == $currentid)} return
7667 if {[info exists tagids($sha1string)]} {
7668 set id $tagids($sha1string)
7669 } elseif {[info exists headids($sha1string)]} {
7670 set id $headids($sha1string)
7671 } else {
7672 set id [string tolower $sha1string]
7673 if {[regexp {^[0-9a-f]{4,39}$} $id]} {
7674 set matches [longid $id]
7675 if {$matches ne {}} {
7676 if {[llength $matches] > 1} {
7677 error_popup [mc "Short SHA1 id %s is ambiguous" $id]
7678 return
7679 }
7680 set id [lindex $matches 0]
7681 }
7682 }
7683 }
7684 if {[commitinview $id $curview]} {
7685 selectline [rowofcommit $id] 1
7686 return
7687 }
7688 if {[regexp {^[0-9a-fA-F]{4,}$} $sha1string]} {
7689 set msg [mc "SHA1 id %s is not known" $sha1string]
7690 } else {
7691 set msg [mc "Tag/Head %s is not known" $sha1string]
7692 }
7693 error_popup $msg
7694}
7695
7696proc lineenter {x y id} {
7697 global hoverx hovery hoverid hovertimer
7698 global commitinfo canv
7699
7700 if {![info exists commitinfo($id)] && ![getcommit $id]} return
7701 set hoverx $x
7702 set hovery $y
7703 set hoverid $id
7704 if {[info exists hovertimer]} {
7705 after cancel $hovertimer
7706 }
7707 set hovertimer [after 500 linehover]
7708 $canv delete hover
7709}
7710
7711proc linemotion {x y id} {
7712 global hoverx hovery hoverid hovertimer
7713
7714 if {[info exists hoverid] && $id == $hoverid} {
7715 set hoverx $x
7716 set hovery $y
7717 if {[info exists hovertimer]} {
7718 after cancel $hovertimer
7719 }
7720 set hovertimer [after 500 linehover]
7721 }
7722}
7723
7724proc lineleave {id} {
7725 global hoverid hovertimer canv
7726
7727 if {[info exists hoverid] && $id == $hoverid} {
7728 $canv delete hover
7729 if {[info exists hovertimer]} {
7730 after cancel $hovertimer
7731 unset hovertimer
7732 }
7733 unset hoverid
7734 }
7735}
7736
7737proc linehover {} {
7738 global hoverx hovery hoverid hovertimer
7739 global canv linespc lthickness
7740 global commitinfo
7741
7742 set text [lindex $commitinfo($hoverid) 0]
7743 set ymax [lindex [$canv cget -scrollregion] 3]
7744 if {$ymax == {}} return
7745 set yfrac [lindex [$canv yview] 0]
7746 set x [expr {$hoverx + 2 * $linespc}]
7747 set y [expr {$hovery + $yfrac * $ymax - $linespc / 2}]
7748 set x0 [expr {$x - 2 * $lthickness}]
7749 set y0 [expr {$y - 2 * $lthickness}]
7750 set x1 [expr {$x + [font measure mainfont $text] + 2 * $lthickness}]
7751 set y1 [expr {$y + $linespc + 2 * $lthickness}]
7752 set t [$canv create rectangle $x0 $y0 $x1 $y1 \
7753 -fill \#ffff80 -outline black -width 1 -tags hover]
7754 $canv raise $t
7755 set t [$canv create text $x $y -anchor nw -text $text -tags hover \
7756 -font mainfont]
7757 $canv raise $t
7758}
7759
7760proc clickisonarrow {id y} {
7761 global lthickness
7762
7763 set ranges [rowranges $id]
7764 set thresh [expr {2 * $lthickness + 6}]
7765 set n [expr {[llength $ranges] - 1}]
7766 for {set i 1} {$i < $n} {incr i} {
7767 set row [lindex $ranges $i]
7768 if {abs([yc $row] - $y) < $thresh} {
7769 return $i
7770 }
7771 }
7772 return {}
7773}
7774
7775proc arrowjump {id n y} {
7776 global canv
7777
7778 # 1 <-> 2, 3 <-> 4, etc...
7779 set n [expr {(($n - 1) ^ 1) + 1}]
7780 set row [lindex [rowranges $id] $n]
7781 set yt [yc $row]
7782 set ymax [lindex [$canv cget -scrollregion] 3]
7783 if {$ymax eq {} || $ymax <= 0} return
7784 set view [$canv yview]
7785 set yspan [expr {[lindex $view 1] - [lindex $view 0]}]
7786 set yfrac [expr {$yt / $ymax - $yspan / 2}]
7787 if {$yfrac < 0} {
7788 set yfrac 0
7789 }
7790 allcanvs yview moveto $yfrac
7791}
7792
7793proc lineclick {x y id isnew} {
7794 global ctext commitinfo children canv thickerline curview
7795
7796 if {![info exists commitinfo($id)] && ![getcommit $id]} return
7797 unmarkmatches
7798 unselectline
7799 normalline
7800 $canv delete hover
7801 # draw this line thicker than normal
7802 set thickerline $id
7803 drawlines $id
7804 if {$isnew} {
7805 set ymax [lindex [$canv cget -scrollregion] 3]
7806 if {$ymax eq {}} return
7807 set yfrac [lindex [$canv yview] 0]
7808 set y [expr {$y + $yfrac * $ymax}]
7809 }
7810 set dirn [clickisonarrow $id $y]
7811 if {$dirn ne {}} {
7812 arrowjump $id $dirn $y
7813 return
7814 }
7815
7816 if {$isnew} {
7817 addtohistory [list lineclick $x $y $id 0]
7818 }
7819 # fill the details pane with info about this line
7820 $ctext conf -state normal
7821 clear_ctext
7822 settabs 0
7823 $ctext insert end "[mc "Parent"]:\t"
7824 $ctext insert end $id link0
7825 setlink $id link0
7826 set info $commitinfo($id)
7827 $ctext insert end "\n\t[lindex $info 0]\n"
7828 $ctext insert end "\t[mc "Author"]:\t[lindex $info 1]\n"
7829 set date [formatdate [lindex $info 2]]
7830 $ctext insert end "\t[mc "Date"]:\t$date\n"
7831 set kids $children($curview,$id)
7832 if {$kids ne {}} {
7833 $ctext insert end "\n[mc "Children"]:"
7834 set i 0
7835 foreach child $kids {
7836 incr i
7837 if {![info exists commitinfo($child)] && ![getcommit $child]} continue
7838 set info $commitinfo($child)
7839 $ctext insert end "\n\t"
7840 $ctext insert end $child link$i
7841 setlink $child link$i
7842 $ctext insert end "\n\t[lindex $info 0]"
7843 $ctext insert end "\n\t[mc "Author"]:\t[lindex $info 1]"
7844 set date [formatdate [lindex $info 2]]
7845 $ctext insert end "\n\t[mc "Date"]:\t$date\n"
7846 }
7847 }
7848 $ctext conf -state disabled
7849 init_flist {}
7850}
7851
7852proc normalline {} {
7853 global thickerline
7854 if {[info exists thickerline]} {
7855 set id $thickerline
7856 unset thickerline
7857 drawlines $id
7858 }
7859}
7860
7861proc selbyid {id} {
7862 global curview
7863 if {[commitinview $id $curview]} {
7864 selectline [rowofcommit $id] 1
7865 }
7866}
7867
7868proc mstime {} {
7869 global startmstime
7870 if {![info exists startmstime]} {
7871 set startmstime [clock clicks -milliseconds]
7872 }
7873 return [format "%.3f" [expr {([clock click -milliseconds] - $startmstime) / 1000.0}]]
7874}
7875
7876proc rowmenu {x y id} {
7877 global rowctxmenu selectedline rowmenuid curview
7878 global nullid nullid2 fakerowmenu mainhead
7879
7880 stopfinding
7881 set rowmenuid $id
7882 if {$selectedline eq {} || [rowofcommit $id] eq $selectedline} {
7883 set state disabled
7884 } else {
7885 set state normal
7886 }
7887 if {$id ne $nullid && $id ne $nullid2} {
7888 set menu $rowctxmenu
7889 if {$mainhead ne {}} {
7890 $menu entryconfigure 7 -label [mc "Reset %s branch to here" $mainhead]
7891 } else {
7892 $menu entryconfigure 7 -label [mc "Detached head: can't reset" $mainhead] -state disabled
7893 }
7894 } else {
7895 set menu $fakerowmenu
7896 }
7897 $menu entryconfigure [mca "Diff this -> selected"] -state $state
7898 $menu entryconfigure [mca "Diff selected -> this"] -state $state
7899 $menu entryconfigure [mca "Make patch"] -state $state
7900 tk_popup $menu $x $y
7901}
7902
7903proc diffvssel {dirn} {
7904 global rowmenuid selectedline
7905
7906 if {$selectedline eq {}} return
7907 if {$dirn} {
7908 set oldid [commitonrow $selectedline]
7909 set newid $rowmenuid
7910 } else {
7911 set oldid $rowmenuid
7912 set newid [commitonrow $selectedline]
7913 }
7914 addtohistory [list doseldiff $oldid $newid]
7915 doseldiff $oldid $newid
7916}
7917
7918proc doseldiff {oldid newid} {
7919 global ctext
7920 global commitinfo
7921
7922 $ctext conf -state normal
7923 clear_ctext
7924 init_flist [mc "Top"]
7925 $ctext insert end "[mc "From"] "
7926 $ctext insert end $oldid link0
7927 setlink $oldid link0
7928 $ctext insert end "\n "
7929 $ctext insert end [lindex $commitinfo($oldid) 0]
7930 $ctext insert end "\n\n[mc "To"] "
7931 $ctext insert end $newid link1
7932 setlink $newid link1
7933 $ctext insert end "\n "
7934 $ctext insert end [lindex $commitinfo($newid) 0]
7935 $ctext insert end "\n"
7936 $ctext conf -state disabled
7937 $ctext tag remove found 1.0 end
7938 startdiff [list $oldid $newid]
7939}
7940
7941proc mkpatch {} {
7942 global rowmenuid currentid commitinfo patchtop patchnum
7943
7944 if {![info exists currentid]} return
7945 set oldid $currentid
7946 set oldhead [lindex $commitinfo($oldid) 0]
7947 set newid $rowmenuid
7948 set newhead [lindex $commitinfo($newid) 0]
7949 set top .patch
7950 set patchtop $top
7951 catch {destroy $top}
7952 toplevel $top
7953 make_transient $top .
7954 label $top.title -text [mc "Generate patch"]
7955 grid $top.title - -pady 10
7956 label $top.from -text [mc "From:"]
7957 entry $top.fromsha1 -width 40 -relief flat
7958 $top.fromsha1 insert 0 $oldid
7959 $top.fromsha1 conf -state readonly
7960 grid $top.from $top.fromsha1 -sticky w
7961 entry $top.fromhead -width 60 -relief flat
7962 $top.fromhead insert 0 $oldhead
7963 $top.fromhead conf -state readonly
7964 grid x $top.fromhead -sticky w
7965 label $top.to -text [mc "To:"]
7966 entry $top.tosha1 -width 40 -relief flat
7967 $top.tosha1 insert 0 $newid
7968 $top.tosha1 conf -state readonly
7969 grid $top.to $top.tosha1 -sticky w
7970 entry $top.tohead -width 60 -relief flat
7971 $top.tohead insert 0 $newhead
7972 $top.tohead conf -state readonly
7973 grid x $top.tohead -sticky w
7974 button $top.rev -text [mc "Reverse"] -command mkpatchrev -padx 5
7975 grid $top.rev x -pady 10
7976 label $top.flab -text [mc "Output file:"]
7977 entry $top.fname -width 60
7978 $top.fname insert 0 [file normalize "patch$patchnum.patch"]
7979 incr patchnum
7980 grid $top.flab $top.fname -sticky w
7981 frame $top.buts
7982 button $top.buts.gen -text [mc "Generate"] -command mkpatchgo
7983 button $top.buts.can -text [mc "Cancel"] -command mkpatchcan
7984 bind $top <Key-Return> mkpatchgo
7985 bind $top <Key-Escape> mkpatchcan
7986 grid $top.buts.gen $top.buts.can
7987 grid columnconfigure $top.buts 0 -weight 1 -uniform a
7988 grid columnconfigure $top.buts 1 -weight 1 -uniform a
7989 grid $top.buts - -pady 10 -sticky ew
7990 focus $top.fname
7991}
7992
7993proc mkpatchrev {} {
7994 global patchtop
7995
7996 set oldid [$patchtop.fromsha1 get]
7997 set oldhead [$patchtop.fromhead get]
7998 set newid [$patchtop.tosha1 get]
7999 set newhead [$patchtop.tohead get]
8000 foreach e [list fromsha1 fromhead tosha1 tohead] \
8001 v [list $newid $newhead $oldid $oldhead] {
8002 $patchtop.$e conf -state normal
8003 $patchtop.$e delete 0 end
8004 $patchtop.$e insert 0 $v
8005 $patchtop.$e conf -state readonly
8006 }
8007}
8008
8009proc mkpatchgo {} {
8010 global patchtop nullid nullid2
8011
8012 set oldid [$patchtop.fromsha1 get]
8013 set newid [$patchtop.tosha1 get]
8014 set fname [$patchtop.fname get]
8015 set cmd [diffcmd [list $oldid $newid] -p]
8016 # trim off the initial "|"
8017 set cmd [lrange $cmd 1 end]
8018 lappend cmd >$fname &
8019 if {[catch {eval exec $cmd} err]} {
8020 error_popup "[mc "Error creating patch:"] $err" $patchtop
8021 }
8022 catch {destroy $patchtop}
8023 unset patchtop
8024}
8025
8026proc mkpatchcan {} {
8027 global patchtop
8028
8029 catch {destroy $patchtop}
8030 unset patchtop
8031}
8032
8033proc mktag {} {
8034 global rowmenuid mktagtop commitinfo
8035
8036 set top .maketag
8037 set mktagtop $top
8038 catch {destroy $top}
8039 toplevel $top
8040 make_transient $top .
8041 label $top.title -text [mc "Create tag"]
8042 grid $top.title - -pady 10
8043 label $top.id -text [mc "ID:"]
8044 entry $top.sha1 -width 40 -relief flat
8045 $top.sha1 insert 0 $rowmenuid
8046 $top.sha1 conf -state readonly
8047 grid $top.id $top.sha1 -sticky w
8048 entry $top.head -width 60 -relief flat
8049 $top.head insert 0 [lindex $commitinfo($rowmenuid) 0]
8050 $top.head conf -state readonly
8051 grid x $top.head -sticky w
8052 label $top.tlab -text [mc "Tag name:"]
8053 entry $top.tag -width 60
8054 grid $top.tlab $top.tag -sticky w
8055 frame $top.buts
8056 button $top.buts.gen -text [mc "Create"] -command mktaggo
8057 button $top.buts.can -text [mc "Cancel"] -command mktagcan
8058 bind $top <Key-Return> mktaggo
8059 bind $top <Key-Escape> mktagcan
8060 grid $top.buts.gen $top.buts.can
8061 grid columnconfigure $top.buts 0 -weight 1 -uniform a
8062 grid columnconfigure $top.buts 1 -weight 1 -uniform a
8063 grid $top.buts - -pady 10 -sticky ew
8064 focus $top.tag
8065}
8066
8067proc domktag {} {
8068 global mktagtop env tagids idtags
8069
8070 set id [$mktagtop.sha1 get]
8071 set tag [$mktagtop.tag get]
8072 if {$tag == {}} {
8073 error_popup [mc "No tag name specified"] $mktagtop
8074 return 0
8075 }
8076 if {[info exists tagids($tag)]} {
8077 error_popup [mc "Tag \"%s\" already exists" $tag] $mktagtop
8078 return 0
8079 }
8080 if {[catch {
8081 exec git tag $tag $id
8082 } err]} {
8083 error_popup "[mc "Error creating tag:"] $err" $mktagtop
8084 return 0
8085 }
8086
8087 set tagids($tag) $id
8088 lappend idtags($id) $tag
8089 redrawtags $id
8090 addedtag $id
8091 dispneartags 0
8092 run refill_reflist
8093 return 1
8094}
8095
8096proc redrawtags {id} {
8097 global canv linehtag idpos currentid curview cmitlisted
8098 global canvxmax iddrawn circleitem mainheadid circlecolors
8099
8100 if {![commitinview $id $curview]} return
8101 if {![info exists iddrawn($id)]} return
8102 set row [rowofcommit $id]
8103 if {$id eq $mainheadid} {
8104 set ofill yellow
8105 } else {
8106 set ofill [lindex $circlecolors $cmitlisted($curview,$id)]
8107 }
8108 $canv itemconf $circleitem($row) -fill $ofill
8109 $canv delete tag.$id
8110 set xt [eval drawtags $id $idpos($id)]
8111 $canv coords $linehtag($id) $xt [lindex $idpos($id) 2]
8112 set text [$canv itemcget $linehtag($id) -text]
8113 set font [$canv itemcget $linehtag($id) -font]
8114 set xr [expr {$xt + [font measure $font $text]}]
8115 if {$xr > $canvxmax} {
8116 set canvxmax $xr
8117 setcanvscroll
8118 }
8119 if {[info exists currentid] && $currentid == $id} {
8120 make_secsel $id
8121 }
8122}
8123
8124proc mktagcan {} {
8125 global mktagtop
8126
8127 catch {destroy $mktagtop}
8128 unset mktagtop
8129}
8130
8131proc mktaggo {} {
8132 if {![domktag]} return
8133 mktagcan
8134}
8135
8136proc writecommit {} {
8137 global rowmenuid wrcomtop commitinfo wrcomcmd
8138
8139 set top .writecommit
8140 set wrcomtop $top
8141 catch {destroy $top}
8142 toplevel $top
8143 make_transient $top .
8144 label $top.title -text [mc "Write commit to file"]
8145 grid $top.title - -pady 10
8146 label $top.id -text [mc "ID:"]
8147 entry $top.sha1 -width 40 -relief flat
8148 $top.sha1 insert 0 $rowmenuid
8149 $top.sha1 conf -state readonly
8150 grid $top.id $top.sha1 -sticky w
8151 entry $top.head -width 60 -relief flat
8152 $top.head insert 0 [lindex $commitinfo($rowmenuid) 0]
8153 $top.head conf -state readonly
8154 grid x $top.head -sticky w
8155 label $top.clab -text [mc "Command:"]
8156 entry $top.cmd -width 60 -textvariable wrcomcmd
8157 grid $top.clab $top.cmd -sticky w -pady 10
8158 label $top.flab -text [mc "Output file:"]
8159 entry $top.fname -width 60
8160 $top.fname insert 0 [file normalize "commit-[string range $rowmenuid 0 6]"]
8161 grid $top.flab $top.fname -sticky w
8162 frame $top.buts
8163 button $top.buts.gen -text [mc "Write"] -command wrcomgo
8164 button $top.buts.can -text [mc "Cancel"] -command wrcomcan
8165 bind $top <Key-Return> wrcomgo
8166 bind $top <Key-Escape> wrcomcan
8167 grid $top.buts.gen $top.buts.can
8168 grid columnconfigure $top.buts 0 -weight 1 -uniform a
8169 grid columnconfigure $top.buts 1 -weight 1 -uniform a
8170 grid $top.buts - -pady 10 -sticky ew
8171 focus $top.fname
8172}
8173
8174proc wrcomgo {} {
8175 global wrcomtop
8176
8177 set id [$wrcomtop.sha1 get]
8178 set cmd "echo $id | [$wrcomtop.cmd get]"
8179 set fname [$wrcomtop.fname get]
8180 if {[catch {exec sh -c $cmd >$fname &} err]} {
8181 error_popup "[mc "Error writing commit:"] $err" $wrcomtop
8182 }
8183 catch {destroy $wrcomtop}
8184 unset wrcomtop
8185}
8186
8187proc wrcomcan {} {
8188 global wrcomtop
8189
8190 catch {destroy $wrcomtop}
8191 unset wrcomtop
8192}
8193
8194proc mkbranch {} {
8195 global rowmenuid mkbrtop
8196
8197 set top .makebranch
8198 catch {destroy $top}
8199 toplevel $top
8200 make_transient $top .
8201 label $top.title -text [mc "Create new branch"]
8202 grid $top.title - -pady 10
8203 label $top.id -text [mc "ID:"]
8204 entry $top.sha1 -width 40 -relief flat
8205 $top.sha1 insert 0 $rowmenuid
8206 $top.sha1 conf -state readonly
8207 grid $top.id $top.sha1 -sticky w
8208 label $top.nlab -text [mc "Name:"]
8209 entry $top.name -width 40
8210 bind $top.name <Key-Return> "[list mkbrgo $top]"
8211 grid $top.nlab $top.name -sticky w
8212 frame $top.buts
8213 button $top.buts.go -text [mc "Create"] -command [list mkbrgo $top]
8214 button $top.buts.can -text [mc "Cancel"] -command "catch {destroy $top}"
8215 bind $top <Key-Return> [list mkbrgo $top]
8216 bind $top <Key-Escape> "catch {destroy $top}"
8217 grid $top.buts.go $top.buts.can
8218 grid columnconfigure $top.buts 0 -weight 1 -uniform a
8219 grid columnconfigure $top.buts 1 -weight 1 -uniform a
8220 grid $top.buts - -pady 10 -sticky ew
8221 focus $top.name
8222}
8223
8224proc mkbrgo {top} {
8225 global headids idheads
8226
8227 set name [$top.name get]
8228 set id [$top.sha1 get]
8229 set cmdargs {}
8230 set old_id {}
8231 if {$name eq {}} {
8232 error_popup [mc "Please specify a name for the new branch"] $top
8233 return
8234 }
8235 if {[info exists headids($name)]} {
8236 if {![confirm_popup [mc \
8237 "Branch '%s' already exists. Overwrite?" $name] $top]} {
8238 return
8239 }
8240 set old_id $headids($name)
8241 lappend cmdargs -f
8242 }
8243 catch {destroy $top}
8244 lappend cmdargs $name $id
8245 nowbusy newbranch
8246 update
8247 if {[catch {
8248 eval exec git branch $cmdargs
8249 } err]} {
8250 notbusy newbranch
8251 error_popup $err
8252 } else {
8253 notbusy newbranch
8254 if {$old_id ne {}} {
8255 movehead $id $name
8256 movedhead $id $name
8257 redrawtags $old_id
8258 redrawtags $id
8259 } else {
8260 set headids($name) $id
8261 lappend idheads($id) $name
8262 addedhead $id $name
8263 redrawtags $id
8264 }
8265 dispneartags 0
8266 run refill_reflist
8267 }
8268}
8269
8270proc exec_citool {tool_args {baseid {}}} {
8271 global commitinfo env
8272
8273 set save_env [array get env GIT_AUTHOR_*]
8274
8275 if {$baseid ne {}} {
8276 if {![info exists commitinfo($baseid)]} {
8277 getcommit $baseid
8278 }
8279 set author [lindex $commitinfo($baseid) 1]
8280 set date [lindex $commitinfo($baseid) 2]
8281 if {[regexp {^\s*(\S.*\S|\S)\s*<(.*)>\s*$} \
8282 $author author name email]
8283 && $date ne {}} {
8284 set env(GIT_AUTHOR_NAME) $name
8285 set env(GIT_AUTHOR_EMAIL) $email
8286 set env(GIT_AUTHOR_DATE) $date
8287 }
8288 }
8289
8290 eval exec git citool $tool_args &
8291
8292 array unset env GIT_AUTHOR_*
8293 array set env $save_env
8294}
8295
8296proc cherrypick {} {
8297 global rowmenuid curview
8298 global mainhead mainheadid
8299
8300 set oldhead [exec git rev-parse HEAD]
8301 set dheads [descheads $rowmenuid]
8302 if {$dheads ne {} && [lsearch -exact $dheads $oldhead] >= 0} {
8303 set ok [confirm_popup [mc "Commit %s is already\
8304 included in branch %s -- really re-apply it?" \
8305 [string range $rowmenuid 0 7] $mainhead]]
8306 if {!$ok} return
8307 }
8308 nowbusy cherrypick [mc "Cherry-picking"]
8309 update
8310 # Unfortunately git-cherry-pick writes stuff to stderr even when
8311 # no error occurs, and exec takes that as an indication of error...
8312 if {[catch {exec sh -c "git cherry-pick -r $rowmenuid 2>&1"} err]} {
8313 notbusy cherrypick
8314 if {[regexp -line \
8315 {Entry '(.*)' (would be overwritten by merge|not uptodate)} \
8316 $err msg fname]} {
8317 error_popup [mc "Cherry-pick failed because of local changes\
8318 to file '%s'.\nPlease commit, reset or stash\
8319 your changes and try again." $fname]
8320 } elseif {[regexp -line \
8321 {^(CONFLICT \(.*\):|Automatic cherry-pick failed)} \
8322 $err]} {
8323 if {[confirm_popup [mc "Cherry-pick failed because of merge\
8324 conflict.\nDo you wish to run git citool to\
8325 resolve it?"]]} {
8326 # Force citool to read MERGE_MSG
8327 file delete [file join [gitdir] "GITGUI_MSG"]
8328 exec_citool {} $rowmenuid
8329 }
8330 } else {
8331 error_popup $err
8332 }
8333 run updatecommits
8334 return
8335 }
8336 set newhead [exec git rev-parse HEAD]
8337 if {$newhead eq $oldhead} {
8338 notbusy cherrypick
8339 error_popup [mc "No changes committed"]
8340 return
8341 }
8342 addnewchild $newhead $oldhead
8343 if {[commitinview $oldhead $curview]} {
8344 insertrow $newhead $oldhead $curview
8345 if {$mainhead ne {}} {
8346 movehead $newhead $mainhead
8347 movedhead $newhead $mainhead
8348 }
8349 set mainheadid $newhead
8350 redrawtags $oldhead
8351 redrawtags $newhead
8352 selbyid $newhead
8353 }
8354 notbusy cherrypick
8355}
8356
8357proc resethead {} {
8358 global mainhead rowmenuid confirm_ok resettype
8359
8360 set confirm_ok 0
8361 set w ".confirmreset"
8362 toplevel $w
8363 make_transient $w .
8364 wm title $w [mc "Confirm reset"]
8365 message $w.m -text \
8366 [mc "Reset branch %s to %s?" $mainhead [string range $rowmenuid 0 7]] \
8367 -justify center -aspect 1000
8368 pack $w.m -side top -fill x -padx 20 -pady 20
8369 frame $w.f -relief sunken -border 2
8370 message $w.f.rt -text [mc "Reset type:"] -aspect 1000
8371 grid $w.f.rt -sticky w
8372 set resettype mixed
8373 radiobutton $w.f.soft -value soft -variable resettype -justify left \
8374 -text [mc "Soft: Leave working tree and index untouched"]
8375 grid $w.f.soft -sticky w
8376 radiobutton $w.f.mixed -value mixed -variable resettype -justify left \
8377 -text [mc "Mixed: Leave working tree untouched, reset index"]
8378 grid $w.f.mixed -sticky w
8379 radiobutton $w.f.hard -value hard -variable resettype -justify left \
8380 -text [mc "Hard: Reset working tree and index\n(discard ALL local changes)"]
8381 grid $w.f.hard -sticky w
8382 pack $w.f -side top -fill x
8383 button $w.ok -text [mc OK] -command "set confirm_ok 1; destroy $w"
8384 pack $w.ok -side left -fill x -padx 20 -pady 20
8385 button $w.cancel -text [mc Cancel] -command "destroy $w"
8386 bind $w <Key-Escape> [list destroy $w]
8387 pack $w.cancel -side right -fill x -padx 20 -pady 20
8388 bind $w <Visibility> "grab $w; focus $w"
8389 tkwait window $w
8390 if {!$confirm_ok} return
8391 if {[catch {set fd [open \
8392 [list | git reset --$resettype $rowmenuid 2>@1] r]} err]} {
8393 error_popup $err
8394 } else {
8395 dohidelocalchanges
8396 filerun $fd [list readresetstat $fd]
8397 nowbusy reset [mc "Resetting"]
8398 selbyid $rowmenuid
8399 }
8400}
8401
8402proc readresetstat {fd} {
8403 global mainhead mainheadid showlocalchanges rprogcoord
8404
8405 if {[gets $fd line] >= 0} {
8406 if {[regexp {([0-9]+)% \(([0-9]+)/([0-9]+)\)} $line match p m n]} {
8407 set rprogcoord [expr {1.0 * $m / $n}]
8408 adjustprogress
8409 }
8410 return 1
8411 }
8412 set rprogcoord 0
8413 adjustprogress
8414 notbusy reset
8415 if {[catch {close $fd} err]} {
8416 error_popup $err
8417 }
8418 set oldhead $mainheadid
8419 set newhead [exec git rev-parse HEAD]
8420 if {$newhead ne $oldhead} {
8421 movehead $newhead $mainhead
8422 movedhead $newhead $mainhead
8423 set mainheadid $newhead
8424 redrawtags $oldhead
8425 redrawtags $newhead
8426 }
8427 if {$showlocalchanges} {
8428 doshowlocalchanges
8429 }
8430 return 0
8431}
8432
8433# context menu for a head
8434proc headmenu {x y id head} {
8435 global headmenuid headmenuhead headctxmenu mainhead
8436
8437 stopfinding
8438 set headmenuid $id
8439 set headmenuhead $head
8440 set state normal
8441 if {$head eq $mainhead} {
8442 set state disabled
8443 }
8444 $headctxmenu entryconfigure 0 -state $state
8445 $headctxmenu entryconfigure 1 -state $state
8446 tk_popup $headctxmenu $x $y
8447}
8448
8449proc cobranch {} {
8450 global headmenuid headmenuhead headids
8451 global showlocalchanges mainheadid
8452
8453 # check the tree is clean first??
8454 nowbusy checkout [mc "Checking out"]
8455 update
8456 dohidelocalchanges
8457 if {[catch {
8458 set fd [open [list | git checkout $headmenuhead 2>@1] r]
8459 } err]} {
8460 notbusy checkout
8461 error_popup $err
8462 if {$showlocalchanges} {
8463 dodiffindex
8464 }
8465 } else {
8466 filerun $fd [list readcheckoutstat $fd $headmenuhead $headmenuid]
8467 }
8468}
8469
8470proc readcheckoutstat {fd newhead newheadid} {
8471 global mainhead mainheadid headids showlocalchanges progresscoords
8472
8473 if {[gets $fd line] >= 0} {
8474 if {[regexp {([0-9]+)% \(([0-9]+)/([0-9]+)\)} $line match p m n]} {
8475 set progresscoords [list 0 [expr {1.0 * $m / $n}]]
8476 adjustprogress
8477 }
8478 return 1
8479 }
8480 set progresscoords {0 0}
8481 adjustprogress
8482 notbusy checkout
8483 if {[catch {close $fd} err]} {
8484 error_popup $err
8485 }
8486 set oldmainid $mainheadid
8487 set mainhead $newhead
8488 set mainheadid $newheadid
8489 redrawtags $oldmainid
8490 redrawtags $newheadid
8491 selbyid $newheadid
8492 if {$showlocalchanges} {
8493 dodiffindex
8494 }
8495}
8496
8497proc rmbranch {} {
8498 global headmenuid headmenuhead mainhead
8499 global idheads
8500
8501 set head $headmenuhead
8502 set id $headmenuid
8503 # this check shouldn't be needed any more...
8504 if {$head eq $mainhead} {
8505 error_popup [mc "Cannot delete the currently checked-out branch"]
8506 return
8507 }
8508 set dheads [descheads $id]
8509 if {[llength $dheads] == 1 && $idheads($dheads) eq $head} {
8510 # the stuff on this branch isn't on any other branch
8511 if {![confirm_popup [mc "The commits on branch %s aren't on any other\
8512 branch.\nReally delete branch %s?" $head $head]]} return
8513 }
8514 nowbusy rmbranch
8515 update
8516 if {[catch {exec git branch -D $head} err]} {
8517 notbusy rmbranch
8518 error_popup $err
8519 return
8520 }
8521 removehead $id $head
8522 removedhead $id $head
8523 redrawtags $id
8524 notbusy rmbranch
8525 dispneartags 0
8526 run refill_reflist
8527}
8528
8529# Display a list of tags and heads
8530proc showrefs {} {
8531 global showrefstop bgcolor fgcolor selectbgcolor
8532 global bglist fglist reflistfilter reflist maincursor
8533
8534 set top .showrefs
8535 set showrefstop $top
8536 if {[winfo exists $top]} {
8537 raise $top
8538 refill_reflist
8539 return
8540 }
8541 toplevel $top
8542 wm title $top [mc "Tags and heads: %s" [file tail [pwd]]]
8543 make_transient $top .
8544 text $top.list -background $bgcolor -foreground $fgcolor \
8545 -selectbackground $selectbgcolor -font mainfont \
8546 -xscrollcommand "$top.xsb set" -yscrollcommand "$top.ysb set" \
8547 -width 30 -height 20 -cursor $maincursor \
8548 -spacing1 1 -spacing3 1 -state disabled
8549 $top.list tag configure highlight -background $selectbgcolor
8550 lappend bglist $top.list
8551 lappend fglist $top.list
8552 scrollbar $top.ysb -command "$top.list yview" -orient vertical
8553 scrollbar $top.xsb -command "$top.list xview" -orient horizontal
8554 grid $top.list $top.ysb -sticky nsew
8555 grid $top.xsb x -sticky ew
8556 frame $top.f
8557 label $top.f.l -text "[mc "Filter"]: "
8558 entry $top.f.e -width 20 -textvariable reflistfilter
8559 set reflistfilter "*"
8560 trace add variable reflistfilter write reflistfilter_change
8561 pack $top.f.e -side right -fill x -expand 1
8562 pack $top.f.l -side left
8563 grid $top.f - -sticky ew -pady 2
8564 button $top.close -command [list destroy $top] -text [mc "Close"]
8565 bind $top <Key-Escape> [list destroy $top]
8566 grid $top.close -
8567 grid columnconfigure $top 0 -weight 1
8568 grid rowconfigure $top 0 -weight 1
8569 bind $top.list <1> {break}
8570 bind $top.list <B1-Motion> {break}
8571 bind $top.list <ButtonRelease-1> {sel_reflist %W %x %y; break}
8572 set reflist {}
8573 refill_reflist
8574}
8575
8576proc sel_reflist {w x y} {
8577 global showrefstop reflist headids tagids otherrefids
8578
8579 if {![winfo exists $showrefstop]} return
8580 set l [lindex [split [$w index "@$x,$y"] "."] 0]
8581 set ref [lindex $reflist [expr {$l-1}]]
8582 set n [lindex $ref 0]
8583 switch -- [lindex $ref 1] {
8584 "H" {selbyid $headids($n)}
8585 "T" {selbyid $tagids($n)}
8586 "o" {selbyid $otherrefids($n)}
8587 }
8588 $showrefstop.list tag add highlight $l.0 "$l.0 lineend"
8589}
8590
8591proc unsel_reflist {} {
8592 global showrefstop
8593
8594 if {![info exists showrefstop] || ![winfo exists $showrefstop]} return
8595 $showrefstop.list tag remove highlight 0.0 end
8596}
8597
8598proc reflistfilter_change {n1 n2 op} {
8599 global reflistfilter
8600
8601 after cancel refill_reflist
8602 after 200 refill_reflist
8603}
8604
8605proc refill_reflist {} {
8606 global reflist reflistfilter showrefstop headids tagids otherrefids
8607 global curview
8608
8609 if {![info exists showrefstop] || ![winfo exists $showrefstop]} return
8610 set refs {}
8611 foreach n [array names headids] {
8612 if {[string match $reflistfilter $n]} {
8613 if {[commitinview $headids($n) $curview]} {
8614 lappend refs [list $n H]
8615 } else {
8616 interestedin $headids($n) {run refill_reflist}
8617 }
8618 }
8619 }
8620 foreach n [array names tagids] {
8621 if {[string match $reflistfilter $n]} {
8622 if {[commitinview $tagids($n) $curview]} {
8623 lappend refs [list $n T]
8624 } else {
8625 interestedin $tagids($n) {run refill_reflist}
8626 }
8627 }
8628 }
8629 foreach n [array names otherrefids] {
8630 if {[string match $reflistfilter $n]} {
8631 if {[commitinview $otherrefids($n) $curview]} {
8632 lappend refs [list $n o]
8633 } else {
8634 interestedin $otherrefids($n) {run refill_reflist}
8635 }
8636 }
8637 }
8638 set refs [lsort -index 0 $refs]
8639 if {$refs eq $reflist} return
8640
8641 # Update the contents of $showrefstop.list according to the
8642 # differences between $reflist (old) and $refs (new)
8643 $showrefstop.list conf -state normal
8644 $showrefstop.list insert end "\n"
8645 set i 0
8646 set j 0
8647 while {$i < [llength $reflist] || $j < [llength $refs]} {
8648 if {$i < [llength $reflist]} {
8649 if {$j < [llength $refs]} {
8650 set cmp [string compare [lindex $reflist $i 0] \
8651 [lindex $refs $j 0]]
8652 if {$cmp == 0} {
8653 set cmp [string compare [lindex $reflist $i 1] \
8654 [lindex $refs $j 1]]
8655 }
8656 } else {
8657 set cmp -1
8658 }
8659 } else {
8660 set cmp 1
8661 }
8662 switch -- $cmp {
8663 -1 {
8664 $showrefstop.list delete "[expr {$j+1}].0" "[expr {$j+2}].0"
8665 incr i
8666 }
8667 0 {
8668 incr i
8669 incr j
8670 }
8671 1 {
8672 set l [expr {$j + 1}]
8673 $showrefstop.list image create $l.0 -align baseline \
8674 -image reficon-[lindex $refs $j 1] -padx 2
8675 $showrefstop.list insert $l.1 "[lindex $refs $j 0]\n"
8676 incr j
8677 }
8678 }
8679 }
8680 set reflist $refs
8681 # delete last newline
8682 $showrefstop.list delete end-2c end-1c
8683 $showrefstop.list conf -state disabled
8684}
8685
8686# Stuff for finding nearby tags
8687proc getallcommits {} {
8688 global allcommits nextarc seeds allccache allcwait cachedarcs allcupdate
8689 global idheads idtags idotherrefs allparents tagobjid
8690
8691 if {![info exists allcommits]} {
8692 set nextarc 0
8693 set allcommits 0
8694 set seeds {}
8695 set allcwait 0
8696 set cachedarcs 0
8697 set allccache [file join [gitdir] "gitk.cache"]
8698 if {![catch {
8699 set f [open $allccache r]
8700 set allcwait 1
8701 getcache $f
8702 }]} return
8703 }
8704
8705 if {$allcwait} {
8706 return
8707 }
8708 set cmd [list | git rev-list --parents]
8709 set allcupdate [expr {$seeds ne {}}]
8710 if {!$allcupdate} {
8711 set ids "--all"
8712 } else {
8713 set refs [concat [array names idheads] [array names idtags] \
8714 [array names idotherrefs]]
8715 set ids {}
8716 set tagobjs {}
8717 foreach name [array names tagobjid] {
8718 lappend tagobjs $tagobjid($name)
8719 }
8720 foreach id [lsort -unique $refs] {
8721 if {![info exists allparents($id)] &&
8722 [lsearch -exact $tagobjs $id] < 0} {
8723 lappend ids $id
8724 }
8725 }
8726 if {$ids ne {}} {
8727 foreach id $seeds {
8728 lappend ids "^$id"
8729 }
8730 }
8731 }
8732 if {$ids ne {}} {
8733 set fd [open [concat $cmd $ids] r]
8734 fconfigure $fd -blocking 0
8735 incr allcommits
8736 nowbusy allcommits
8737 filerun $fd [list getallclines $fd]
8738 } else {
8739 dispneartags 0
8740 }
8741}
8742
8743# Since most commits have 1 parent and 1 child, we group strings of
8744# such commits into "arcs" joining branch/merge points (BMPs), which
8745# are commits that either don't have 1 parent or don't have 1 child.
8746#
8747# arcnos(id) - incoming arcs for BMP, arc we're on for other nodes
8748# arcout(id) - outgoing arcs for BMP
8749# arcids(a) - list of IDs on arc including end but not start
8750# arcstart(a) - BMP ID at start of arc
8751# arcend(a) - BMP ID at end of arc
8752# growing(a) - arc a is still growing
8753# arctags(a) - IDs out of arcids (excluding end) that have tags
8754# archeads(a) - IDs out of arcids (excluding end) that have heads
8755# The start of an arc is at the descendent end, so "incoming" means
8756# coming from descendents, and "outgoing" means going towards ancestors.
8757
8758proc getallclines {fd} {
8759 global allparents allchildren idtags idheads nextarc
8760 global arcnos arcids arctags arcout arcend arcstart archeads growing
8761 global seeds allcommits cachedarcs allcupdate
8762
8763 set nid 0
8764 while {[incr nid] <= 1000 && [gets $fd line] >= 0} {
8765 set id [lindex $line 0]
8766 if {[info exists allparents($id)]} {
8767 # seen it already
8768 continue
8769 }
8770 set cachedarcs 0
8771 set olds [lrange $line 1 end]
8772 set allparents($id) $olds
8773 if {![info exists allchildren($id)]} {
8774 set allchildren($id) {}
8775 set arcnos($id) {}
8776 lappend seeds $id
8777 } else {
8778 set a $arcnos($id)
8779 if {[llength $olds] == 1 && [llength $a] == 1} {
8780 lappend arcids($a) $id
8781 if {[info exists idtags($id)]} {
8782 lappend arctags($a) $id
8783 }
8784 if {[info exists idheads($id)]} {
8785 lappend archeads($a) $id
8786 }
8787 if {[info exists allparents($olds)]} {
8788 # seen parent already
8789 if {![info exists arcout($olds)]} {
8790 splitarc $olds
8791 }
8792 lappend arcids($a) $olds
8793 set arcend($a) $olds
8794 unset growing($a)
8795 }
8796 lappend allchildren($olds) $id
8797 lappend arcnos($olds) $a
8798 continue
8799 }
8800 }
8801 foreach a $arcnos($id) {
8802 lappend arcids($a) $id
8803 set arcend($a) $id
8804 unset growing($a)
8805 }
8806
8807 set ao {}
8808 foreach p $olds {
8809 lappend allchildren($p) $id
8810 set a [incr nextarc]
8811 set arcstart($a) $id
8812 set archeads($a) {}
8813 set arctags($a) {}
8814 set archeads($a) {}
8815 set arcids($a) {}
8816 lappend ao $a
8817 set growing($a) 1
8818 if {[info exists allparents($p)]} {
8819 # seen it already, may need to make a new branch
8820 if {![info exists arcout($p)]} {
8821 splitarc $p
8822 }
8823 lappend arcids($a) $p
8824 set arcend($a) $p
8825 unset growing($a)
8826 }
8827 lappend arcnos($p) $a
8828 }
8829 set arcout($id) $ao
8830 }
8831 if {$nid > 0} {
8832 global cached_dheads cached_dtags cached_atags
8833 catch {unset cached_dheads}
8834 catch {unset cached_dtags}
8835 catch {unset cached_atags}
8836 }
8837 if {![eof $fd]} {
8838 return [expr {$nid >= 1000? 2: 1}]
8839 }
8840 set cacheok 1
8841 if {[catch {
8842 fconfigure $fd -blocking 1
8843 close $fd
8844 } err]} {
8845 # got an error reading the list of commits
8846 # if we were updating, try rereading the whole thing again
8847 if {$allcupdate} {
8848 incr allcommits -1
8849 dropcache $err
8850 return
8851 }
8852 error_popup "[mc "Error reading commit topology information;\
8853 branch and preceding/following tag information\
8854 will be incomplete."]\n($err)"
8855 set cacheok 0
8856 }
8857 if {[incr allcommits -1] == 0} {
8858 notbusy allcommits
8859 if {$cacheok} {
8860 run savecache
8861 }
8862 }
8863 dispneartags 0
8864 return 0
8865}
8866
8867proc recalcarc {a} {
8868 global arctags archeads arcids idtags idheads
8869
8870 set at {}
8871 set ah {}
8872 foreach id [lrange $arcids($a) 0 end-1] {
8873 if {[info exists idtags($id)]} {
8874 lappend at $id
8875 }
8876 if {[info exists idheads($id)]} {
8877 lappend ah $id
8878 }
8879 }
8880 set arctags($a) $at
8881 set archeads($a) $ah
8882}
8883
8884proc splitarc {p} {
8885 global arcnos arcids nextarc arctags archeads idtags idheads
8886 global arcstart arcend arcout allparents growing
8887
8888 set a $arcnos($p)
8889 if {[llength $a] != 1} {
8890 puts "oops splitarc called but [llength $a] arcs already"
8891 return
8892 }
8893 set a [lindex $a 0]
8894 set i [lsearch -exact $arcids($a) $p]
8895 if {$i < 0} {
8896 puts "oops splitarc $p not in arc $a"
8897 return
8898 }
8899 set na [incr nextarc]
8900 if {[info exists arcend($a)]} {
8901 set arcend($na) $arcend($a)
8902 } else {
8903 set l [lindex $allparents([lindex $arcids($a) end]) 0]
8904 set j [lsearch -exact $arcnos($l) $a]
8905 set arcnos($l) [lreplace $arcnos($l) $j $j $na]
8906 }
8907 set tail [lrange $arcids($a) [expr {$i+1}] end]
8908 set arcids($a) [lrange $arcids($a) 0 $i]
8909 set arcend($a) $p
8910 set arcstart($na) $p
8911 set arcout($p) $na
8912 set arcids($na) $tail
8913 if {[info exists growing($a)]} {
8914 set growing($na) 1
8915 unset growing($a)
8916 }
8917
8918 foreach id $tail {
8919 if {[llength $arcnos($id)] == 1} {
8920 set arcnos($id) $na
8921 } else {
8922 set j [lsearch -exact $arcnos($id) $a]
8923 set arcnos($id) [lreplace $arcnos($id) $j $j $na]
8924 }
8925 }
8926
8927 # reconstruct tags and heads lists
8928 if {$arctags($a) ne {} || $archeads($a) ne {}} {
8929 recalcarc $a
8930 recalcarc $na
8931 } else {
8932 set arctags($na) {}
8933 set archeads($na) {}
8934 }
8935}
8936
8937# Update things for a new commit added that is a child of one
8938# existing commit. Used when cherry-picking.
8939proc addnewchild {id p} {
8940 global allparents allchildren idtags nextarc
8941 global arcnos arcids arctags arcout arcend arcstart archeads growing
8942 global seeds allcommits
8943
8944 if {![info exists allcommits] || ![info exists arcnos($p)]} return
8945 set allparents($id) [list $p]
8946 set allchildren($id) {}
8947 set arcnos($id) {}
8948 lappend seeds $id
8949 lappend allchildren($p) $id
8950 set a [incr nextarc]
8951 set arcstart($a) $id
8952 set archeads($a) {}
8953 set arctags($a) {}
8954 set arcids($a) [list $p]
8955 set arcend($a) $p
8956 if {![info exists arcout($p)]} {
8957 splitarc $p
8958 }
8959 lappend arcnos($p) $a
8960 set arcout($id) [list $a]
8961}
8962
8963# This implements a cache for the topology information.
8964# The cache saves, for each arc, the start and end of the arc,
8965# the ids on the arc, and the outgoing arcs from the end.
8966proc readcache {f} {
8967 global arcnos arcids arcout arcstart arcend arctags archeads nextarc
8968 global idtags idheads allparents cachedarcs possible_seeds seeds growing
8969 global allcwait
8970
8971 set a $nextarc
8972 set lim $cachedarcs
8973 if {$lim - $a > 500} {
8974 set lim [expr {$a + 500}]
8975 }
8976 if {[catch {
8977 if {$a == $lim} {
8978 # finish reading the cache and setting up arctags, etc.
8979 set line [gets $f]
8980 if {$line ne "1"} {error "bad final version"}
8981 close $f
8982 foreach id [array names idtags] {
8983 if {[info exists arcnos($id)] && [llength $arcnos($id)] == 1 &&
8984 [llength $allparents($id)] == 1} {
8985 set a [lindex $arcnos($id) 0]
8986 if {$arctags($a) eq {}} {
8987 recalcarc $a
8988 }
8989 }
8990 }
8991 foreach id [array names idheads] {
8992 if {[info exists arcnos($id)] && [llength $arcnos($id)] == 1 &&
8993 [llength $allparents($id)] == 1} {
8994 set a [lindex $arcnos($id) 0]
8995 if {$archeads($a) eq {}} {
8996 recalcarc $a
8997 }
8998 }
8999 }
9000 foreach id [lsort -unique $possible_seeds] {
9001 if {$arcnos($id) eq {}} {
9002 lappend seeds $id
9003 }
9004 }
9005 set allcwait 0
9006 } else {
9007 while {[incr a] <= $lim} {
9008 set line [gets $f]
9009 if {[llength $line] != 3} {error "bad line"}
9010 set s [lindex $line 0]
9011 set arcstart($a) $s
9012 lappend arcout($s) $a
9013 if {![info exists arcnos($s)]} {
9014 lappend possible_seeds $s
9015 set arcnos($s) {}
9016 }
9017 set e [lindex $line 1]
9018 if {$e eq {}} {
9019 set growing($a) 1
9020 } else {
9021 set arcend($a) $e
9022 if {![info exists arcout($e)]} {
9023 set arcout($e) {}
9024 }
9025 }
9026 set arcids($a) [lindex $line 2]
9027 foreach id $arcids($a) {
9028 lappend allparents($s) $id
9029 set s $id
9030 lappend arcnos($id) $a
9031 }
9032 if {![info exists allparents($s)]} {
9033 set allparents($s) {}
9034 }
9035 set arctags($a) {}
9036 set archeads($a) {}
9037 }
9038 set nextarc [expr {$a - 1}]
9039 }
9040 } err]} {
9041 dropcache $err
9042 return 0
9043 }
9044 if {!$allcwait} {
9045 getallcommits
9046 }
9047 return $allcwait
9048}
9049
9050proc getcache {f} {
9051 global nextarc cachedarcs possible_seeds
9052
9053 if {[catch {
9054 set line [gets $f]
9055 if {[llength $line] != 2 || [lindex $line 0] ne "1"} {error "bad version"}
9056 # make sure it's an integer
9057 set cachedarcs [expr {int([lindex $line 1])}]
9058 if {$cachedarcs < 0} {error "bad number of arcs"}
9059 set nextarc 0
9060 set possible_seeds {}
9061 run readcache $f
9062 } err]} {
9063 dropcache $err
9064 }
9065 return 0
9066}
9067
9068proc dropcache {err} {
9069 global allcwait nextarc cachedarcs seeds
9070
9071 #puts "dropping cache ($err)"
9072 foreach v {arcnos arcout arcids arcstart arcend growing \
9073 arctags archeads allparents allchildren} {
9074 global $v
9075 catch {unset $v}
9076 }
9077 set allcwait 0
9078 set nextarc 0
9079 set cachedarcs 0
9080 set seeds {}
9081 getallcommits
9082}
9083
9084proc writecache {f} {
9085 global cachearc cachedarcs allccache
9086 global arcstart arcend arcnos arcids arcout
9087
9088 set a $cachearc
9089 set lim $cachedarcs
9090 if {$lim - $a > 1000} {
9091 set lim [expr {$a + 1000}]
9092 }
9093 if {[catch {
9094 while {[incr a] <= $lim} {
9095 if {[info exists arcend($a)]} {
9096 puts $f [list $arcstart($a) $arcend($a) $arcids($a)]
9097 } else {
9098 puts $f [list $arcstart($a) {} $arcids($a)]
9099 }
9100 }
9101 } err]} {
9102 catch {close $f}
9103 catch {file delete $allccache}
9104 #puts "writing cache failed ($err)"
9105 return 0
9106 }
9107 set cachearc [expr {$a - 1}]
9108 if {$a > $cachedarcs} {
9109 puts $f "1"
9110 close $f
9111 return 0
9112 }
9113 return 1
9114}
9115
9116proc savecache {} {
9117 global nextarc cachedarcs cachearc allccache
9118
9119 if {$nextarc == $cachedarcs} return
9120 set cachearc 0
9121 set cachedarcs $nextarc
9122 catch {
9123 set f [open $allccache w]
9124 puts $f [list 1 $cachedarcs]
9125 run writecache $f
9126 }
9127}
9128
9129# Returns 1 if a is an ancestor of b, -1 if b is an ancestor of a,
9130# or 0 if neither is true.
9131proc anc_or_desc {a b} {
9132 global arcout arcstart arcend arcnos cached_isanc
9133
9134 if {$arcnos($a) eq $arcnos($b)} {
9135 # Both are on the same arc(s); either both are the same BMP,
9136 # or if one is not a BMP, the other is also not a BMP or is
9137 # the BMP at end of the arc (and it only has 1 incoming arc).
9138 # Or both can be BMPs with no incoming arcs.
9139 if {$a eq $b || $arcnos($a) eq {}} {
9140 return 0
9141 }
9142 # assert {[llength $arcnos($a)] == 1}
9143 set arc [lindex $arcnos($a) 0]
9144 set i [lsearch -exact $arcids($arc) $a]
9145 set j [lsearch -exact $arcids($arc) $b]
9146 if {$i < 0 || $i > $j} {
9147 return 1
9148 } else {
9149 return -1
9150 }
9151 }
9152
9153 if {![info exists arcout($a)]} {
9154 set arc [lindex $arcnos($a) 0]
9155 if {[info exists arcend($arc)]} {
9156 set aend $arcend($arc)
9157 } else {
9158 set aend {}
9159 }
9160 set a $arcstart($arc)
9161 } else {
9162 set aend $a
9163 }
9164 if {![info exists arcout($b)]} {
9165 set arc [lindex $arcnos($b) 0]
9166 if {[info exists arcend($arc)]} {
9167 set bend $arcend($arc)
9168 } else {
9169 set bend {}
9170 }
9171 set b $arcstart($arc)
9172 } else {
9173 set bend $b
9174 }
9175 if {$a eq $bend} {
9176 return 1
9177 }
9178 if {$b eq $aend} {
9179 return -1
9180 }
9181 if {[info exists cached_isanc($a,$bend)]} {
9182 if {$cached_isanc($a,$bend)} {
9183 return 1
9184 }
9185 }
9186 if {[info exists cached_isanc($b,$aend)]} {
9187 if {$cached_isanc($b,$aend)} {
9188 return -1
9189 }
9190 if {[info exists cached_isanc($a,$bend)]} {
9191 return 0
9192 }
9193 }
9194
9195 set todo [list $a $b]
9196 set anc($a) a
9197 set anc($b) b
9198 for {set i 0} {$i < [llength $todo]} {incr i} {
9199 set x [lindex $todo $i]
9200 if {$anc($x) eq {}} {
9201 continue
9202 }
9203 foreach arc $arcnos($x) {
9204 set xd $arcstart($arc)
9205 if {$xd eq $bend} {
9206 set cached_isanc($a,$bend) 1
9207 set cached_isanc($b,$aend) 0
9208 return 1
9209 } elseif {$xd eq $aend} {
9210 set cached_isanc($b,$aend) 1
9211 set cached_isanc($a,$bend) 0
9212 return -1
9213 }
9214 if {![info exists anc($xd)]} {
9215 set anc($xd) $anc($x)
9216 lappend todo $xd
9217 } elseif {$anc($xd) ne $anc($x)} {
9218 set anc($xd) {}
9219 }
9220 }
9221 }
9222 set cached_isanc($a,$bend) 0
9223 set cached_isanc($b,$aend) 0
9224 return 0
9225}
9226
9227# This identifies whether $desc has an ancestor that is
9228# a growing tip of the graph and which is not an ancestor of $anc
9229# and returns 0 if so and 1 if not.
9230# If we subsequently discover a tag on such a growing tip, and that
9231# turns out to be a descendent of $anc (which it could, since we
9232# don't necessarily see children before parents), then $desc
9233# isn't a good choice to display as a descendent tag of
9234# $anc (since it is the descendent of another tag which is
9235# a descendent of $anc). Similarly, $anc isn't a good choice to
9236# display as a ancestor tag of $desc.
9237#
9238proc is_certain {desc anc} {
9239 global arcnos arcout arcstart arcend growing problems
9240
9241 set certain {}
9242 if {[llength $arcnos($anc)] == 1} {
9243 # tags on the same arc are certain
9244 if {$arcnos($desc) eq $arcnos($anc)} {
9245 return 1
9246 }
9247 if {![info exists arcout($anc)]} {
9248 # if $anc is partway along an arc, use the start of the arc instead
9249 set a [lindex $arcnos($anc) 0]
9250 set anc $arcstart($a)
9251 }
9252 }
9253 if {[llength $arcnos($desc)] > 1 || [info exists arcout($desc)]} {
9254 set x $desc
9255 } else {
9256 set a [lindex $arcnos($desc) 0]
9257 set x $arcend($a)
9258 }
9259 if {$x == $anc} {
9260 return 1
9261 }
9262 set anclist [list $x]
9263 set dl($x) 1
9264 set nnh 1
9265 set ngrowanc 0
9266 for {set i 0} {$i < [llength $anclist] && ($nnh > 0 || $ngrowanc > 0)} {incr i} {
9267 set x [lindex $anclist $i]
9268 if {$dl($x)} {
9269 incr nnh -1
9270 }
9271 set done($x) 1
9272 foreach a $arcout($x) {
9273 if {[info exists growing($a)]} {
9274 if {![info exists growanc($x)] && $dl($x)} {
9275 set growanc($x) 1
9276 incr ngrowanc
9277 }
9278 } else {
9279 set y $arcend($a)
9280 if {[info exists dl($y)]} {
9281 if {$dl($y)} {
9282 if {!$dl($x)} {
9283 set dl($y) 0
9284 if {![info exists done($y)]} {
9285 incr nnh -1
9286 }
9287 if {[info exists growanc($x)]} {
9288 incr ngrowanc -1
9289 }
9290 set xl [list $y]
9291 for {set k 0} {$k < [llength $xl]} {incr k} {
9292 set z [lindex $xl $k]
9293 foreach c $arcout($z) {
9294 if {[info exists arcend($c)]} {
9295 set v $arcend($c)
9296 if {[info exists dl($v)] && $dl($v)} {
9297 set dl($v) 0
9298 if {![info exists done($v)]} {
9299 incr nnh -1
9300 }
9301 if {[info exists growanc($v)]} {
9302 incr ngrowanc -1
9303 }
9304 lappend xl $v
9305 }
9306 }
9307 }
9308 }
9309 }
9310 }
9311 } elseif {$y eq $anc || !$dl($x)} {
9312 set dl($y) 0
9313 lappend anclist $y
9314 } else {
9315 set dl($y) 1
9316 lappend anclist $y
9317 incr nnh
9318 }
9319 }
9320 }
9321 }
9322 foreach x [array names growanc] {
9323 if {$dl($x)} {
9324 return 0
9325 }
9326 return 0
9327 }
9328 return 1
9329}
9330
9331proc validate_arctags {a} {
9332 global arctags idtags
9333
9334 set i -1
9335 set na $arctags($a)
9336 foreach id $arctags($a) {
9337 incr i
9338 if {![info exists idtags($id)]} {
9339 set na [lreplace $na $i $i]
9340 incr i -1
9341 }
9342 }
9343 set arctags($a) $na
9344}
9345
9346proc validate_archeads {a} {
9347 global archeads idheads
9348
9349 set i -1
9350 set na $archeads($a)
9351 foreach id $archeads($a) {
9352 incr i
9353 if {![info exists idheads($id)]} {
9354 set na [lreplace $na $i $i]
9355 incr i -1
9356 }
9357 }
9358 set archeads($a) $na
9359}
9360
9361# Return the list of IDs that have tags that are descendents of id,
9362# ignoring IDs that are descendents of IDs already reported.
9363proc desctags {id} {
9364 global arcnos arcstart arcids arctags idtags allparents
9365 global growing cached_dtags
9366
9367 if {![info exists allparents($id)]} {
9368 return {}
9369 }
9370 set t1 [clock clicks -milliseconds]
9371 set argid $id
9372 if {[llength $arcnos($id)] == 1 && [llength $allparents($id)] == 1} {
9373 # part-way along an arc; check that arc first
9374 set a [lindex $arcnos($id) 0]
9375 if {$arctags($a) ne {}} {
9376 validate_arctags $a
9377 set i [lsearch -exact $arcids($a) $id]
9378 set tid {}
9379 foreach t $arctags($a) {
9380 set j [lsearch -exact $arcids($a) $t]
9381 if {$j >= $i} break
9382 set tid $t
9383 }
9384 if {$tid ne {}} {
9385 return $tid
9386 }
9387 }
9388 set id $arcstart($a)
9389 if {[info exists idtags($id)]} {
9390 return $id
9391 }
9392 }
9393 if {[info exists cached_dtags($id)]} {
9394 return $cached_dtags($id)
9395 }
9396
9397 set origid $id
9398 set todo [list $id]
9399 set queued($id) 1
9400 set nc 1
9401 for {set i 0} {$i < [llength $todo] && $nc > 0} {incr i} {
9402 set id [lindex $todo $i]
9403 set done($id) 1
9404 set ta [info exists hastaggedancestor($id)]
9405 if {!$ta} {
9406 incr nc -1
9407 }
9408 # ignore tags on starting node
9409 if {!$ta && $i > 0} {
9410 if {[info exists idtags($id)]} {
9411 set tagloc($id) $id
9412 set ta 1
9413 } elseif {[info exists cached_dtags($id)]} {
9414 set tagloc($id) $cached_dtags($id)
9415 set ta 1
9416 }
9417 }
9418 foreach a $arcnos($id) {
9419 set d $arcstart($a)
9420 if {!$ta && $arctags($a) ne {}} {
9421 validate_arctags $a
9422 if {$arctags($a) ne {}} {
9423 lappend tagloc($id) [lindex $arctags($a) end]
9424 }
9425 }
9426 if {$ta || $arctags($a) ne {}} {
9427 set tomark [list $d]
9428 for {set j 0} {$j < [llength $tomark]} {incr j} {
9429 set dd [lindex $tomark $j]
9430 if {![info exists hastaggedancestor($dd)]} {
9431 if {[info exists done($dd)]} {
9432 foreach b $arcnos($dd) {
9433 lappend tomark $arcstart($b)
9434 }
9435 if {[info exists tagloc($dd)]} {
9436 unset tagloc($dd)
9437 }
9438 } elseif {[info exists queued($dd)]} {
9439 incr nc -1
9440 }
9441 set hastaggedancestor($dd) 1
9442 }
9443 }
9444 }
9445 if {![info exists queued($d)]} {
9446 lappend todo $d
9447 set queued($d) 1
9448 if {![info exists hastaggedancestor($d)]} {
9449 incr nc
9450 }
9451 }
9452 }
9453 }
9454 set tags {}
9455 foreach id [array names tagloc] {
9456 if {![info exists hastaggedancestor($id)]} {
9457 foreach t $tagloc($id) {
9458 if {[lsearch -exact $tags $t] < 0} {
9459 lappend tags $t
9460 }
9461 }
9462 }
9463 }
9464 set t2 [clock clicks -milliseconds]
9465 set loopix $i
9466
9467 # remove tags that are descendents of other tags
9468 for {set i 0} {$i < [llength $tags]} {incr i} {
9469 set a [lindex $tags $i]
9470 for {set j 0} {$j < $i} {incr j} {
9471 set b [lindex $tags $j]
9472 set r [anc_or_desc $a $b]
9473 if {$r == 1} {
9474 set tags [lreplace $tags $j $j]
9475 incr j -1
9476 incr i -1
9477 } elseif {$r == -1} {
9478 set tags [lreplace $tags $i $i]
9479 incr i -1
9480 break
9481 }
9482 }
9483 }
9484
9485 if {[array names growing] ne {}} {
9486 # graph isn't finished, need to check if any tag could get
9487 # eclipsed by another tag coming later. Simply ignore any
9488 # tags that could later get eclipsed.
9489 set ctags {}
9490 foreach t $tags {
9491 if {[is_certain $t $origid]} {
9492 lappend ctags $t
9493 }
9494 }
9495 if {$tags eq $ctags} {
9496 set cached_dtags($origid) $tags
9497 } else {
9498 set tags $ctags
9499 }
9500 } else {
9501 set cached_dtags($origid) $tags
9502 }
9503 set t3 [clock clicks -milliseconds]
9504 if {0 && $t3 - $t1 >= 100} {
9505 puts "iterating descendents ($loopix/[llength $todo] nodes) took\
9506 [expr {$t2-$t1}]+[expr {$t3-$t2}]ms, $nc candidates left"
9507 }
9508 return $tags
9509}
9510
9511proc anctags {id} {
9512 global arcnos arcids arcout arcend arctags idtags allparents
9513 global growing cached_atags
9514
9515 if {![info exists allparents($id)]} {
9516 return {}
9517 }
9518 set t1 [clock clicks -milliseconds]
9519 set argid $id
9520 if {[llength $arcnos($id)] == 1 && [llength $allparents($id)] == 1} {
9521 # part-way along an arc; check that arc first
9522 set a [lindex $arcnos($id) 0]
9523 if {$arctags($a) ne {}} {
9524 validate_arctags $a
9525 set i [lsearch -exact $arcids($a) $id]
9526 foreach t $arctags($a) {
9527 set j [lsearch -exact $arcids($a) $t]
9528 if {$j > $i} {
9529 return $t
9530 }
9531 }
9532 }
9533 if {![info exists arcend($a)]} {
9534 return {}
9535 }
9536 set id $arcend($a)
9537 if {[info exists idtags($id)]} {
9538 return $id
9539 }
9540 }
9541 if {[info exists cached_atags($id)]} {
9542 return $cached_atags($id)
9543 }
9544
9545 set origid $id
9546 set todo [list $id]
9547 set queued($id) 1
9548 set taglist {}
9549 set nc 1
9550 for {set i 0} {$i < [llength $todo] && $nc > 0} {incr i} {
9551 set id [lindex $todo $i]
9552 set done($id) 1
9553 set td [info exists hastaggeddescendent($id)]
9554 if {!$td} {
9555 incr nc -1
9556 }
9557 # ignore tags on starting node
9558 if {!$td && $i > 0} {
9559 if {[info exists idtags($id)]} {
9560 set tagloc($id) $id
9561 set td 1
9562 } elseif {[info exists cached_atags($id)]} {
9563 set tagloc($id) $cached_atags($id)
9564 set td 1
9565 }
9566 }
9567 foreach a $arcout($id) {
9568 if {!$td && $arctags($a) ne {}} {
9569 validate_arctags $a
9570 if {$arctags($a) ne {}} {
9571 lappend tagloc($id) [lindex $arctags($a) 0]
9572 }
9573 }
9574 if {![info exists arcend($a)]} continue
9575 set d $arcend($a)
9576 if {$td || $arctags($a) ne {}} {
9577 set tomark [list $d]
9578 for {set j 0} {$j < [llength $tomark]} {incr j} {
9579 set dd [lindex $tomark $j]
9580 if {![info exists hastaggeddescendent($dd)]} {
9581 if {[info exists done($dd)]} {
9582 foreach b $arcout($dd) {
9583 if {[info exists arcend($b)]} {
9584 lappend tomark $arcend($b)
9585 }
9586 }
9587 if {[info exists tagloc($dd)]} {
9588 unset tagloc($dd)
9589 }
9590 } elseif {[info exists queued($dd)]} {
9591 incr nc -1
9592 }
9593 set hastaggeddescendent($dd) 1
9594 }
9595 }
9596 }
9597 if {![info exists queued($d)]} {
9598 lappend todo $d
9599 set queued($d) 1
9600 if {![info exists hastaggeddescendent($d)]} {
9601 incr nc
9602 }
9603 }
9604 }
9605 }
9606 set t2 [clock clicks -milliseconds]
9607 set loopix $i
9608 set tags {}
9609 foreach id [array names tagloc] {
9610 if {![info exists hastaggeddescendent($id)]} {
9611 foreach t $tagloc($id) {
9612 if {[lsearch -exact $tags $t] < 0} {
9613 lappend tags $t
9614 }
9615 }
9616 }
9617 }
9618
9619 # remove tags that are ancestors of other tags
9620 for {set i 0} {$i < [llength $tags]} {incr i} {
9621 set a [lindex $tags $i]
9622 for {set j 0} {$j < $i} {incr j} {
9623 set b [lindex $tags $j]
9624 set r [anc_or_desc $a $b]
9625 if {$r == -1} {
9626 set tags [lreplace $tags $j $j]
9627 incr j -1
9628 incr i -1
9629 } elseif {$r == 1} {
9630 set tags [lreplace $tags $i $i]
9631 incr i -1
9632 break
9633 }
9634 }
9635 }
9636
9637 if {[array names growing] ne {}} {
9638 # graph isn't finished, need to check if any tag could get
9639 # eclipsed by another tag coming later. Simply ignore any
9640 # tags that could later get eclipsed.
9641 set ctags {}
9642 foreach t $tags {
9643 if {[is_certain $origid $t]} {
9644 lappend ctags $t
9645 }
9646 }
9647 if {$tags eq $ctags} {
9648 set cached_atags($origid) $tags
9649 } else {
9650 set tags $ctags
9651 }
9652 } else {
9653 set cached_atags($origid) $tags
9654 }
9655 set t3 [clock clicks -milliseconds]
9656 if {0 && $t3 - $t1 >= 100} {
9657 puts "iterating ancestors ($loopix/[llength $todo] nodes) took\
9658 [expr {$t2-$t1}]+[expr {$t3-$t2}]ms, $nc candidates left"
9659 }
9660 return $tags
9661}
9662
9663# Return the list of IDs that have heads that are descendents of id,
9664# including id itself if it has a head.
9665proc descheads {id} {
9666 global arcnos arcstart arcids archeads idheads cached_dheads
9667 global allparents
9668
9669 if {![info exists allparents($id)]} {
9670 return {}
9671 }
9672 set aret {}
9673 if {[llength $arcnos($id)] == 1 && [llength $allparents($id)] == 1} {
9674 # part-way along an arc; check it first
9675 set a [lindex $arcnos($id) 0]
9676 if {$archeads($a) ne {}} {
9677 validate_archeads $a
9678 set i [lsearch -exact $arcids($a) $id]
9679 foreach t $archeads($a) {
9680 set j [lsearch -exact $arcids($a) $t]
9681 if {$j > $i} break
9682 lappend aret $t
9683 }
9684 }
9685 set id $arcstart($a)
9686 }
9687 set origid $id
9688 set todo [list $id]
9689 set seen($id) 1
9690 set ret {}
9691 for {set i 0} {$i < [llength $todo]} {incr i} {
9692 set id [lindex $todo $i]
9693 if {[info exists cached_dheads($id)]} {
9694 set ret [concat $ret $cached_dheads($id)]
9695 } else {
9696 if {[info exists idheads($id)]} {
9697 lappend ret $id
9698 }
9699 foreach a $arcnos($id) {
9700 if {$archeads($a) ne {}} {
9701 validate_archeads $a
9702 if {$archeads($a) ne {}} {
9703 set ret [concat $ret $archeads($a)]
9704 }
9705 }
9706 set d $arcstart($a)
9707 if {![info exists seen($d)]} {
9708 lappend todo $d
9709 set seen($d) 1
9710 }
9711 }
9712 }
9713 }
9714 set ret [lsort -unique $ret]
9715 set cached_dheads($origid) $ret
9716 return [concat $ret $aret]
9717}
9718
9719proc addedtag {id} {
9720 global arcnos arcout cached_dtags cached_atags
9721
9722 if {![info exists arcnos($id)]} return
9723 if {![info exists arcout($id)]} {
9724 recalcarc [lindex $arcnos($id) 0]
9725 }
9726 catch {unset cached_dtags}
9727 catch {unset cached_atags}
9728}
9729
9730proc addedhead {hid head} {
9731 global arcnos arcout cached_dheads
9732
9733 if {![info exists arcnos($hid)]} return
9734 if {![info exists arcout($hid)]} {
9735 recalcarc [lindex $arcnos($hid) 0]
9736 }
9737 catch {unset cached_dheads}
9738}
9739
9740proc removedhead {hid head} {
9741 global cached_dheads
9742
9743 catch {unset cached_dheads}
9744}
9745
9746proc movedhead {hid head} {
9747 global arcnos arcout cached_dheads
9748
9749 if {![info exists arcnos($hid)]} return
9750 if {![info exists arcout($hid)]} {
9751 recalcarc [lindex $arcnos($hid) 0]
9752 }
9753 catch {unset cached_dheads}
9754}
9755
9756proc changedrefs {} {
9757 global cached_dheads cached_dtags cached_atags
9758 global arctags archeads arcnos arcout idheads idtags
9759
9760 foreach id [concat [array names idheads] [array names idtags]] {
9761 if {[info exists arcnos($id)] && ![info exists arcout($id)]} {
9762 set a [lindex $arcnos($id) 0]
9763 if {![info exists donearc($a)]} {
9764 recalcarc $a
9765 set donearc($a) 1
9766 }
9767 }
9768 }
9769 catch {unset cached_dtags}
9770 catch {unset cached_atags}
9771 catch {unset cached_dheads}
9772}
9773
9774proc rereadrefs {} {
9775 global idtags idheads idotherrefs mainheadid
9776
9777 set refids [concat [array names idtags] \
9778 [array names idheads] [array names idotherrefs]]
9779 foreach id $refids {
9780 if {![info exists ref($id)]} {
9781 set ref($id) [listrefs $id]
9782 }
9783 }
9784 set oldmainhead $mainheadid
9785 readrefs
9786 changedrefs
9787 set refids [lsort -unique [concat $refids [array names idtags] \
9788 [array names idheads] [array names idotherrefs]]]
9789 foreach id $refids {
9790 set v [listrefs $id]
9791 if {![info exists ref($id)] || $ref($id) != $v} {
9792 redrawtags $id
9793 }
9794 }
9795 if {$oldmainhead ne $mainheadid} {
9796 redrawtags $oldmainhead
9797 redrawtags $mainheadid
9798 }
9799 run refill_reflist
9800}
9801
9802proc listrefs {id} {
9803 global idtags idheads idotherrefs
9804
9805 set x {}
9806 if {[info exists idtags($id)]} {
9807 set x $idtags($id)
9808 }
9809 set y {}
9810 if {[info exists idheads($id)]} {
9811 set y $idheads($id)
9812 }
9813 set z {}
9814 if {[info exists idotherrefs($id)]} {
9815 set z $idotherrefs($id)
9816 }
9817 return [list $x $y $z]
9818}
9819
9820proc showtag {tag isnew} {
9821 global ctext tagcontents tagids linknum tagobjid
9822
9823 if {$isnew} {
9824 addtohistory [list showtag $tag 0]
9825 }
9826 $ctext conf -state normal
9827 clear_ctext
9828 settabs 0
9829 set linknum 0
9830 if {![info exists tagcontents($tag)]} {
9831 catch {
9832 set tagcontents($tag) [exec git cat-file tag $tagobjid($tag)]
9833 }
9834 }
9835 if {[info exists tagcontents($tag)]} {
9836 set text $tagcontents($tag)
9837 } else {
9838 set text "[mc "Tag"]: $tag\n[mc "Id"]: $tagids($tag)"
9839 }
9840 appendwithlinks $text {}
9841 $ctext conf -state disabled
9842 init_flist {}
9843}
9844
9845proc doquit {} {
9846 global stopped
9847 global gitktmpdir
9848
9849 set stopped 100
9850 savestuff .
9851 destroy .
9852
9853 if {[info exists gitktmpdir]} {
9854 catch {file delete -force $gitktmpdir}
9855 }
9856}
9857
9858proc mkfontdisp {font top which} {
9859 global fontattr fontpref $font
9860
9861 set fontpref($font) [set $font]
9862 button $top.${font}but -text $which -font optionfont \
9863 -command [list choosefont $font $which]
9864 label $top.$font -relief flat -font $font \
9865 -text $fontattr($font,family) -justify left
9866 grid x $top.${font}but $top.$font -sticky w
9867}
9868
9869proc choosefont {font which} {
9870 global fontparam fontlist fonttop fontattr
9871 global prefstop
9872
9873 set fontparam(which) $which
9874 set fontparam(font) $font
9875 set fontparam(family) [font actual $font -family]
9876 set fontparam(size) $fontattr($font,size)
9877 set fontparam(weight) $fontattr($font,weight)
9878 set fontparam(slant) $fontattr($font,slant)
9879 set top .gitkfont
9880 set fonttop $top
9881 if {![winfo exists $top]} {
9882 font create sample
9883 eval font config sample [font actual $font]
9884 toplevel $top
9885 make_transient $top $prefstop
9886 wm title $top [mc "Gitk font chooser"]
9887 label $top.l -textvariable fontparam(which)
9888 pack $top.l -side top
9889 set fontlist [lsort [font families]]
9890 frame $top.f
9891 listbox $top.f.fam -listvariable fontlist \
9892 -yscrollcommand [list $top.f.sb set]
9893 bind $top.f.fam <<ListboxSelect>> selfontfam
9894 scrollbar $top.f.sb -command [list $top.f.fam yview]
9895 pack $top.f.sb -side right -fill y
9896 pack $top.f.fam -side left -fill both -expand 1
9897 pack $top.f -side top -fill both -expand 1
9898 frame $top.g
9899 spinbox $top.g.size -from 4 -to 40 -width 4 \
9900 -textvariable fontparam(size) \
9901 -validatecommand {string is integer -strict %s}
9902 checkbutton $top.g.bold -padx 5 \
9903 -font {{Times New Roman} 12 bold} -text [mc "B"] -indicatoron 0 \
9904 -variable fontparam(weight) -onvalue bold -offvalue normal
9905 checkbutton $top.g.ital -padx 5 \
9906 -font {{Times New Roman} 12 italic} -text [mc "I"] -indicatoron 0 \
9907 -variable fontparam(slant) -onvalue italic -offvalue roman
9908 pack $top.g.size $top.g.bold $top.g.ital -side left
9909 pack $top.g -side top
9910 canvas $top.c -width 150 -height 50 -border 2 -relief sunk \
9911 -background white
9912 $top.c create text 100 25 -anchor center -text $which -font sample \
9913 -fill black -tags text
9914 bind $top.c <Configure> [list centertext $top.c]
9915 pack $top.c -side top -fill x
9916 frame $top.buts
9917 button $top.buts.ok -text [mc "OK"] -command fontok -default active
9918 button $top.buts.can -text [mc "Cancel"] -command fontcan -default normal
9919 bind $top <Key-Return> fontok
9920 bind $top <Key-Escape> fontcan
9921 grid $top.buts.ok $top.buts.can
9922 grid columnconfigure $top.buts 0 -weight 1 -uniform a
9923 grid columnconfigure $top.buts 1 -weight 1 -uniform a
9924 pack $top.buts -side bottom -fill x
9925 trace add variable fontparam write chg_fontparam
9926 } else {
9927 raise $top
9928 $top.c itemconf text -text $which
9929 }
9930 set i [lsearch -exact $fontlist $fontparam(family)]
9931 if {$i >= 0} {
9932 $top.f.fam selection set $i
9933 $top.f.fam see $i
9934 }
9935}
9936
9937proc centertext {w} {
9938 $w coords text [expr {[winfo width $w] / 2}] [expr {[winfo height $w] / 2}]
9939}
9940
9941proc fontok {} {
9942 global fontparam fontpref prefstop
9943
9944 set f $fontparam(font)
9945 set fontpref($f) [list $fontparam(family) $fontparam(size)]
9946 if {$fontparam(weight) eq "bold"} {
9947 lappend fontpref($f) "bold"
9948 }
9949 if {$fontparam(slant) eq "italic"} {
9950 lappend fontpref($f) "italic"
9951 }
9952 set w $prefstop.$f
9953 $w conf -text $fontparam(family) -font $fontpref($f)
9954
9955 fontcan
9956}
9957
9958proc fontcan {} {
9959 global fonttop fontparam
9960
9961 if {[info exists fonttop]} {
9962 catch {destroy $fonttop}
9963 catch {font delete sample}
9964 unset fonttop
9965 unset fontparam
9966 }
9967}
9968
9969proc selfontfam {} {
9970 global fonttop fontparam
9971
9972 set i [$fonttop.f.fam curselection]
9973 if {$i ne {}} {
9974 set fontparam(family) [$fonttop.f.fam get $i]
9975 }
9976}
9977
9978proc chg_fontparam {v sub op} {
9979 global fontparam
9980
9981 font config sample -$sub $fontparam($sub)
9982}
9983
9984proc doprefs {} {
9985 global maxwidth maxgraphpct
9986 global oldprefs prefstop showneartags showlocalchanges
9987 global bgcolor fgcolor ctext diffcolors selectbgcolor markbgcolor
9988 global tabstop limitdiffs autoselect extdifftool perfile_attrs
9989
9990 set top .gitkprefs
9991 set prefstop $top
9992 if {[winfo exists $top]} {
9993 raise $top
9994 return
9995 }
9996 foreach v {maxwidth maxgraphpct showneartags showlocalchanges \
9997 limitdiffs tabstop perfile_attrs} {
9998 set oldprefs($v) [set $v]
9999 }
10000 toplevel $top
10001 wm title $top [mc "Gitk preferences"]
10002 make_transient $top .
10003 label $top.ldisp -text [mc "Commit list display options"]
10004 grid $top.ldisp - -sticky w -pady 10
10005 label $top.spacer -text " "
10006 label $top.maxwidthl -text [mc "Maximum graph width (lines)"] \
10007 -font optionfont
10008 spinbox $top.maxwidth -from 0 -to 100 -width 4 -textvariable maxwidth
10009 grid $top.spacer $top.maxwidthl $top.maxwidth -sticky w
10010 label $top.maxpctl -text [mc "Maximum graph width (% of pane)"] \
10011 -font optionfont
10012 spinbox $top.maxpct -from 1 -to 100 -width 4 -textvariable maxgraphpct
10013 grid x $top.maxpctl $top.maxpct -sticky w
10014 frame $top.showlocal
10015 label $top.showlocal.l -text [mc "Show local changes"] -font optionfont
10016 checkbutton $top.showlocal.b -variable showlocalchanges
10017 pack $top.showlocal.b $top.showlocal.l -side left
10018 grid x $top.showlocal -sticky w
10019 frame $top.autoselect
10020 label $top.autoselect.l -text [mc "Auto-select SHA1"] -font optionfont
10021 checkbutton $top.autoselect.b -variable autoselect
10022 pack $top.autoselect.b $top.autoselect.l -side left
10023 grid x $top.autoselect -sticky w
10024
10025 label $top.ddisp -text [mc "Diff display options"]
10026 grid $top.ddisp - -sticky w -pady 10
10027 label $top.tabstopl -text [mc "Tab spacing"] -font optionfont
10028 spinbox $top.tabstop -from 1 -to 20 -width 4 -textvariable tabstop
10029 grid x $top.tabstopl $top.tabstop -sticky w
10030 frame $top.ntag
10031 label $top.ntag.l -text [mc "Display nearby tags"] -font optionfont
10032 checkbutton $top.ntag.b -variable showneartags
10033 pack $top.ntag.b $top.ntag.l -side left
10034 grid x $top.ntag -sticky w
10035 frame $top.ldiff
10036 label $top.ldiff.l -text [mc "Limit diffs to listed paths"] -font optionfont
10037 checkbutton $top.ldiff.b -variable limitdiffs
10038 pack $top.ldiff.b $top.ldiff.l -side left
10039 grid x $top.ldiff -sticky w
10040 frame $top.lattr
10041 label $top.lattr.l -text [mc "Support per-file encodings"] -font optionfont
10042 checkbutton $top.lattr.b -variable perfile_attrs
10043 pack $top.lattr.b $top.lattr.l -side left
10044 grid x $top.lattr -sticky w
10045
10046 entry $top.extdifft -textvariable extdifftool
10047 frame $top.extdifff
10048 label $top.extdifff.l -text [mc "External diff tool" ] -font optionfont \
10049 -padx 10
10050 button $top.extdifff.b -text [mc "Choose..."] -font optionfont \
10051 -command choose_extdiff
10052 pack $top.extdifff.l $top.extdifff.b -side left
10053 grid x $top.extdifff $top.extdifft -sticky w
10054
10055 label $top.cdisp -text [mc "Colors: press to choose"]
10056 grid $top.cdisp - -sticky w -pady 10
10057 label $top.bg -padx 40 -relief sunk -background $bgcolor
10058 button $top.bgbut -text [mc "Background"] -font optionfont \
10059 -command [list choosecolor bgcolor {} $top.bg background setbg]
10060 grid x $top.bgbut $top.bg -sticky w
10061 label $top.fg -padx 40 -relief sunk -background $fgcolor
10062 button $top.fgbut -text [mc "Foreground"] -font optionfont \
10063 -command [list choosecolor fgcolor {} $top.fg foreground setfg]
10064 grid x $top.fgbut $top.fg -sticky w
10065 label $top.diffold -padx 40 -relief sunk -background [lindex $diffcolors 0]
10066 button $top.diffoldbut -text [mc "Diff: old lines"] -font optionfont \
10067 -command [list choosecolor diffcolors 0 $top.diffold "diff old lines" \
10068 [list $ctext tag conf d0 -foreground]]
10069 grid x $top.diffoldbut $top.diffold -sticky w
10070 label $top.diffnew -padx 40 -relief sunk -background [lindex $diffcolors 1]
10071 button $top.diffnewbut -text [mc "Diff: new lines"] -font optionfont \
10072 -command [list choosecolor diffcolors 1 $top.diffnew "diff new lines" \
10073 [list $ctext tag conf dresult -foreground]]
10074 grid x $top.diffnewbut $top.diffnew -sticky w
10075 label $top.hunksep -padx 40 -relief sunk -background [lindex $diffcolors 2]
10076 button $top.hunksepbut -text [mc "Diff: hunk header"] -font optionfont \
10077 -command [list choosecolor diffcolors 2 $top.hunksep \
10078 "diff hunk header" \
10079 [list $ctext tag conf hunksep -foreground]]
10080 grid x $top.hunksepbut $top.hunksep -sticky w
10081 label $top.markbgsep -padx 40 -relief sunk -background $markbgcolor
10082 button $top.markbgbut -text [mc "Marked line bg"] -font optionfont \
10083 -command [list choosecolor markbgcolor {} $top.markbgsep \
10084 [mc "marked line background"] \
10085 [list $ctext tag conf omark -background]]
10086 grid x $top.markbgbut $top.markbgsep -sticky w
10087 label $top.selbgsep -padx 40 -relief sunk -background $selectbgcolor
10088 button $top.selbgbut -text [mc "Select bg"] -font optionfont \
10089 -command [list choosecolor selectbgcolor {} $top.selbgsep background setselbg]
10090 grid x $top.selbgbut $top.selbgsep -sticky w
10091
10092 label $top.cfont -text [mc "Fonts: press to choose"]
10093 grid $top.cfont - -sticky w -pady 10
10094 mkfontdisp mainfont $top [mc "Main font"]
10095 mkfontdisp textfont $top [mc "Diff display font"]
10096 mkfontdisp uifont $top [mc "User interface font"]
10097
10098 frame $top.buts
10099 button $top.buts.ok -text [mc "OK"] -command prefsok -default active
10100 button $top.buts.can -text [mc "Cancel"] -command prefscan -default normal
10101 bind $top <Key-Return> prefsok
10102 bind $top <Key-Escape> prefscan
10103 grid $top.buts.ok $top.buts.can
10104 grid columnconfigure $top.buts 0 -weight 1 -uniform a
10105 grid columnconfigure $top.buts 1 -weight 1 -uniform a
10106 grid $top.buts - - -pady 10 -sticky ew
10107 bind $top <Visibility> "focus $top.buts.ok"
10108}
10109
10110proc choose_extdiff {} {
10111 global extdifftool
10112
10113 set prog [tk_getOpenFile -title "External diff tool" -multiple false]
10114 if {$prog ne {}} {
10115 set extdifftool $prog
10116 }
10117}
10118
10119proc choosecolor {v vi w x cmd} {
10120 global $v
10121
10122 set c [tk_chooseColor -initialcolor [lindex [set $v] $vi] \
10123 -title [mc "Gitk: choose color for %s" $x]]
10124 if {$c eq {}} return
10125 $w conf -background $c
10126 lset $v $vi $c
10127 eval $cmd $c
10128}
10129
10130proc setselbg {c} {
10131 global bglist cflist
10132 foreach w $bglist {
10133 $w configure -selectbackground $c
10134 }
10135 $cflist tag configure highlight \
10136 -background [$cflist cget -selectbackground]
10137 allcanvs itemconf secsel -fill $c
10138}
10139
10140proc setbg {c} {
10141 global bglist
10142
10143 foreach w $bglist {
10144 $w conf -background $c
10145 }
10146}
10147
10148proc setfg {c} {
10149 global fglist canv
10150
10151 foreach w $fglist {
10152 $w conf -foreground $c
10153 }
10154 allcanvs itemconf text -fill $c
10155 $canv itemconf circle -outline $c
10156}
10157
10158proc prefscan {} {
10159 global oldprefs prefstop
10160
10161 foreach v {maxwidth maxgraphpct showneartags showlocalchanges \
10162 limitdiffs tabstop perfile_attrs} {
10163 global $v
10164 set $v $oldprefs($v)
10165 }
10166 catch {destroy $prefstop}
10167 unset prefstop
10168 fontcan
10169}
10170
10171proc prefsok {} {
10172 global maxwidth maxgraphpct
10173 global oldprefs prefstop showneartags showlocalchanges
10174 global fontpref mainfont textfont uifont
10175 global limitdiffs treediffs perfile_attrs
10176
10177 catch {destroy $prefstop}
10178 unset prefstop
10179 fontcan
10180 set fontchanged 0
10181 if {$mainfont ne $fontpref(mainfont)} {
10182 set mainfont $fontpref(mainfont)
10183 parsefont mainfont $mainfont
10184 eval font configure mainfont [fontflags mainfont]
10185 eval font configure mainfontbold [fontflags mainfont 1]
10186 setcoords
10187 set fontchanged 1
10188 }
10189 if {$textfont ne $fontpref(textfont)} {
10190 set textfont $fontpref(textfont)
10191 parsefont textfont $textfont
10192 eval font configure textfont [fontflags textfont]
10193 eval font configure textfontbold [fontflags textfont 1]
10194 }
10195 if {$uifont ne $fontpref(uifont)} {
10196 set uifont $fontpref(uifont)
10197 parsefont uifont $uifont
10198 eval font configure uifont [fontflags uifont]
10199 }
10200 settabs
10201 if {$showlocalchanges != $oldprefs(showlocalchanges)} {
10202 if {$showlocalchanges} {
10203 doshowlocalchanges
10204 } else {
10205 dohidelocalchanges
10206 }
10207 }
10208 if {$limitdiffs != $oldprefs(limitdiffs) ||
10209 ($perfile_attrs && !$oldprefs(perfile_attrs))} {
10210 # treediffs elements are limited by path;
10211 # won't have encodings cached if perfile_attrs was just turned on
10212 catch {unset treediffs}
10213 }
10214 if {$fontchanged || $maxwidth != $oldprefs(maxwidth)
10215 || $maxgraphpct != $oldprefs(maxgraphpct)} {
10216 redisplay
10217 } elseif {$showneartags != $oldprefs(showneartags) ||
10218 $limitdiffs != $oldprefs(limitdiffs)} {
10219 reselectline
10220 }
10221}
10222
10223proc formatdate {d} {
10224 global datetimeformat
10225 if {$d ne {}} {
10226 set d [clock format $d -format $datetimeformat]
10227 }
10228 return $d
10229}
10230
10231# This list of encoding names and aliases is distilled from
10232# http://www.iana.org/assignments/character-sets.
10233# Not all of them are supported by Tcl.
10234set encoding_aliases {
10235 { ANSI_X3.4-1968 iso-ir-6 ANSI_X3.4-1986 ISO_646.irv:1991 ASCII
10236 ISO646-US US-ASCII us IBM367 cp367 csASCII }
10237 { ISO-10646-UTF-1 csISO10646UTF1 }
10238 { ISO_646.basic:1983 ref csISO646basic1983 }
10239 { INVARIANT csINVARIANT }
10240 { ISO_646.irv:1983 iso-ir-2 irv csISO2IntlRefVersion }
10241 { BS_4730 iso-ir-4 ISO646-GB gb uk csISO4UnitedKingdom }
10242 { NATS-SEFI iso-ir-8-1 csNATSSEFI }
10243 { NATS-SEFI-ADD iso-ir-8-2 csNATSSEFIADD }
10244 { NATS-DANO iso-ir-9-1 csNATSDANO }
10245 { NATS-DANO-ADD iso-ir-9-2 csNATSDANOADD }
10246 { SEN_850200_B iso-ir-10 FI ISO646-FI ISO646-SE se csISO10Swedish }
10247 { SEN_850200_C iso-ir-11 ISO646-SE2 se2 csISO11SwedishForNames }
10248 { KS_C_5601-1987 iso-ir-149 KS_C_5601-1989 KSC_5601 korean csKSC56011987 }
10249 { ISO-2022-KR csISO2022KR }
10250 { EUC-KR csEUCKR }
10251 { ISO-2022-JP csISO2022JP }
10252 { ISO-2022-JP-2 csISO2022JP2 }
10253 { JIS_C6220-1969-jp JIS_C6220-1969 iso-ir-13 katakana x0201-7
10254 csISO13JISC6220jp }
10255 { JIS_C6220-1969-ro iso-ir-14 jp ISO646-JP csISO14JISC6220ro }
10256 { IT iso-ir-15 ISO646-IT csISO15Italian }
10257 { PT iso-ir-16 ISO646-PT csISO16Portuguese }
10258 { ES iso-ir-17 ISO646-ES csISO17Spanish }
10259 { greek7-old iso-ir-18 csISO18Greek7Old }
10260 { latin-greek iso-ir-19 csISO19LatinGreek }
10261 { DIN_66003 iso-ir-21 de ISO646-DE csISO21German }
10262 { NF_Z_62-010_(1973) iso-ir-25 ISO646-FR1 csISO25French }
10263 { Latin-greek-1 iso-ir-27 csISO27LatinGreek1 }
10264 { ISO_5427 iso-ir-37 csISO5427Cyrillic }
10265 { JIS_C6226-1978 iso-ir-42 csISO42JISC62261978 }
10266 { BS_viewdata iso-ir-47 csISO47BSViewdata }
10267 { INIS iso-ir-49 csISO49INIS }
10268 { INIS-8 iso-ir-50 csISO50INIS8 }
10269 { INIS-cyrillic iso-ir-51 csISO51INISCyrillic }
10270 { ISO_5427:1981 iso-ir-54 ISO5427Cyrillic1981 }
10271 { ISO_5428:1980 iso-ir-55 csISO5428Greek }
10272 { GB_1988-80 iso-ir-57 cn ISO646-CN csISO57GB1988 }
10273 { GB_2312-80 iso-ir-58 chinese csISO58GB231280 }
10274 { NS_4551-1 iso-ir-60 ISO646-NO no csISO60DanishNorwegian
10275 csISO60Norwegian1 }
10276 { NS_4551-2 ISO646-NO2 iso-ir-61 no2 csISO61Norwegian2 }
10277 { NF_Z_62-010 iso-ir-69 ISO646-FR fr csISO69French }
10278 { videotex-suppl iso-ir-70 csISO70VideotexSupp1 }
10279 { PT2 iso-ir-84 ISO646-PT2 csISO84Portuguese2 }
10280 { ES2 iso-ir-85 ISO646-ES2 csISO85Spanish2 }
10281 { MSZ_7795.3 iso-ir-86 ISO646-HU hu csISO86Hungarian }
10282 { JIS_C6226-1983 iso-ir-87 x0208 JIS_X0208-1983 csISO87JISX0208 }
10283 { greek7 iso-ir-88 csISO88Greek7 }
10284 { ASMO_449 ISO_9036 arabic7 iso-ir-89 csISO89ASMO449 }
10285 { iso-ir-90 csISO90 }
10286 { JIS_C6229-1984-a iso-ir-91 jp-ocr-a csISO91JISC62291984a }
10287 { JIS_C6229-1984-b iso-ir-92 ISO646-JP-OCR-B jp-ocr-b
10288 csISO92JISC62991984b }
10289 { JIS_C6229-1984-b-add iso-ir-93 jp-ocr-b-add csISO93JIS62291984badd }
10290 { JIS_C6229-1984-hand iso-ir-94 jp-ocr-hand csISO94JIS62291984hand }
10291 { JIS_C6229-1984-hand-add iso-ir-95 jp-ocr-hand-add
10292 csISO95JIS62291984handadd }
10293 { JIS_C6229-1984-kana iso-ir-96 csISO96JISC62291984kana }
10294 { ISO_2033-1983 iso-ir-98 e13b csISO2033 }
10295 { ANSI_X3.110-1983 iso-ir-99 CSA_T500-1983 NAPLPS csISO99NAPLPS }
10296 { ISO_8859-1:1987 iso-ir-100 ISO_8859-1 ISO-8859-1 latin1 l1 IBM819
10297 CP819 csISOLatin1 }
10298 { ISO_8859-2:1987 iso-ir-101 ISO_8859-2 ISO-8859-2 latin2 l2 csISOLatin2 }
10299 { T.61-7bit iso-ir-102 csISO102T617bit }
10300 { T.61-8bit T.61 iso-ir-103 csISO103T618bit }
10301 { ISO_8859-3:1988 iso-ir-109 ISO_8859-3 ISO-8859-3 latin3 l3 csISOLatin3 }
10302 { ISO_8859-4:1988 iso-ir-110 ISO_8859-4 ISO-8859-4 latin4 l4 csISOLatin4 }
10303 { ECMA-cyrillic iso-ir-111 KOI8-E csISO111ECMACyrillic }
10304 { CSA_Z243.4-1985-1 iso-ir-121 ISO646-CA csa7-1 ca csISO121Canadian1 }
10305 { CSA_Z243.4-1985-2 iso-ir-122 ISO646-CA2 csa7-2 csISO122Canadian2 }
10306 { CSA_Z243.4-1985-gr iso-ir-123 csISO123CSAZ24341985gr }
10307 { ISO_8859-6:1987 iso-ir-127 ISO_8859-6 ISO-8859-6 ECMA-114 ASMO-708
10308 arabic csISOLatinArabic }
10309 { ISO_8859-6-E csISO88596E ISO-8859-6-E }
10310 { ISO_8859-6-I csISO88596I ISO-8859-6-I }
10311 { ISO_8859-7:1987 iso-ir-126 ISO_8859-7 ISO-8859-7 ELOT_928 ECMA-118
10312 greek greek8 csISOLatinGreek }
10313 { T.101-G2 iso-ir-128 csISO128T101G2 }
10314 { ISO_8859-8:1988 iso-ir-138 ISO_8859-8 ISO-8859-8 hebrew
10315 csISOLatinHebrew }
10316 { ISO_8859-8-E csISO88598E ISO-8859-8-E }
10317 { ISO_8859-8-I csISO88598I ISO-8859-8-I }
10318 { CSN_369103 iso-ir-139 csISO139CSN369103 }
10319 { JUS_I.B1.002 iso-ir-141 ISO646-YU js yu csISO141JUSIB1002 }
10320 { ISO_6937-2-add iso-ir-142 csISOTextComm }
10321 { IEC_P27-1 iso-ir-143 csISO143IECP271 }
10322 { ISO_8859-5:1988 iso-ir-144 ISO_8859-5 ISO-8859-5 cyrillic
10323 csISOLatinCyrillic }
10324 { JUS_I.B1.003-serb iso-ir-146 serbian csISO146Serbian }
10325 { JUS_I.B1.003-mac macedonian iso-ir-147 csISO147Macedonian }
10326 { ISO_8859-9:1989 iso-ir-148 ISO_8859-9 ISO-8859-9 latin5 l5 csISOLatin5 }
10327 { greek-ccitt iso-ir-150 csISO150 csISO150GreekCCITT }
10328 { NC_NC00-10:81 cuba iso-ir-151 ISO646-CU csISO151Cuba }
10329 { ISO_6937-2-25 iso-ir-152 csISO6937Add }
10330 { GOST_19768-74 ST_SEV_358-88 iso-ir-153 csISO153GOST1976874 }
10331 { ISO_8859-supp iso-ir-154 latin1-2-5 csISO8859Supp }
10332 { ISO_10367-box iso-ir-155 csISO10367Box }
10333 { ISO-8859-10 iso-ir-157 l6 ISO_8859-10:1992 csISOLatin6 latin6 }
10334 { latin-lap lap iso-ir-158 csISO158Lap }
10335 { JIS_X0212-1990 x0212 iso-ir-159 csISO159JISX02121990 }
10336 { DS_2089 DS2089 ISO646-DK dk csISO646Danish }
10337 { us-dk csUSDK }
10338 { dk-us csDKUS }
10339 { JIS_X0201 X0201 csHalfWidthKatakana }
10340 { KSC5636 ISO646-KR csKSC5636 }
10341 { ISO-10646-UCS-2 csUnicode }
10342 { ISO-10646-UCS-4 csUCS4 }
10343 { DEC-MCS dec csDECMCS }
10344 { hp-roman8 roman8 r8 csHPRoman8 }
10345 { macintosh mac csMacintosh }
10346 { IBM037 cp037 ebcdic-cp-us ebcdic-cp-ca ebcdic-cp-wt ebcdic-cp-nl
10347 csIBM037 }
10348 { IBM038 EBCDIC-INT cp038 csIBM038 }
10349 { IBM273 CP273 csIBM273 }
10350 { IBM274 EBCDIC-BE CP274 csIBM274 }
10351 { IBM275 EBCDIC-BR cp275 csIBM275 }
10352 { IBM277 EBCDIC-CP-DK EBCDIC-CP-NO csIBM277 }
10353 { IBM278 CP278 ebcdic-cp-fi ebcdic-cp-se csIBM278 }
10354 { IBM280 CP280 ebcdic-cp-it csIBM280 }
10355 { IBM281 EBCDIC-JP-E cp281 csIBM281 }
10356 { IBM284 CP284 ebcdic-cp-es csIBM284 }
10357 { IBM285 CP285 ebcdic-cp-gb csIBM285 }
10358 { IBM290 cp290 EBCDIC-JP-kana csIBM290 }
10359 { IBM297 cp297 ebcdic-cp-fr csIBM297 }
10360 { IBM420 cp420 ebcdic-cp-ar1 csIBM420 }
10361 { IBM423 cp423 ebcdic-cp-gr csIBM423 }
10362 { IBM424 cp424 ebcdic-cp-he csIBM424 }
10363 { IBM437 cp437 437 csPC8CodePage437 }
10364 { IBM500 CP500 ebcdic-cp-be ebcdic-cp-ch csIBM500 }
10365 { IBM775 cp775 csPC775Baltic }
10366 { IBM850 cp850 850 csPC850Multilingual }
10367 { IBM851 cp851 851 csIBM851 }
10368 { IBM852 cp852 852 csPCp852 }
10369 { IBM855 cp855 855 csIBM855 }
10370 { IBM857 cp857 857 csIBM857 }
10371 { IBM860 cp860 860 csIBM860 }
10372 { IBM861 cp861 861 cp-is csIBM861 }
10373 { IBM862 cp862 862 csPC862LatinHebrew }
10374 { IBM863 cp863 863 csIBM863 }
10375 { IBM864 cp864 csIBM864 }
10376 { IBM865 cp865 865 csIBM865 }
10377 { IBM866 cp866 866 csIBM866 }
10378 { IBM868 CP868 cp-ar csIBM868 }
10379 { IBM869 cp869 869 cp-gr csIBM869 }
10380 { IBM870 CP870 ebcdic-cp-roece ebcdic-cp-yu csIBM870 }
10381 { IBM871 CP871 ebcdic-cp-is csIBM871 }
10382 { IBM880 cp880 EBCDIC-Cyrillic csIBM880 }
10383 { IBM891 cp891 csIBM891 }
10384 { IBM903 cp903 csIBM903 }
10385 { IBM904 cp904 904 csIBBM904 }
10386 { IBM905 CP905 ebcdic-cp-tr csIBM905 }
10387 { IBM918 CP918 ebcdic-cp-ar2 csIBM918 }
10388 { IBM1026 CP1026 csIBM1026 }
10389 { EBCDIC-AT-DE csIBMEBCDICATDE }
10390 { EBCDIC-AT-DE-A csEBCDICATDEA }
10391 { EBCDIC-CA-FR csEBCDICCAFR }
10392 { EBCDIC-DK-NO csEBCDICDKNO }
10393 { EBCDIC-DK-NO-A csEBCDICDKNOA }
10394 { EBCDIC-FI-SE csEBCDICFISE }
10395 { EBCDIC-FI-SE-A csEBCDICFISEA }
10396 { EBCDIC-FR csEBCDICFR }
10397 { EBCDIC-IT csEBCDICIT }
10398 { EBCDIC-PT csEBCDICPT }
10399 { EBCDIC-ES csEBCDICES }
10400 { EBCDIC-ES-A csEBCDICESA }
10401 { EBCDIC-ES-S csEBCDICESS }
10402 { EBCDIC-UK csEBCDICUK }
10403 { EBCDIC-US csEBCDICUS }
10404 { UNKNOWN-8BIT csUnknown8BiT }
10405 { MNEMONIC csMnemonic }
10406 { MNEM csMnem }
10407 { VISCII csVISCII }
10408 { VIQR csVIQR }
10409 { KOI8-R csKOI8R }
10410 { IBM00858 CCSID00858 CP00858 PC-Multilingual-850+euro }
10411 { IBM00924 CCSID00924 CP00924 ebcdic-Latin9--euro }
10412 { IBM01140 CCSID01140 CP01140 ebcdic-us-37+euro }
10413 { IBM01141 CCSID01141 CP01141 ebcdic-de-273+euro }
10414 { IBM01142 CCSID01142 CP01142 ebcdic-dk-277+euro ebcdic-no-277+euro }
10415 { IBM01143 CCSID01143 CP01143 ebcdic-fi-278+euro ebcdic-se-278+euro }
10416 { IBM01144 CCSID01144 CP01144 ebcdic-it-280+euro }
10417 { IBM01145 CCSID01145 CP01145 ebcdic-es-284+euro }
10418 { IBM01146 CCSID01146 CP01146 ebcdic-gb-285+euro }
10419 { IBM01147 CCSID01147 CP01147 ebcdic-fr-297+euro }
10420 { IBM01148 CCSID01148 CP01148 ebcdic-international-500+euro }
10421 { IBM01149 CCSID01149 CP01149 ebcdic-is-871+euro }
10422 { IBM1047 IBM-1047 }
10423 { PTCP154 csPTCP154 PT154 CP154 Cyrillic-Asian }
10424 { Amiga-1251 Ami1251 Amiga1251 Ami-1251 }
10425 { UNICODE-1-1 csUnicode11 }
10426 { CESU-8 csCESU-8 }
10427 { BOCU-1 csBOCU-1 }
10428 { UNICODE-1-1-UTF-7 csUnicode11UTF7 }
10429 { ISO-8859-14 iso-ir-199 ISO_8859-14:1998 ISO_8859-14 latin8 iso-celtic
10430 l8 }
10431 { ISO-8859-15 ISO_8859-15 Latin-9 }
10432 { ISO-8859-16 iso-ir-226 ISO_8859-16:2001 ISO_8859-16 latin10 l10 }
10433 { GBK CP936 MS936 windows-936 }
10434 { JIS_Encoding csJISEncoding }
10435 { Shift_JIS MS_Kanji csShiftJIS ShiftJIS Shift-JIS }
10436 { Extended_UNIX_Code_Packed_Format_for_Japanese csEUCPkdFmtJapanese
10437 EUC-JP }
10438 { Extended_UNIX_Code_Fixed_Width_for_Japanese csEUCFixWidJapanese }
10439 { ISO-10646-UCS-Basic csUnicodeASCII }
10440 { ISO-10646-Unicode-Latin1 csUnicodeLatin1 ISO-10646 }
10441 { ISO-Unicode-IBM-1261 csUnicodeIBM1261 }
10442 { ISO-Unicode-IBM-1268 csUnicodeIBM1268 }
10443 { ISO-Unicode-IBM-1276 csUnicodeIBM1276 }
10444 { ISO-Unicode-IBM-1264 csUnicodeIBM1264 }
10445 { ISO-Unicode-IBM-1265 csUnicodeIBM1265 }
10446 { ISO-8859-1-Windows-3.0-Latin-1 csWindows30Latin1 }
10447 { ISO-8859-1-Windows-3.1-Latin-1 csWindows31Latin1 }
10448 { ISO-8859-2-Windows-Latin-2 csWindows31Latin2 }
10449 { ISO-8859-9-Windows-Latin-5 csWindows31Latin5 }
10450 { Adobe-Standard-Encoding csAdobeStandardEncoding }
10451 { Ventura-US csVenturaUS }
10452 { Ventura-International csVenturaInternational }
10453 { PC8-Danish-Norwegian csPC8DanishNorwegian }
10454 { PC8-Turkish csPC8Turkish }
10455 { IBM-Symbols csIBMSymbols }
10456 { IBM-Thai csIBMThai }
10457 { HP-Legal csHPLegal }
10458 { HP-Pi-font csHPPiFont }
10459 { HP-Math8 csHPMath8 }
10460 { Adobe-Symbol-Encoding csHPPSMath }
10461 { HP-DeskTop csHPDesktop }
10462 { Ventura-Math csVenturaMath }
10463 { Microsoft-Publishing csMicrosoftPublishing }
10464 { Windows-31J csWindows31J }
10465 { GB2312 csGB2312 }
10466 { Big5 csBig5 }
10467}
10468
10469proc tcl_encoding {enc} {
10470 global encoding_aliases tcl_encoding_cache
10471 if {[info exists tcl_encoding_cache($enc)]} {
10472 return $tcl_encoding_cache($enc)
10473 }
10474 set names [encoding names]
10475 set lcnames [string tolower $names]
10476 set enc [string tolower $enc]
10477 set i [lsearch -exact $lcnames $enc]
10478 if {$i < 0} {
10479 # look for "isonnn" instead of "iso-nnn" or "iso_nnn"
10480 if {[regsub {^(iso|cp|ibm|jis)[-_]} $enc {\1} encx]} {
10481 set i [lsearch -exact $lcnames $encx]
10482 }
10483 }
10484 if {$i < 0} {
10485 foreach l $encoding_aliases {
10486 set ll [string tolower $l]
10487 if {[lsearch -exact $ll $enc] < 0} continue
10488 # look through the aliases for one that tcl knows about
10489 foreach e $ll {
10490 set i [lsearch -exact $lcnames $e]
10491 if {$i < 0} {
10492 if {[regsub {^(iso|cp|ibm|jis)[-_]} $e {\1} ex]} {
10493 set i [lsearch -exact $lcnames $ex]
10494 }
10495 }
10496 if {$i >= 0} break
10497 }
10498 break
10499 }
10500 }
10501 set tclenc {}
10502 if {$i >= 0} {
10503 set tclenc [lindex $names $i]
10504 }
10505 set tcl_encoding_cache($enc) $tclenc
10506 return $tclenc
10507}
10508
10509proc gitattr {path attr default} {
10510 global path_attr_cache
10511 if {[info exists path_attr_cache($attr,$path)]} {
10512 set r $path_attr_cache($attr,$path)
10513 } else {
10514 set r "unspecified"
10515 if {![catch {set line [exec git check-attr $attr -- $path]}]} {
10516 regexp "(.*): encoding: (.*)" $line m f r
10517 }
10518 set path_attr_cache($attr,$path) $r
10519 }
10520 if {$r eq "unspecified"} {
10521 return $default
10522 }
10523 return $r
10524}
10525
10526proc cache_gitattr {attr pathlist} {
10527 global path_attr_cache
10528 set newlist {}
10529 foreach path $pathlist {
10530 if {![info exists path_attr_cache($attr,$path)]} {
10531 lappend newlist $path
10532 }
10533 }
10534 set lim 1000
10535 if {[tk windowingsystem] == "win32"} {
10536 # windows has a 32k limit on the arguments to a command...
10537 set lim 30
10538 }
10539 while {$newlist ne {}} {
10540 set head [lrange $newlist 0 [expr {$lim - 1}]]
10541 set newlist [lrange $newlist $lim end]
10542 if {![catch {set rlist [eval exec git check-attr $attr -- $head]}]} {
10543 foreach row [split $rlist "\n"] {
10544 if {[regexp "(.*): encoding: (.*)" $row m path value]} {
10545 if {[string index $path 0] eq "\""} {
10546 set path [encoding convertfrom [lindex $path 0]]
10547 }
10548 set path_attr_cache($attr,$path) $value
10549 }
10550 }
10551 }
10552 }
10553}
10554
10555proc get_path_encoding {path} {
10556 global gui_encoding perfile_attrs
10557 set tcl_enc $gui_encoding
10558 if {$path ne {} && $perfile_attrs} {
10559 set enc2 [tcl_encoding [gitattr $path encoding $tcl_enc]]
10560 if {$enc2 ne {}} {
10561 set tcl_enc $enc2
10562 }
10563 }
10564 return $tcl_enc
10565}
10566
10567# First check that Tcl/Tk is recent enough
10568if {[catch {package require Tk 8.4} err]} {
10569 show_error {} . [mc "Sorry, gitk cannot run with this version of Tcl/Tk.\n\
10570 Gitk requires at least Tcl/Tk 8.4."]
10571 exit 1
10572}
10573
10574# defaults...
10575set wrcomcmd "git diff-tree --stdin -p --pretty"
10576
10577set gitencoding {}
10578catch {
10579 set gitencoding [exec git config --get i18n.commitencoding]
10580}
10581catch {
10582 set gitencoding [exec git config --get i18n.logoutputencoding]
10583}
10584if {$gitencoding == ""} {
10585 set gitencoding "utf-8"
10586}
10587set tclencoding [tcl_encoding $gitencoding]
10588if {$tclencoding == {}} {
10589 puts stderr "Warning: encoding $gitencoding is not supported by Tcl/Tk"
10590}
10591
10592set gui_encoding [encoding system]
10593catch {
10594 set enc [exec git config --get gui.encoding]
10595 if {$enc ne {}} {
10596 set tclenc [tcl_encoding $enc]
10597 if {$tclenc ne {}} {
10598 set gui_encoding $tclenc
10599 } else {
10600 puts stderr "Warning: encoding $enc is not supported by Tcl/Tk"
10601 }
10602 }
10603}
10604
10605set mainfont {Helvetica 9}
10606set textfont {Courier 9}
10607set uifont {Helvetica 9 bold}
10608set tabstop 8
10609set findmergefiles 0
10610set maxgraphpct 50
10611set maxwidth 16
10612set revlistorder 0
10613set fastdate 0
10614set uparrowlen 5
10615set downarrowlen 5
10616set mingaplen 100
10617set cmitmode "patch"
10618set wrapcomment "none"
10619set showneartags 1
10620set maxrefs 20
10621set maxlinelen 200
10622set showlocalchanges 1
10623set limitdiffs 1
10624set datetimeformat "%Y-%m-%d %H:%M:%S"
10625set autoselect 1
10626set perfile_attrs 0
10627
10628set extdifftool "meld"
10629
10630set colors {green red blue magenta darkgrey brown orange}
10631set bgcolor white
10632set fgcolor black
10633set diffcolors {red "#00a000" blue}
10634set diffcontext 3
10635set ignorespace 0
10636set selectbgcolor gray85
10637set markbgcolor "#e0e0ff"
10638
10639set circlecolors {white blue gray blue blue}
10640
10641# button for popping up context menus
10642if {[tk windowingsystem] eq "aqua"} {
10643 set ctxbut <Button-2>
10644} else {
10645 set ctxbut <Button-3>
10646}
10647
10648## For msgcat loading, first locate the installation location.
10649if { [info exists ::env(GITK_MSGSDIR)] } {
10650 ## Msgsdir was manually set in the environment.
10651 set gitk_msgsdir $::env(GITK_MSGSDIR)
10652} else {
10653 ## Let's guess the prefix from argv0.
10654 set gitk_prefix [file dirname [file dirname [file normalize $argv0]]]
10655 set gitk_libdir [file join $gitk_prefix share gitk lib]
10656 set gitk_msgsdir [file join $gitk_libdir msgs]
10657 unset gitk_prefix
10658}
10659
10660## Internationalization (i18n) through msgcat and gettext. See
10661## http://www.gnu.org/software/gettext/manual/html_node/Tcl.html
10662package require msgcat
10663namespace import ::msgcat::mc
10664## And eventually load the actual message catalog
10665::msgcat::mcload $gitk_msgsdir
10666
10667catch {source ~/.gitk}
10668
10669font create optionfont -family sans-serif -size -12
10670
10671parsefont mainfont $mainfont
10672eval font create mainfont [fontflags mainfont]
10673eval font create mainfontbold [fontflags mainfont 1]
10674
10675parsefont textfont $textfont
10676eval font create textfont [fontflags textfont]
10677eval font create textfontbold [fontflags textfont 1]
10678
10679parsefont uifont $uifont
10680eval font create uifont [fontflags uifont]
10681
10682setoptions
10683
10684# check that we can find a .git directory somewhere...
10685if {[catch {set gitdir [gitdir]}]} {
10686 show_error {} . [mc "Cannot find a git repository here."]
10687 exit 1
10688}
10689if {![file isdirectory $gitdir]} {
10690 show_error {} . [mc "Cannot find the git directory \"%s\"." $gitdir]
10691 exit 1
10692}
10693
10694set selecthead {}
10695set selectheadid {}
10696
10697set revtreeargs {}
10698set cmdline_files {}
10699set i 0
10700set revtreeargscmd {}
10701foreach arg $argv {
10702 switch -glob -- $arg {
10703 "" { }
10704 "--" {
10705 set cmdline_files [lrange $argv [expr {$i + 1}] end]
10706 break
10707 }
10708 "--select-commit=*" {
10709 set selecthead [string range $arg 16 end]
10710 }
10711 "--argscmd=*" {
10712 set revtreeargscmd [string range $arg 10 end]
10713 }
10714 default {
10715 lappend revtreeargs $arg
10716 }
10717 }
10718 incr i
10719}
10720
10721if {$selecthead eq "HEAD"} {
10722 set selecthead {}
10723}
10724
10725if {$i >= [llength $argv] && $revtreeargs ne {}} {
10726 # no -- on command line, but some arguments (other than --argscmd)
10727 if {[catch {
10728 set f [eval exec git rev-parse --no-revs --no-flags $revtreeargs]
10729 set cmdline_files [split $f "\n"]
10730 set n [llength $cmdline_files]
10731 set revtreeargs [lrange $revtreeargs 0 end-$n]
10732 # Unfortunately git rev-parse doesn't produce an error when
10733 # something is both a revision and a filename. To be consistent
10734 # with git log and git rev-list, check revtreeargs for filenames.
10735 foreach arg $revtreeargs {
10736 if {[file exists $arg]} {
10737 show_error {} . [mc "Ambiguous argument '%s': both revision\
10738 and filename" $arg]
10739 exit 1
10740 }
10741 }
10742 } err]} {
10743 # unfortunately we get both stdout and stderr in $err,
10744 # so look for "fatal:".
10745 set i [string first "fatal:" $err]
10746 if {$i > 0} {
10747 set err [string range $err [expr {$i + 6}] end]
10748 }
10749 show_error {} . "[mc "Bad arguments to gitk:"]\n$err"
10750 exit 1
10751 }
10752}
10753
10754set nullid "0000000000000000000000000000000000000000"
10755set nullid2 "0000000000000000000000000000000000000001"
10756set nullfile "/dev/null"
10757
10758set have_tk85 [expr {[package vcompare $tk_version "8.5"] >= 0}]
10759
10760set runq {}
10761set history {}
10762set historyindex 0
10763set fh_serial 0
10764set nhl_names {}
10765set highlight_paths {}
10766set findpattern {}
10767set searchdirn -forwards
10768set boldids {}
10769set boldnameids {}
10770set diffelide {0 0}
10771set markingmatches 0
10772set linkentercount 0
10773set need_redisplay 0
10774set nrows_drawn 0
10775set firsttabstop 0
10776
10777set nextviewnum 1
10778set curview 0
10779set selectedview 0
10780set selectedhlview [mc "None"]
10781set highlight_related [mc "None"]
10782set highlight_files {}
10783set viewfiles(0) {}
10784set viewperm(0) 0
10785set viewargs(0) {}
10786set viewargscmd(0) {}
10787
10788set selectedline {}
10789set numcommits 0
10790set loginstance 0
10791set cmdlineok 0
10792set stopped 0
10793set stuffsaved 0
10794set patchnum 0
10795set lserial 0
10796set isworktree [expr {[exec git rev-parse --is-inside-work-tree] == "true"}]
10797setcoords
10798makewindow
10799# wait for the window to become visible
10800tkwait visibility .
10801wm title . "[file tail $argv0]: [file tail [pwd]]"
10802readrefs
10803
10804if {$cmdline_files ne {} || $revtreeargs ne {} || $revtreeargscmd ne {}} {
10805 # create a view for the files/dirs specified on the command line
10806 set curview 1
10807 set selectedview 1
10808 set nextviewnum 2
10809 set viewname(1) [mc "Command line"]
10810 set viewfiles(1) $cmdline_files
10811 set viewargs(1) $revtreeargs
10812 set viewargscmd(1) $revtreeargscmd
10813 set viewperm(1) 0
10814 set vdatemode(1) 0
10815 addviewmenu 1
10816 .bar.view entryconf [mca "Edit view..."] -state normal
10817 .bar.view entryconf [mca "Delete view"] -state normal
10818}
10819
10820if {[info exists permviews]} {
10821 foreach v $permviews {
10822 set n $nextviewnum
10823 incr nextviewnum
10824 set viewname($n) [lindex $v 0]
10825 set viewfiles($n) [lindex $v 1]
10826 set viewargs($n) [lindex $v 2]
10827 set viewargscmd($n) [lindex $v 3]
10828 set viewperm($n) 1
10829 addviewmenu $n
10830 }
10831}
10832getcommits {}