769c79ab07520d5bcdad9493ab383553d2031988
1#!/bin/sh
2# Tcl ignores the next line -*- tcl -*- \
3exec wish "$0" -- "$@"
4
5# Copyright (C) 2005-2006 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
26
27 set script $args
28 if {[info exists isonrunq($script)]} return
29 if {$runq eq {}} {
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
42
43 fileevent $fd readable {}
44 if {$runq eq {}} {
45 after idle dorunq
46 }
47 lappend runq [list $fd $script]
48}
49
50proc dorunq {} {
51 global isonrunq runq
52
53 set tstart [clock clicks -milliseconds]
54 set t0 $tstart
55 while {$runq ne {}} {
56 set fd [lindex $runq 0 0]
57 set script [lindex $runq 0 1]
58 set repeat [eval $script]
59 set t1 [clock clicks -milliseconds]
60 set t [expr {$t1 - $t0}]
61 set runq [lrange $runq 1 end]
62 if {$repeat ne {} && $repeat} {
63 if {$fd eq {} || $repeat == 2} {
64 # script returns 1 if it wants to be readded
65 # file readers return 2 if they could do more straight away
66 lappend runq [list $fd $script]
67 } else {
68 fileevent $fd readable [list filereadable $fd $script]
69 }
70 } elseif {$fd eq {}} {
71 unset isonrunq($script)
72 }
73 set t0 $t1
74 if {$t1 - $tstart >= 80} break
75 }
76 if {$runq ne {}} {
77 after idle dorunq
78 }
79}
80
81# Start off a git rev-list process and arrange to read its output
82proc start_rev_list {view} {
83 global startmsecs
84 global commfd leftover tclencoding datemode
85 global viewargs viewfiles commitidx
86 global lookingforhead showlocalchanges
87
88 set startmsecs [clock clicks -milliseconds]
89 set commitidx($view) 0
90 set order "--topo-order"
91 if {$datemode} {
92 set order "--date-order"
93 }
94 if {[catch {
95 set fd [open [concat | git log -z --pretty=raw $order --parents \
96 --boundary $viewargs($view) "--" $viewfiles($view)] r]
97 } err]} {
98 error_popup "Error executing git rev-list: $err"
99 exit 1
100 }
101 set commfd($view) $fd
102 set leftover($view) {}
103 set lookingforhead $showlocalchanges
104 fconfigure $fd -blocking 0 -translation lf -eofchar {}
105 if {$tclencoding != {}} {
106 fconfigure $fd -encoding $tclencoding
107 }
108 filerun $fd [list getcommitlines $fd $view]
109 nowbusy $view
110}
111
112proc stop_rev_list {} {
113 global commfd curview
114
115 if {![info exists commfd($curview)]} return
116 set fd $commfd($curview)
117 catch {
118 set pid [pid $fd]
119 exec kill $pid
120 }
121 catch {close $fd}
122 unset commfd($curview)
123}
124
125proc getcommits {} {
126 global phase canv mainfont curview
127
128 set phase getcommits
129 initlayout
130 start_rev_list $curview
131 show_status "Reading commits..."
132}
133
134proc getcommitlines {fd view} {
135 global commitlisted
136 global leftover commfd
137 global displayorder commitidx commitrow commitdata
138 global parentlist children curview hlview
139 global vparentlist vdisporder vcmitlisted
140
141 set stuff [read $fd 500000]
142 # git log doesn't terminate the last commit with a null...
143 if {$stuff == {} && $leftover($view) ne {} && [eof $fd]} {
144 set stuff "\0"
145 }
146 if {$stuff == {}} {
147 if {![eof $fd]} {
148 return 1
149 }
150 global viewname
151 unset commfd($view)
152 notbusy $view
153 # set it blocking so we wait for the process to terminate
154 fconfigure $fd -blocking 1
155 if {[catch {close $fd} err]} {
156 set fv {}
157 if {$view != $curview} {
158 set fv " for the \"$viewname($view)\" view"
159 }
160 if {[string range $err 0 4] == "usage"} {
161 set err "Gitk: error reading commits$fv:\
162 bad arguments to git rev-list."
163 if {$viewname($view) eq "Command line"} {
164 append err \
165 " (Note: arguments to gitk are passed to git rev-list\
166 to allow selection of commits to be displayed.)"
167 }
168 } else {
169 set err "Error reading commits$fv: $err"
170 }
171 error_popup $err
172 }
173 if {$view == $curview} {
174 run chewcommits $view
175 }
176 return 0
177 }
178 set start 0
179 set gotsome 0
180 while 1 {
181 set i [string first "\0" $stuff $start]
182 if {$i < 0} {
183 append leftover($view) [string range $stuff $start end]
184 break
185 }
186 if {$start == 0} {
187 set cmit $leftover($view)
188 append cmit [string range $stuff 0 [expr {$i - 1}]]
189 set leftover($view) {}
190 } else {
191 set cmit [string range $stuff $start [expr {$i - 1}]]
192 }
193 set start [expr {$i + 1}]
194 set j [string first "\n" $cmit]
195 set ok 0
196 set listed 1
197 if {$j >= 0 && [string match "commit *" $cmit]} {
198 set ids [string range $cmit 7 [expr {$j - 1}]]
199 if {[string match {[-<>]*} $ids]} {
200 switch -- [string index $ids 0] {
201 "-" {set listed 0}
202 "<" {set listed 2}
203 ">" {set listed 3}
204 }
205 set ids [string range $ids 1 end]
206 }
207 set ok 1
208 foreach id $ids {
209 if {[string length $id] != 40} {
210 set ok 0
211 break
212 }
213 }
214 }
215 if {!$ok} {
216 set shortcmit $cmit
217 if {[string length $shortcmit] > 80} {
218 set shortcmit "[string range $shortcmit 0 80]..."
219 }
220 error_popup "Can't parse git log output: {$shortcmit}"
221 exit 1
222 }
223 set id [lindex $ids 0]
224 if {$listed} {
225 set olds [lrange $ids 1 end]
226 set i 0
227 foreach p $olds {
228 if {$i == 0 || [lsearch -exact $olds $p] >= $i} {
229 lappend children($view,$p) $id
230 }
231 incr i
232 }
233 } else {
234 set olds {}
235 }
236 if {![info exists children($view,$id)]} {
237 set children($view,$id) {}
238 }
239 set commitdata($id) [string range $cmit [expr {$j + 1}] end]
240 set commitrow($view,$id) $commitidx($view)
241 incr commitidx($view)
242 if {$view == $curview} {
243 lappend parentlist $olds
244 lappend displayorder $id
245 lappend commitlisted $listed
246 } else {
247 lappend vparentlist($view) $olds
248 lappend vdisporder($view) $id
249 lappend vcmitlisted($view) $listed
250 }
251 set gotsome 1
252 }
253 if {$gotsome} {
254 run chewcommits $view
255 }
256 return 2
257}
258
259proc chewcommits {view} {
260 global curview hlview commfd
261 global selectedline pending_select
262
263 set more 0
264 if {$view == $curview} {
265 set allread [expr {![info exists commfd($view)]}]
266 set tlimit [expr {[clock clicks -milliseconds] + 50}]
267 set more [layoutmore $tlimit $allread]
268 if {$allread && !$more} {
269 global displayorder commitidx phase
270 global numcommits startmsecs
271
272 if {[info exists pending_select]} {
273 set row [first_real_row]
274 selectline $row 1
275 }
276 if {$commitidx($curview) > 0} {
277 #set ms [expr {[clock clicks -milliseconds] - $startmsecs}]
278 #puts "overall $ms ms for $numcommits commits"
279 } else {
280 show_status "No commits selected"
281 }
282 notbusy layout
283 set phase {}
284 }
285 }
286 if {[info exists hlview] && $view == $hlview} {
287 vhighlightmore
288 }
289 return $more
290}
291
292proc readcommit {id} {
293 if {[catch {set contents [exec git cat-file commit $id]}]} return
294 parsecommit $id $contents 0
295}
296
297proc updatecommits {} {
298 global viewdata curview phase displayorder
299 global children commitrow selectedline thickerline
300
301 if {$phase ne {}} {
302 stop_rev_list
303 set phase {}
304 }
305 set n $curview
306 foreach id $displayorder {
307 catch {unset children($n,$id)}
308 catch {unset commitrow($n,$id)}
309 }
310 set curview -1
311 catch {unset selectedline}
312 catch {unset thickerline}
313 catch {unset viewdata($n)}
314 readrefs
315 changedrefs
316 regetallcommits
317 showview $n
318}
319
320proc parsecommit {id contents listed} {
321 global commitinfo cdate
322
323 set inhdr 1
324 set comment {}
325 set headline {}
326 set auname {}
327 set audate {}
328 set comname {}
329 set comdate {}
330 set hdrend [string first "\n\n" $contents]
331 if {$hdrend < 0} {
332 # should never happen...
333 set hdrend [string length $contents]
334 }
335 set header [string range $contents 0 [expr {$hdrend - 1}]]
336 set comment [string range $contents [expr {$hdrend + 2}] end]
337 foreach line [split $header "\n"] {
338 set tag [lindex $line 0]
339 if {$tag == "author"} {
340 set audate [lindex $line end-1]
341 set auname [lrange $line 1 end-2]
342 } elseif {$tag == "committer"} {
343 set comdate [lindex $line end-1]
344 set comname [lrange $line 1 end-2]
345 }
346 }
347 set headline {}
348 # take the first non-blank line of the comment as the headline
349 set headline [string trimleft $comment]
350 set i [string first "\n" $headline]
351 if {$i >= 0} {
352 set headline [string range $headline 0 $i]
353 }
354 set headline [string trimright $headline]
355 set i [string first "\r" $headline]
356 if {$i >= 0} {
357 set headline [string trimright [string range $headline 0 $i]]
358 }
359 if {!$listed} {
360 # git rev-list indents the comment by 4 spaces;
361 # if we got this via git cat-file, add the indentation
362 set newcomment {}
363 foreach line [split $comment "\n"] {
364 append newcomment " "
365 append newcomment $line
366 append newcomment "\n"
367 }
368 set comment $newcomment
369 }
370 if {$comdate != {}} {
371 set cdate($id) $comdate
372 }
373 set commitinfo($id) [list $headline $auname $audate \
374 $comname $comdate $comment]
375}
376
377proc getcommit {id} {
378 global commitdata commitinfo
379
380 if {[info exists commitdata($id)]} {
381 parsecommit $id $commitdata($id) 1
382 } else {
383 readcommit $id
384 if {![info exists commitinfo($id)]} {
385 set commitinfo($id) {"No commit information available"}
386 }
387 }
388 return 1
389}
390
391proc readrefs {} {
392 global tagids idtags headids idheads tagobjid
393 global otherrefids idotherrefs mainhead mainheadid
394
395 foreach v {tagids idtags headids idheads otherrefids idotherrefs} {
396 catch {unset $v}
397 }
398 set refd [open [list | git show-ref -d] r]
399 while {[gets $refd line] >= 0} {
400 if {[string index $line 40] ne " "} continue
401 set id [string range $line 0 39]
402 set ref [string range $line 41 end]
403 if {![string match "refs/*" $ref]} continue
404 set name [string range $ref 5 end]
405 if {[string match "remotes/*" $name]} {
406 if {![string match "*/HEAD" $name]} {
407 set headids($name) $id
408 lappend idheads($id) $name
409 }
410 } elseif {[string match "heads/*" $name]} {
411 set name [string range $name 6 end]
412 set headids($name) $id
413 lappend idheads($id) $name
414 } elseif {[string match "tags/*" $name]} {
415 # this lets refs/tags/foo^{} overwrite refs/tags/foo,
416 # which is what we want since the former is the commit ID
417 set name [string range $name 5 end]
418 if {[string match "*^{}" $name]} {
419 set name [string range $name 0 end-3]
420 } else {
421 set tagobjid($name) $id
422 }
423 set tagids($name) $id
424 lappend idtags($id) $name
425 } else {
426 set otherrefids($name) $id
427 lappend idotherrefs($id) $name
428 }
429 }
430 catch {close $refd}
431 set mainhead {}
432 set mainheadid {}
433 catch {
434 set thehead [exec git symbolic-ref HEAD]
435 if {[string match "refs/heads/*" $thehead]} {
436 set mainhead [string range $thehead 11 end]
437 if {[info exists headids($mainhead)]} {
438 set mainheadid $headids($mainhead)
439 }
440 }
441 }
442}
443
444# skip over fake commits
445proc first_real_row {} {
446 global nullid nullid2 displayorder numcommits
447
448 for {set row 0} {$row < $numcommits} {incr row} {
449 set id [lindex $displayorder $row]
450 if {$id ne $nullid && $id ne $nullid2} {
451 break
452 }
453 }
454 return $row
455}
456
457# update things for a head moved to a child of its previous location
458proc movehead {id name} {
459 global headids idheads
460
461 removehead $headids($name) $name
462 set headids($name) $id
463 lappend idheads($id) $name
464}
465
466# update things when a head has been removed
467proc removehead {id name} {
468 global headids idheads
469
470 if {$idheads($id) eq $name} {
471 unset idheads($id)
472 } else {
473 set i [lsearch -exact $idheads($id) $name]
474 if {$i >= 0} {
475 set idheads($id) [lreplace $idheads($id) $i $i]
476 }
477 }
478 unset headids($name)
479}
480
481proc show_error {w top msg} {
482 message $w.m -text $msg -justify center -aspect 400
483 pack $w.m -side top -fill x -padx 20 -pady 20
484 button $w.ok -text OK -command "destroy $top"
485 pack $w.ok -side bottom -fill x
486 bind $top <Visibility> "grab $top; focus $top"
487 bind $top <Key-Return> "destroy $top"
488 tkwait window $top
489}
490
491proc error_popup msg {
492 set w .error
493 toplevel $w
494 wm transient $w .
495 show_error $w $w $msg
496}
497
498proc confirm_popup msg {
499 global confirm_ok
500 set confirm_ok 0
501 set w .confirm
502 toplevel $w
503 wm transient $w .
504 message $w.m -text $msg -justify center -aspect 400
505 pack $w.m -side top -fill x -padx 20 -pady 20
506 button $w.ok -text OK -command "set confirm_ok 1; destroy $w"
507 pack $w.ok -side left -fill x
508 button $w.cancel -text Cancel -command "destroy $w"
509 pack $w.cancel -side right -fill x
510 bind $w <Visibility> "grab $w; focus $w"
511 tkwait window $w
512 return $confirm_ok
513}
514
515proc makewindow {} {
516 global canv canv2 canv3 linespc charspc ctext cflist
517 global textfont mainfont uifont tabstop
518 global findtype findtypemenu findloc findstring fstring geometry
519 global entries sha1entry sha1string sha1but
520 global maincursor textcursor curtextcursor
521 global rowctxmenu fakerowmenu mergemax wrapcomment
522 global highlight_files gdttype
523 global searchstring sstring
524 global bgcolor fgcolor bglist fglist diffcolors selectbgcolor
525 global headctxmenu
526
527 menu .bar
528 .bar add cascade -label "File" -menu .bar.file
529 .bar configure -font $uifont
530 menu .bar.file
531 .bar.file add command -label "Update" -command updatecommits
532 .bar.file add command -label "Reread references" -command rereadrefs
533 .bar.file add command -label "Quit" -command doquit
534 .bar.file configure -font $uifont
535 menu .bar.edit
536 .bar add cascade -label "Edit" -menu .bar.edit
537 .bar.edit add command -label "Preferences" -command doprefs
538 .bar.edit configure -font $uifont
539
540 menu .bar.view -font $uifont
541 .bar add cascade -label "View" -menu .bar.view
542 .bar.view add command -label "New view..." -command {newview 0}
543 .bar.view add command -label "Edit view..." -command editview \
544 -state disabled
545 .bar.view add command -label "Delete view" -command delview -state disabled
546 .bar.view add separator
547 .bar.view add radiobutton -label "All files" -command {showview 0} \
548 -variable selectedview -value 0
549
550 menu .bar.help
551 .bar add cascade -label "Help" -menu .bar.help
552 .bar.help add command -label "About gitk" -command about
553 .bar.help add command -label "Key bindings" -command keys
554 .bar.help configure -font $uifont
555 . configure -menu .bar
556
557 # the gui has upper and lower half, parts of a paned window.
558 panedwindow .ctop -orient vertical
559
560 # possibly use assumed geometry
561 if {![info exists geometry(pwsash0)]} {
562 set geometry(topheight) [expr {15 * $linespc}]
563 set geometry(topwidth) [expr {80 * $charspc}]
564 set geometry(botheight) [expr {15 * $linespc}]
565 set geometry(botwidth) [expr {50 * $charspc}]
566 set geometry(pwsash0) "[expr {40 * $charspc}] 2"
567 set geometry(pwsash1) "[expr {60 * $charspc}] 2"
568 }
569
570 # the upper half will have a paned window, a scroll bar to the right, and some stuff below
571 frame .tf -height $geometry(topheight) -width $geometry(topwidth)
572 frame .tf.histframe
573 panedwindow .tf.histframe.pwclist -orient horizontal -sashpad 0 -handlesize 4
574
575 # create three canvases
576 set cscroll .tf.histframe.csb
577 set canv .tf.histframe.pwclist.canv
578 canvas $canv \
579 -selectbackground $selectbgcolor \
580 -background $bgcolor -bd 0 \
581 -yscrollincr $linespc -yscrollcommand "scrollcanv $cscroll"
582 .tf.histframe.pwclist add $canv
583 set canv2 .tf.histframe.pwclist.canv2
584 canvas $canv2 \
585 -selectbackground $selectbgcolor \
586 -background $bgcolor -bd 0 -yscrollincr $linespc
587 .tf.histframe.pwclist add $canv2
588 set canv3 .tf.histframe.pwclist.canv3
589 canvas $canv3 \
590 -selectbackground $selectbgcolor \
591 -background $bgcolor -bd 0 -yscrollincr $linespc
592 .tf.histframe.pwclist add $canv3
593 eval .tf.histframe.pwclist sash place 0 $geometry(pwsash0)
594 eval .tf.histframe.pwclist sash place 1 $geometry(pwsash1)
595
596 # a scroll bar to rule them
597 scrollbar $cscroll -command {allcanvs yview} -highlightthickness 0
598 pack $cscroll -side right -fill y
599 bind .tf.histframe.pwclist <Configure> {resizeclistpanes %W %w}
600 lappend bglist $canv $canv2 $canv3
601 pack .tf.histframe.pwclist -fill both -expand 1 -side left
602
603 # we have two button bars at bottom of top frame. Bar 1
604 frame .tf.bar
605 frame .tf.lbar -height 15
606
607 set sha1entry .tf.bar.sha1
608 set entries $sha1entry
609 set sha1but .tf.bar.sha1label
610 button $sha1but -text "SHA1 ID: " -state disabled -relief flat \
611 -command gotocommit -width 8 -font $uifont
612 $sha1but conf -disabledforeground [$sha1but cget -foreground]
613 pack .tf.bar.sha1label -side left
614 entry $sha1entry -width 40 -font $textfont -textvariable sha1string
615 trace add variable sha1string write sha1change
616 pack $sha1entry -side left -pady 2
617
618 image create bitmap bm-left -data {
619 #define left_width 16
620 #define left_height 16
621 static unsigned char left_bits[] = {
622 0x00, 0x00, 0xc0, 0x01, 0xe0, 0x00, 0x70, 0x00, 0x38, 0x00, 0x1c, 0x00,
623 0x0e, 0x00, 0xff, 0x7f, 0xff, 0x7f, 0xff, 0x7f, 0x0e, 0x00, 0x1c, 0x00,
624 0x38, 0x00, 0x70, 0x00, 0xe0, 0x00, 0xc0, 0x01};
625 }
626 image create bitmap bm-right -data {
627 #define right_width 16
628 #define right_height 16
629 static unsigned char right_bits[] = {
630 0x00, 0x00, 0xc0, 0x01, 0x80, 0x03, 0x00, 0x07, 0x00, 0x0e, 0x00, 0x1c,
631 0x00, 0x38, 0xff, 0x7f, 0xff, 0x7f, 0xff, 0x7f, 0x00, 0x38, 0x00, 0x1c,
632 0x00, 0x0e, 0x00, 0x07, 0x80, 0x03, 0xc0, 0x01};
633 }
634 button .tf.bar.leftbut -image bm-left -command goback \
635 -state disabled -width 26
636 pack .tf.bar.leftbut -side left -fill y
637 button .tf.bar.rightbut -image bm-right -command goforw \
638 -state disabled -width 26
639 pack .tf.bar.rightbut -side left -fill y
640
641 button .tf.bar.findbut -text "Find" -command dofind -font $uifont
642 pack .tf.bar.findbut -side left
643 set findstring {}
644 set fstring .tf.bar.findstring
645 lappend entries $fstring
646 entry $fstring -width 30 -font $textfont -textvariable findstring
647 trace add variable findstring write find_change
648 pack $fstring -side left -expand 1 -fill x -in .tf.bar
649 set findtype Exact
650 set findtypemenu [tk_optionMenu .tf.bar.findtype \
651 findtype Exact IgnCase Regexp]
652 trace add variable findtype write find_change
653 .tf.bar.findtype configure -font $uifont
654 .tf.bar.findtype.menu configure -font $uifont
655 set findloc "All fields"
656 tk_optionMenu .tf.bar.findloc findloc "All fields" Headline \
657 Comments Author Committer
658 trace add variable findloc write find_change
659 .tf.bar.findloc configure -font $uifont
660 .tf.bar.findloc.menu configure -font $uifont
661 pack .tf.bar.findloc -side right
662 pack .tf.bar.findtype -side right
663
664 # build up the bottom bar of upper window
665 label .tf.lbar.flabel -text "Highlight: Commits " \
666 -font $uifont
667 pack .tf.lbar.flabel -side left -fill y
668 set gdttype "touching paths:"
669 set gm [tk_optionMenu .tf.lbar.gdttype gdttype "touching paths:" \
670 "adding/removing string:"]
671 trace add variable gdttype write hfiles_change
672 $gm conf -font $uifont
673 .tf.lbar.gdttype conf -font $uifont
674 pack .tf.lbar.gdttype -side left -fill y
675 entry .tf.lbar.fent -width 25 -font $textfont \
676 -textvariable highlight_files
677 trace add variable highlight_files write hfiles_change
678 lappend entries .tf.lbar.fent
679 pack .tf.lbar.fent -side left -fill x -expand 1
680 label .tf.lbar.vlabel -text " OR in view" -font $uifont
681 pack .tf.lbar.vlabel -side left -fill y
682 global viewhlmenu selectedhlview
683 set viewhlmenu [tk_optionMenu .tf.lbar.vhl selectedhlview None]
684 $viewhlmenu entryconf None -command delvhighlight
685 $viewhlmenu conf -font $uifont
686 .tf.lbar.vhl conf -font $uifont
687 pack .tf.lbar.vhl -side left -fill y
688 label .tf.lbar.rlabel -text " OR " -font $uifont
689 pack .tf.lbar.rlabel -side left -fill y
690 global highlight_related
691 set m [tk_optionMenu .tf.lbar.relm highlight_related None \
692 "Descendent" "Not descendent" "Ancestor" "Not ancestor"]
693 $m conf -font $uifont
694 .tf.lbar.relm conf -font $uifont
695 trace add variable highlight_related write vrel_change
696 pack .tf.lbar.relm -side left -fill y
697
698 # Finish putting the upper half of the viewer together
699 pack .tf.lbar -in .tf -side bottom -fill x
700 pack .tf.bar -in .tf -side bottom -fill x
701 pack .tf.histframe -fill both -side top -expand 1
702 .ctop add .tf
703 .ctop paneconfigure .tf -height $geometry(topheight)
704 .ctop paneconfigure .tf -width $geometry(topwidth)
705
706 # now build up the bottom
707 panedwindow .pwbottom -orient horizontal
708
709 # lower left, a text box over search bar, scroll bar to the right
710 # if we know window height, then that will set the lower text height, otherwise
711 # we set lower text height which will drive window height
712 if {[info exists geometry(main)]} {
713 frame .bleft -width $geometry(botwidth)
714 } else {
715 frame .bleft -width $geometry(botwidth) -height $geometry(botheight)
716 }
717 frame .bleft.top
718 frame .bleft.mid
719
720 button .bleft.top.search -text "Search" -command dosearch \
721 -font $uifont
722 pack .bleft.top.search -side left -padx 5
723 set sstring .bleft.top.sstring
724 entry $sstring -width 20 -font $textfont -textvariable searchstring
725 lappend entries $sstring
726 trace add variable searchstring write incrsearch
727 pack $sstring -side left -expand 1 -fill x
728 radiobutton .bleft.mid.diff -text "Diff" \
729 -command changediffdisp -variable diffelide -value {0 0}
730 radiobutton .bleft.mid.old -text "Old version" \
731 -command changediffdisp -variable diffelide -value {0 1}
732 radiobutton .bleft.mid.new -text "New version" \
733 -command changediffdisp -variable diffelide -value {1 0}
734 pack .bleft.mid.diff .bleft.mid.old .bleft.mid.new -side left
735 set ctext .bleft.ctext
736 text $ctext -background $bgcolor -foreground $fgcolor \
737 -tabs "[expr {$tabstop * $charspc}]" \
738 -state disabled -font $textfont \
739 -yscrollcommand scrolltext -wrap none
740 scrollbar .bleft.sb -command "$ctext yview"
741 pack .bleft.top -side top -fill x
742 pack .bleft.mid -side top -fill x
743 pack .bleft.sb -side right -fill y
744 pack $ctext -side left -fill both -expand 1
745 lappend bglist $ctext
746 lappend fglist $ctext
747
748 $ctext tag conf comment -wrap $wrapcomment
749 $ctext tag conf filesep -font [concat $textfont bold] -back "#aaaaaa"
750 $ctext tag conf hunksep -fore [lindex $diffcolors 2]
751 $ctext tag conf d0 -fore [lindex $diffcolors 0]
752 $ctext tag conf d1 -fore [lindex $diffcolors 1]
753 $ctext tag conf m0 -fore red
754 $ctext tag conf m1 -fore blue
755 $ctext tag conf m2 -fore green
756 $ctext tag conf m3 -fore purple
757 $ctext tag conf m4 -fore brown
758 $ctext tag conf m5 -fore "#009090"
759 $ctext tag conf m6 -fore magenta
760 $ctext tag conf m7 -fore "#808000"
761 $ctext tag conf m8 -fore "#009000"
762 $ctext tag conf m9 -fore "#ff0080"
763 $ctext tag conf m10 -fore cyan
764 $ctext tag conf m11 -fore "#b07070"
765 $ctext tag conf m12 -fore "#70b0f0"
766 $ctext tag conf m13 -fore "#70f0b0"
767 $ctext tag conf m14 -fore "#f0b070"
768 $ctext tag conf m15 -fore "#ff70b0"
769 $ctext tag conf mmax -fore darkgrey
770 set mergemax 16
771 $ctext tag conf mresult -font [concat $textfont bold]
772 $ctext tag conf msep -font [concat $textfont bold]
773 $ctext tag conf found -back yellow
774
775 .pwbottom add .bleft
776 .pwbottom paneconfigure .bleft -width $geometry(botwidth)
777
778 # lower right
779 frame .bright
780 frame .bright.mode
781 radiobutton .bright.mode.patch -text "Patch" \
782 -command reselectline -variable cmitmode -value "patch"
783 .bright.mode.patch configure -font $uifont
784 radiobutton .bright.mode.tree -text "Tree" \
785 -command reselectline -variable cmitmode -value "tree"
786 .bright.mode.tree configure -font $uifont
787 grid .bright.mode.patch .bright.mode.tree -sticky ew
788 pack .bright.mode -side top -fill x
789 set cflist .bright.cfiles
790 set indent [font measure $mainfont "nn"]
791 text $cflist \
792 -selectbackground $selectbgcolor \
793 -background $bgcolor -foreground $fgcolor \
794 -font $mainfont \
795 -tabs [list $indent [expr {2 * $indent}]] \
796 -yscrollcommand ".bright.sb set" \
797 -cursor [. cget -cursor] \
798 -spacing1 1 -spacing3 1
799 lappend bglist $cflist
800 lappend fglist $cflist
801 scrollbar .bright.sb -command "$cflist yview"
802 pack .bright.sb -side right -fill y
803 pack $cflist -side left -fill both -expand 1
804 $cflist tag configure highlight \
805 -background [$cflist cget -selectbackground]
806 $cflist tag configure bold -font [concat $mainfont bold]
807
808 .pwbottom add .bright
809 .ctop add .pwbottom
810
811 # restore window position if known
812 if {[info exists geometry(main)]} {
813 wm geometry . "$geometry(main)"
814 }
815
816 if {[tk windowingsystem] eq {aqua}} {
817 set M1B M1
818 } else {
819 set M1B Control
820 }
821
822 bind .pwbottom <Configure> {resizecdetpanes %W %w}
823 pack .ctop -fill both -expand 1
824 bindall <1> {selcanvline %W %x %y}
825 #bindall <B1-Motion> {selcanvline %W %x %y}
826 if {[tk windowingsystem] == "win32"} {
827 bind . <MouseWheel> { windows_mousewheel_redirector %W %X %Y %D }
828 bind $ctext <MouseWheel> { windows_mousewheel_redirector %W %X %Y %D ; break }
829 } else {
830 bindall <ButtonRelease-4> "allcanvs yview scroll -5 units"
831 bindall <ButtonRelease-5> "allcanvs yview scroll 5 units"
832 }
833 bindall <2> "canvscan mark %W %x %y"
834 bindall <B2-Motion> "canvscan dragto %W %x %y"
835 bindkey <Home> selfirstline
836 bindkey <End> sellastline
837 bind . <Key-Up> "selnextline -1"
838 bind . <Key-Down> "selnextline 1"
839 bind . <Shift-Key-Up> "next_highlight -1"
840 bind . <Shift-Key-Down> "next_highlight 1"
841 bindkey <Key-Right> "goforw"
842 bindkey <Key-Left> "goback"
843 bind . <Key-Prior> "selnextpage -1"
844 bind . <Key-Next> "selnextpage 1"
845 bind . <$M1B-Home> "allcanvs yview moveto 0.0"
846 bind . <$M1B-End> "allcanvs yview moveto 1.0"
847 bind . <$M1B-Key-Up> "allcanvs yview scroll -1 units"
848 bind . <$M1B-Key-Down> "allcanvs yview scroll 1 units"
849 bind . <$M1B-Key-Prior> "allcanvs yview scroll -1 pages"
850 bind . <$M1B-Key-Next> "allcanvs yview scroll 1 pages"
851 bindkey <Key-Delete> "$ctext yview scroll -1 pages"
852 bindkey <Key-BackSpace> "$ctext yview scroll -1 pages"
853 bindkey <Key-space> "$ctext yview scroll 1 pages"
854 bindkey p "selnextline -1"
855 bindkey n "selnextline 1"
856 bindkey z "goback"
857 bindkey x "goforw"
858 bindkey i "selnextline -1"
859 bindkey k "selnextline 1"
860 bindkey j "goback"
861 bindkey l "goforw"
862 bindkey b "$ctext yview scroll -1 pages"
863 bindkey d "$ctext yview scroll 18 units"
864 bindkey u "$ctext yview scroll -18 units"
865 bindkey / {findnext 1}
866 bindkey <Key-Return> {findnext 0}
867 bindkey ? findprev
868 bindkey f nextfile
869 bindkey <F5> updatecommits
870 bind . <$M1B-q> doquit
871 bind . <$M1B-f> dofind
872 bind . <$M1B-g> {findnext 0}
873 bind . <$M1B-r> dosearchback
874 bind . <$M1B-s> dosearch
875 bind . <$M1B-equal> {incrfont 1}
876 bind . <$M1B-KP_Add> {incrfont 1}
877 bind . <$M1B-minus> {incrfont -1}
878 bind . <$M1B-KP_Subtract> {incrfont -1}
879 wm protocol . WM_DELETE_WINDOW doquit
880 bind . <Button-1> "click %W"
881 bind $fstring <Key-Return> dofind
882 bind $sha1entry <Key-Return> gotocommit
883 bind $sha1entry <<PasteSelection>> clearsha1
884 bind $cflist <1> {sel_flist %W %x %y; break}
885 bind $cflist <B1-Motion> {sel_flist %W %x %y; break}
886 bind $cflist <ButtonRelease-1> {treeclick %W %x %y}
887 bind $cflist <Button-3> {pop_flist_menu %W %X %Y %x %y}
888
889 set maincursor [. cget -cursor]
890 set textcursor [$ctext cget -cursor]
891 set curtextcursor $textcursor
892
893 set rowctxmenu .rowctxmenu
894 menu $rowctxmenu -tearoff 0
895 $rowctxmenu add command -label "Diff this -> selected" \
896 -command {diffvssel 0}
897 $rowctxmenu add command -label "Diff selected -> this" \
898 -command {diffvssel 1}
899 $rowctxmenu add command -label "Make patch" -command mkpatch
900 $rowctxmenu add command -label "Create tag" -command mktag
901 $rowctxmenu add command -label "Write commit to file" -command writecommit
902 $rowctxmenu add command -label "Create new branch" -command mkbranch
903 $rowctxmenu add command -label "Cherry-pick this commit" \
904 -command cherrypick
905 $rowctxmenu add command -label "Reset HEAD branch to here" \
906 -command resethead
907
908 set fakerowmenu .fakerowmenu
909 menu $fakerowmenu -tearoff 0
910 $fakerowmenu add command -label "Diff this -> selected" \
911 -command {diffvssel 0}
912 $fakerowmenu add command -label "Diff selected -> this" \
913 -command {diffvssel 1}
914 $fakerowmenu add command -label "Make patch" -command mkpatch
915# $fakerowmenu add command -label "Commit" -command {mkcommit 0}
916# $fakerowmenu add command -label "Commit all" -command {mkcommit 1}
917# $fakerowmenu add command -label "Revert local changes" -command revertlocal
918
919 set headctxmenu .headctxmenu
920 menu $headctxmenu -tearoff 0
921 $headctxmenu add command -label "Check out this branch" \
922 -command cobranch
923 $headctxmenu add command -label "Remove this branch" \
924 -command rmbranch
925
926 global flist_menu
927 set flist_menu .flistctxmenu
928 menu $flist_menu -tearoff 0
929 $flist_menu add command -label "Highlight this too" \
930 -command {flist_hl 0}
931 $flist_menu add command -label "Highlight this only" \
932 -command {flist_hl 1}
933}
934
935# Windows sends all mouse wheel events to the current focused window, not
936# the one where the mouse hovers, so bind those events here and redirect
937# to the correct window
938proc windows_mousewheel_redirector {W X Y D} {
939 global canv canv2 canv3
940 set w [winfo containing -displayof $W $X $Y]
941 if {$w ne ""} {
942 set u [expr {$D < 0 ? 5 : -5}]
943 if {$w == $canv || $w == $canv2 || $w == $canv3} {
944 allcanvs yview scroll $u units
945 } else {
946 catch {
947 $w yview scroll $u units
948 }
949 }
950 }
951}
952
953# mouse-2 makes all windows scan vertically, but only the one
954# the cursor is in scans horizontally
955proc canvscan {op w x y} {
956 global canv canv2 canv3
957 foreach c [list $canv $canv2 $canv3] {
958 if {$c == $w} {
959 $c scan $op $x $y
960 } else {
961 $c scan $op 0 $y
962 }
963 }
964}
965
966proc scrollcanv {cscroll f0 f1} {
967 $cscroll set $f0 $f1
968 drawfrac $f0 $f1
969 flushhighlights
970}
971
972# when we make a key binding for the toplevel, make sure
973# it doesn't get triggered when that key is pressed in the
974# find string entry widget.
975proc bindkey {ev script} {
976 global entries
977 bind . $ev $script
978 set escript [bind Entry $ev]
979 if {$escript == {}} {
980 set escript [bind Entry <Key>]
981 }
982 foreach e $entries {
983 bind $e $ev "$escript; break"
984 }
985}
986
987# set the focus back to the toplevel for any click outside
988# the entry widgets
989proc click {w} {
990 global ctext entries
991 foreach e [concat $entries $ctext] {
992 if {$w == $e} return
993 }
994 focus .
995}
996
997proc savestuff {w} {
998 global canv canv2 canv3 ctext cflist mainfont textfont uifont tabstop
999 global stuffsaved findmergefiles maxgraphpct
1000 global maxwidth showneartags showlocalchanges
1001 global viewname viewfiles viewargs viewperm nextviewnum
1002 global cmitmode wrapcomment
1003 global colors bgcolor fgcolor diffcolors selectbgcolor
1004
1005 if {$stuffsaved} return
1006 if {![winfo viewable .]} return
1007 catch {
1008 set f [open "~/.gitk-new" w]
1009 puts $f [list set mainfont $mainfont]
1010 puts $f [list set textfont $textfont]
1011 puts $f [list set uifont $uifont]
1012 puts $f [list set tabstop $tabstop]
1013 puts $f [list set findmergefiles $findmergefiles]
1014 puts $f [list set maxgraphpct $maxgraphpct]
1015 puts $f [list set maxwidth $maxwidth]
1016 puts $f [list set cmitmode $cmitmode]
1017 puts $f [list set wrapcomment $wrapcomment]
1018 puts $f [list set showneartags $showneartags]
1019 puts $f [list set showlocalchanges $showlocalchanges]
1020 puts $f [list set bgcolor $bgcolor]
1021 puts $f [list set fgcolor $fgcolor]
1022 puts $f [list set colors $colors]
1023 puts $f [list set diffcolors $diffcolors]
1024 puts $f [list set selectbgcolor $selectbgcolor]
1025
1026 puts $f "set geometry(main) [wm geometry .]"
1027 puts $f "set geometry(topwidth) [winfo width .tf]"
1028 puts $f "set geometry(topheight) [winfo height .tf]"
1029 puts $f "set geometry(pwsash0) \"[.tf.histframe.pwclist sash coord 0]\""
1030 puts $f "set geometry(pwsash1) \"[.tf.histframe.pwclist sash coord 1]\""
1031 puts $f "set geometry(botwidth) [winfo width .bleft]"
1032 puts $f "set geometry(botheight) [winfo height .bleft]"
1033
1034 puts -nonewline $f "set permviews {"
1035 for {set v 0} {$v < $nextviewnum} {incr v} {
1036 if {$viewperm($v)} {
1037 puts $f "{[list $viewname($v) $viewfiles($v) $viewargs($v)]}"
1038 }
1039 }
1040 puts $f "}"
1041 close $f
1042 file rename -force "~/.gitk-new" "~/.gitk"
1043 }
1044 set stuffsaved 1
1045}
1046
1047proc resizeclistpanes {win w} {
1048 global oldwidth
1049 if {[info exists oldwidth($win)]} {
1050 set s0 [$win sash coord 0]
1051 set s1 [$win sash coord 1]
1052 if {$w < 60} {
1053 set sash0 [expr {int($w/2 - 2)}]
1054 set sash1 [expr {int($w*5/6 - 2)}]
1055 } else {
1056 set factor [expr {1.0 * $w / $oldwidth($win)}]
1057 set sash0 [expr {int($factor * [lindex $s0 0])}]
1058 set sash1 [expr {int($factor * [lindex $s1 0])}]
1059 if {$sash0 < 30} {
1060 set sash0 30
1061 }
1062 if {$sash1 < $sash0 + 20} {
1063 set sash1 [expr {$sash0 + 20}]
1064 }
1065 if {$sash1 > $w - 10} {
1066 set sash1 [expr {$w - 10}]
1067 if {$sash0 > $sash1 - 20} {
1068 set sash0 [expr {$sash1 - 20}]
1069 }
1070 }
1071 }
1072 $win sash place 0 $sash0 [lindex $s0 1]
1073 $win sash place 1 $sash1 [lindex $s1 1]
1074 }
1075 set oldwidth($win) $w
1076}
1077
1078proc resizecdetpanes {win w} {
1079 global oldwidth
1080 if {[info exists oldwidth($win)]} {
1081 set s0 [$win sash coord 0]
1082 if {$w < 60} {
1083 set sash0 [expr {int($w*3/4 - 2)}]
1084 } else {
1085 set factor [expr {1.0 * $w / $oldwidth($win)}]
1086 set sash0 [expr {int($factor * [lindex $s0 0])}]
1087 if {$sash0 < 45} {
1088 set sash0 45
1089 }
1090 if {$sash0 > $w - 15} {
1091 set sash0 [expr {$w - 15}]
1092 }
1093 }
1094 $win sash place 0 $sash0 [lindex $s0 1]
1095 }
1096 set oldwidth($win) $w
1097}
1098
1099proc allcanvs args {
1100 global canv canv2 canv3
1101 eval $canv $args
1102 eval $canv2 $args
1103 eval $canv3 $args
1104}
1105
1106proc bindall {event action} {
1107 global canv canv2 canv3
1108 bind $canv $event $action
1109 bind $canv2 $event $action
1110 bind $canv3 $event $action
1111}
1112
1113proc about {} {
1114 global uifont
1115 set w .about
1116 if {[winfo exists $w]} {
1117 raise $w
1118 return
1119 }
1120 toplevel $w
1121 wm title $w "About gitk"
1122 message $w.m -text {
1123Gitk - a commit viewer for git
1124
1125Copyright © 2005-2006 Paul Mackerras
1126
1127Use and redistribute under the terms of the GNU General Public License} \
1128 -justify center -aspect 400 -border 2 -bg white -relief groove
1129 pack $w.m -side top -fill x -padx 2 -pady 2
1130 $w.m configure -font $uifont
1131 button $w.ok -text Close -command "destroy $w" -default active
1132 pack $w.ok -side bottom
1133 $w.ok configure -font $uifont
1134 bind $w <Visibility> "focus $w.ok"
1135 bind $w <Key-Escape> "destroy $w"
1136 bind $w <Key-Return> "destroy $w"
1137}
1138
1139proc keys {} {
1140 global uifont
1141 set w .keys
1142 if {[winfo exists $w]} {
1143 raise $w
1144 return
1145 }
1146 if {[tk windowingsystem] eq {aqua}} {
1147 set M1T Cmd
1148 } else {
1149 set M1T Ctrl
1150 }
1151 toplevel $w
1152 wm title $w "Gitk key bindings"
1153 message $w.m -text "
1154Gitk key bindings:
1155
1156<$M1T-Q> Quit
1157<Home> Move to first commit
1158<End> Move to last commit
1159<Up>, p, i Move up one commit
1160<Down>, n, k Move down one commit
1161<Left>, z, j Go back in history list
1162<Right>, x, l Go forward in history list
1163<PageUp> Move up one page in commit list
1164<PageDown> Move down one page in commit list
1165<$M1T-Home> Scroll to top of commit list
1166<$M1T-End> Scroll to bottom of commit list
1167<$M1T-Up> Scroll commit list up one line
1168<$M1T-Down> Scroll commit list down one line
1169<$M1T-PageUp> Scroll commit list up one page
1170<$M1T-PageDown> Scroll commit list down one page
1171<Shift-Up> Move to previous highlighted line
1172<Shift-Down> Move to next highlighted line
1173<Delete>, b Scroll diff view up one page
1174<Backspace> Scroll diff view up one page
1175<Space> Scroll diff view down one page
1176u Scroll diff view up 18 lines
1177d Scroll diff view down 18 lines
1178<$M1T-F> Find
1179<$M1T-G> Move to next find hit
1180<Return> Move to next find hit
1181/ Move to next find hit, or redo find
1182? Move to previous find hit
1183f Scroll diff view to next file
1184<$M1T-S> Search for next hit in diff view
1185<$M1T-R> Search for previous hit in diff view
1186<$M1T-KP+> Increase font size
1187<$M1T-plus> Increase font size
1188<$M1T-KP-> Decrease font size
1189<$M1T-minus> Decrease font size
1190<F5> Update
1191" \
1192 -justify left -bg white -border 2 -relief groove
1193 pack $w.m -side top -fill both -padx 2 -pady 2
1194 $w.m configure -font $uifont
1195 button $w.ok -text Close -command "destroy $w" -default active
1196 pack $w.ok -side bottom
1197 $w.ok configure -font $uifont
1198 bind $w <Visibility> "focus $w.ok"
1199 bind $w <Key-Escape> "destroy $w"
1200 bind $w <Key-Return> "destroy $w"
1201}
1202
1203# Procedures for manipulating the file list window at the
1204# bottom right of the overall window.
1205
1206proc treeview {w l openlevs} {
1207 global treecontents treediropen treeheight treeparent treeindex
1208
1209 set ix 0
1210 set treeindex() 0
1211 set lev 0
1212 set prefix {}
1213 set prefixend -1
1214 set prefendstack {}
1215 set htstack {}
1216 set ht 0
1217 set treecontents() {}
1218 $w conf -state normal
1219 foreach f $l {
1220 while {[string range $f 0 $prefixend] ne $prefix} {
1221 if {$lev <= $openlevs} {
1222 $w mark set e:$treeindex($prefix) "end -1c"
1223 $w mark gravity e:$treeindex($prefix) left
1224 }
1225 set treeheight($prefix) $ht
1226 incr ht [lindex $htstack end]
1227 set htstack [lreplace $htstack end end]
1228 set prefixend [lindex $prefendstack end]
1229 set prefendstack [lreplace $prefendstack end end]
1230 set prefix [string range $prefix 0 $prefixend]
1231 incr lev -1
1232 }
1233 set tail [string range $f [expr {$prefixend+1}] end]
1234 while {[set slash [string first "/" $tail]] >= 0} {
1235 lappend htstack $ht
1236 set ht 0
1237 lappend prefendstack $prefixend
1238 incr prefixend [expr {$slash + 1}]
1239 set d [string range $tail 0 $slash]
1240 lappend treecontents($prefix) $d
1241 set oldprefix $prefix
1242 append prefix $d
1243 set treecontents($prefix) {}
1244 set treeindex($prefix) [incr ix]
1245 set treeparent($prefix) $oldprefix
1246 set tail [string range $tail [expr {$slash+1}] end]
1247 if {$lev <= $openlevs} {
1248 set ht 1
1249 set treediropen($prefix) [expr {$lev < $openlevs}]
1250 set bm [expr {$lev == $openlevs? "tri-rt": "tri-dn"}]
1251 $w mark set d:$ix "end -1c"
1252 $w mark gravity d:$ix left
1253 set str "\n"
1254 for {set i 0} {$i < $lev} {incr i} {append str "\t"}
1255 $w insert end $str
1256 $w image create end -align center -image $bm -padx 1 \
1257 -name a:$ix
1258 $w insert end $d [highlight_tag $prefix]
1259 $w mark set s:$ix "end -1c"
1260 $w mark gravity s:$ix left
1261 }
1262 incr lev
1263 }
1264 if {$tail ne {}} {
1265 if {$lev <= $openlevs} {
1266 incr ht
1267 set str "\n"
1268 for {set i 0} {$i < $lev} {incr i} {append str "\t"}
1269 $w insert end $str
1270 $w insert end $tail [highlight_tag $f]
1271 }
1272 lappend treecontents($prefix) $tail
1273 }
1274 }
1275 while {$htstack ne {}} {
1276 set treeheight($prefix) $ht
1277 incr ht [lindex $htstack end]
1278 set htstack [lreplace $htstack end end]
1279 set prefixend [lindex $prefendstack end]
1280 set prefendstack [lreplace $prefendstack end end]
1281 set prefix [string range $prefix 0 $prefixend]
1282 }
1283 $w conf -state disabled
1284}
1285
1286proc linetoelt {l} {
1287 global treeheight treecontents
1288
1289 set y 2
1290 set prefix {}
1291 while {1} {
1292 foreach e $treecontents($prefix) {
1293 if {$y == $l} {
1294 return "$prefix$e"
1295 }
1296 set n 1
1297 if {[string index $e end] eq "/"} {
1298 set n $treeheight($prefix$e)
1299 if {$y + $n > $l} {
1300 append prefix $e
1301 incr y
1302 break
1303 }
1304 }
1305 incr y $n
1306 }
1307 }
1308}
1309
1310proc highlight_tree {y prefix} {
1311 global treeheight treecontents cflist
1312
1313 foreach e $treecontents($prefix) {
1314 set path $prefix$e
1315 if {[highlight_tag $path] ne {}} {
1316 $cflist tag add bold $y.0 "$y.0 lineend"
1317 }
1318 incr y
1319 if {[string index $e end] eq "/" && $treeheight($path) > 1} {
1320 set y [highlight_tree $y $path]
1321 }
1322 }
1323 return $y
1324}
1325
1326proc treeclosedir {w dir} {
1327 global treediropen treeheight treeparent treeindex
1328
1329 set ix $treeindex($dir)
1330 $w conf -state normal
1331 $w delete s:$ix e:$ix
1332 set treediropen($dir) 0
1333 $w image configure a:$ix -image tri-rt
1334 $w conf -state disabled
1335 set n [expr {1 - $treeheight($dir)}]
1336 while {$dir ne {}} {
1337 incr treeheight($dir) $n
1338 set dir $treeparent($dir)
1339 }
1340}
1341
1342proc treeopendir {w dir} {
1343 global treediropen treeheight treeparent treecontents treeindex
1344
1345 set ix $treeindex($dir)
1346 $w conf -state normal
1347 $w image configure a:$ix -image tri-dn
1348 $w mark set e:$ix s:$ix
1349 $w mark gravity e:$ix right
1350 set lev 0
1351 set str "\n"
1352 set n [llength $treecontents($dir)]
1353 for {set x $dir} {$x ne {}} {set x $treeparent($x)} {
1354 incr lev
1355 append str "\t"
1356 incr treeheight($x) $n
1357 }
1358 foreach e $treecontents($dir) {
1359 set de $dir$e
1360 if {[string index $e end] eq "/"} {
1361 set iy $treeindex($de)
1362 $w mark set d:$iy e:$ix
1363 $w mark gravity d:$iy left
1364 $w insert e:$ix $str
1365 set treediropen($de) 0
1366 $w image create e:$ix -align center -image tri-rt -padx 1 \
1367 -name a:$iy
1368 $w insert e:$ix $e [highlight_tag $de]
1369 $w mark set s:$iy e:$ix
1370 $w mark gravity s:$iy left
1371 set treeheight($de) 1
1372 } else {
1373 $w insert e:$ix $str
1374 $w insert e:$ix $e [highlight_tag $de]
1375 }
1376 }
1377 $w mark gravity e:$ix left
1378 $w conf -state disabled
1379 set treediropen($dir) 1
1380 set top [lindex [split [$w index @0,0] .] 0]
1381 set ht [$w cget -height]
1382 set l [lindex [split [$w index s:$ix] .] 0]
1383 if {$l < $top} {
1384 $w yview $l.0
1385 } elseif {$l + $n + 1 > $top + $ht} {
1386 set top [expr {$l + $n + 2 - $ht}]
1387 if {$l < $top} {
1388 set top $l
1389 }
1390 $w yview $top.0
1391 }
1392}
1393
1394proc treeclick {w x y} {
1395 global treediropen cmitmode ctext cflist cflist_top
1396
1397 if {$cmitmode ne "tree"} return
1398 if {![info exists cflist_top]} return
1399 set l [lindex [split [$w index "@$x,$y"] "."] 0]
1400 $cflist tag remove highlight $cflist_top.0 "$cflist_top.0 lineend"
1401 $cflist tag add highlight $l.0 "$l.0 lineend"
1402 set cflist_top $l
1403 if {$l == 1} {
1404 $ctext yview 1.0
1405 return
1406 }
1407 set e [linetoelt $l]
1408 if {[string index $e end] ne "/"} {
1409 showfile $e
1410 } elseif {$treediropen($e)} {
1411 treeclosedir $w $e
1412 } else {
1413 treeopendir $w $e
1414 }
1415}
1416
1417proc setfilelist {id} {
1418 global treefilelist cflist
1419
1420 treeview $cflist $treefilelist($id) 0
1421}
1422
1423image create bitmap tri-rt -background black -foreground blue -data {
1424 #define tri-rt_width 13
1425 #define tri-rt_height 13
1426 static unsigned char tri-rt_bits[] = {
1427 0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x30, 0x00, 0x70, 0x00, 0xf0, 0x00,
1428 0xf0, 0x01, 0xf0, 0x00, 0x70, 0x00, 0x30, 0x00, 0x10, 0x00, 0x00, 0x00,
1429 0x00, 0x00};
1430} -maskdata {
1431 #define tri-rt-mask_width 13
1432 #define tri-rt-mask_height 13
1433 static unsigned char tri-rt-mask_bits[] = {
1434 0x08, 0x00, 0x18, 0x00, 0x38, 0x00, 0x78, 0x00, 0xf8, 0x00, 0xf8, 0x01,
1435 0xf8, 0x03, 0xf8, 0x01, 0xf8, 0x00, 0x78, 0x00, 0x38, 0x00, 0x18, 0x00,
1436 0x08, 0x00};
1437}
1438image create bitmap tri-dn -background black -foreground blue -data {
1439 #define tri-dn_width 13
1440 #define tri-dn_height 13
1441 static unsigned char tri-dn_bits[] = {
1442 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xfc, 0x07, 0xf8, 0x03,
1443 0xf0, 0x01, 0xe0, 0x00, 0x40, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
1444 0x00, 0x00};
1445} -maskdata {
1446 #define tri-dn-mask_width 13
1447 #define tri-dn-mask_height 13
1448 static unsigned char tri-dn-mask_bits[] = {
1449 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0x1f, 0xfe, 0x0f, 0xfc, 0x07,
1450 0xf8, 0x03, 0xf0, 0x01, 0xe0, 0x00, 0x40, 0x00, 0x00, 0x00, 0x00, 0x00,
1451 0x00, 0x00};
1452}
1453
1454proc init_flist {first} {
1455 global cflist cflist_top selectedline difffilestart
1456
1457 $cflist conf -state normal
1458 $cflist delete 0.0 end
1459 if {$first ne {}} {
1460 $cflist insert end $first
1461 set cflist_top 1
1462 $cflist tag add highlight 1.0 "1.0 lineend"
1463 } else {
1464 catch {unset cflist_top}
1465 }
1466 $cflist conf -state disabled
1467 set difffilestart {}
1468}
1469
1470proc highlight_tag {f} {
1471 global highlight_paths
1472
1473 foreach p $highlight_paths {
1474 if {[string match $p $f]} {
1475 return "bold"
1476 }
1477 }
1478 return {}
1479}
1480
1481proc highlight_filelist {} {
1482 global cmitmode cflist
1483
1484 $cflist conf -state normal
1485 if {$cmitmode ne "tree"} {
1486 set end [lindex [split [$cflist index end] .] 0]
1487 for {set l 2} {$l < $end} {incr l} {
1488 set line [$cflist get $l.0 "$l.0 lineend"]
1489 if {[highlight_tag $line] ne {}} {
1490 $cflist tag add bold $l.0 "$l.0 lineend"
1491 }
1492 }
1493 } else {
1494 highlight_tree 2 {}
1495 }
1496 $cflist conf -state disabled
1497}
1498
1499proc unhighlight_filelist {} {
1500 global cflist
1501
1502 $cflist conf -state normal
1503 $cflist tag remove bold 1.0 end
1504 $cflist conf -state disabled
1505}
1506
1507proc add_flist {fl} {
1508 global cflist
1509
1510 $cflist conf -state normal
1511 foreach f $fl {
1512 $cflist insert end "\n"
1513 $cflist insert end $f [highlight_tag $f]
1514 }
1515 $cflist conf -state disabled
1516}
1517
1518proc sel_flist {w x y} {
1519 global ctext difffilestart cflist cflist_top cmitmode
1520
1521 if {$cmitmode eq "tree"} return
1522 if {![info exists cflist_top]} return
1523 set l [lindex [split [$w index "@$x,$y"] "."] 0]
1524 $cflist tag remove highlight $cflist_top.0 "$cflist_top.0 lineend"
1525 $cflist tag add highlight $l.0 "$l.0 lineend"
1526 set cflist_top $l
1527 if {$l == 1} {
1528 $ctext yview 1.0
1529 } else {
1530 catch {$ctext yview [lindex $difffilestart [expr {$l - 2}]]}
1531 }
1532}
1533
1534proc pop_flist_menu {w X Y x y} {
1535 global ctext cflist cmitmode flist_menu flist_menu_file
1536 global treediffs diffids
1537
1538 set l [lindex [split [$w index "@$x,$y"] "."] 0]
1539 if {$l <= 1} return
1540 if {$cmitmode eq "tree"} {
1541 set e [linetoelt $l]
1542 if {[string index $e end] eq "/"} return
1543 } else {
1544 set e [lindex $treediffs($diffids) [expr {$l-2}]]
1545 }
1546 set flist_menu_file $e
1547 tk_popup $flist_menu $X $Y
1548}
1549
1550proc flist_hl {only} {
1551 global flist_menu_file highlight_files
1552
1553 set x [shellquote $flist_menu_file]
1554 if {$only || $highlight_files eq {}} {
1555 set highlight_files $x
1556 } else {
1557 append highlight_files " " $x
1558 }
1559}
1560
1561# Functions for adding and removing shell-type quoting
1562
1563proc shellquote {str} {
1564 if {![string match "*\['\"\\ \t]*" $str]} {
1565 return $str
1566 }
1567 if {![string match "*\['\"\\]*" $str]} {
1568 return "\"$str\""
1569 }
1570 if {![string match "*'*" $str]} {
1571 return "'$str'"
1572 }
1573 return "\"[string map {\" \\\" \\ \\\\} $str]\""
1574}
1575
1576proc shellarglist {l} {
1577 set str {}
1578 foreach a $l {
1579 if {$str ne {}} {
1580 append str " "
1581 }
1582 append str [shellquote $a]
1583 }
1584 return $str
1585}
1586
1587proc shelldequote {str} {
1588 set ret {}
1589 set used -1
1590 while {1} {
1591 incr used
1592 if {![regexp -start $used -indices "\['\"\\\\ \t]" $str first]} {
1593 append ret [string range $str $used end]
1594 set used [string length $str]
1595 break
1596 }
1597 set first [lindex $first 0]
1598 set ch [string index $str $first]
1599 if {$first > $used} {
1600 append ret [string range $str $used [expr {$first - 1}]]
1601 set used $first
1602 }
1603 if {$ch eq " " || $ch eq "\t"} break
1604 incr used
1605 if {$ch eq "'"} {
1606 set first [string first "'" $str $used]
1607 if {$first < 0} {
1608 error "unmatched single-quote"
1609 }
1610 append ret [string range $str $used [expr {$first - 1}]]
1611 set used $first
1612 continue
1613 }
1614 if {$ch eq "\\"} {
1615 if {$used >= [string length $str]} {
1616 error "trailing backslash"
1617 }
1618 append ret [string index $str $used]
1619 continue
1620 }
1621 # here ch == "\""
1622 while {1} {
1623 if {![regexp -start $used -indices "\[\"\\\\]" $str first]} {
1624 error "unmatched double-quote"
1625 }
1626 set first [lindex $first 0]
1627 set ch [string index $str $first]
1628 if {$first > $used} {
1629 append ret [string range $str $used [expr {$first - 1}]]
1630 set used $first
1631 }
1632 if {$ch eq "\""} break
1633 incr used
1634 append ret [string index $str $used]
1635 incr used
1636 }
1637 }
1638 return [list $used $ret]
1639}
1640
1641proc shellsplit {str} {
1642 set l {}
1643 while {1} {
1644 set str [string trimleft $str]
1645 if {$str eq {}} break
1646 set dq [shelldequote $str]
1647 set n [lindex $dq 0]
1648 set word [lindex $dq 1]
1649 set str [string range $str $n end]
1650 lappend l $word
1651 }
1652 return $l
1653}
1654
1655# Code to implement multiple views
1656
1657proc newview {ishighlight} {
1658 global nextviewnum newviewname newviewperm uifont newishighlight
1659 global newviewargs revtreeargs
1660
1661 set newishighlight $ishighlight
1662 set top .gitkview
1663 if {[winfo exists $top]} {
1664 raise $top
1665 return
1666 }
1667 set newviewname($nextviewnum) "View $nextviewnum"
1668 set newviewperm($nextviewnum) 0
1669 set newviewargs($nextviewnum) [shellarglist $revtreeargs]
1670 vieweditor $top $nextviewnum "Gitk view definition"
1671}
1672
1673proc editview {} {
1674 global curview
1675 global viewname viewperm newviewname newviewperm
1676 global viewargs newviewargs
1677
1678 set top .gitkvedit-$curview
1679 if {[winfo exists $top]} {
1680 raise $top
1681 return
1682 }
1683 set newviewname($curview) $viewname($curview)
1684 set newviewperm($curview) $viewperm($curview)
1685 set newviewargs($curview) [shellarglist $viewargs($curview)]
1686 vieweditor $top $curview "Gitk: edit view $viewname($curview)"
1687}
1688
1689proc vieweditor {top n title} {
1690 global newviewname newviewperm viewfiles
1691 global uifont
1692
1693 toplevel $top
1694 wm title $top $title
1695 label $top.nl -text "Name" -font $uifont
1696 entry $top.name -width 20 -textvariable newviewname($n) -font $uifont
1697 grid $top.nl $top.name -sticky w -pady 5
1698 checkbutton $top.perm -text "Remember this view" -variable newviewperm($n) \
1699 -font $uifont
1700 grid $top.perm - -pady 5 -sticky w
1701 message $top.al -aspect 1000 -font $uifont \
1702 -text "Commits to include (arguments to git rev-list):"
1703 grid $top.al - -sticky w -pady 5
1704 entry $top.args -width 50 -textvariable newviewargs($n) \
1705 -background white -font $uifont
1706 grid $top.args - -sticky ew -padx 5
1707 message $top.l -aspect 1000 -font $uifont \
1708 -text "Enter files and directories to include, one per line:"
1709 grid $top.l - -sticky w
1710 text $top.t -width 40 -height 10 -background white -font $uifont
1711 if {[info exists viewfiles($n)]} {
1712 foreach f $viewfiles($n) {
1713 $top.t insert end $f
1714 $top.t insert end "\n"
1715 }
1716 $top.t delete {end - 1c} end
1717 $top.t mark set insert 0.0
1718 }
1719 grid $top.t - -sticky ew -padx 5
1720 frame $top.buts
1721 button $top.buts.ok -text "OK" -command [list newviewok $top $n] \
1722 -font $uifont
1723 button $top.buts.can -text "Cancel" -command [list destroy $top] \
1724 -font $uifont
1725 grid $top.buts.ok $top.buts.can
1726 grid columnconfigure $top.buts 0 -weight 1 -uniform a
1727 grid columnconfigure $top.buts 1 -weight 1 -uniform a
1728 grid $top.buts - -pady 10 -sticky ew
1729 focus $top.t
1730}
1731
1732proc doviewmenu {m first cmd op argv} {
1733 set nmenu [$m index end]
1734 for {set i $first} {$i <= $nmenu} {incr i} {
1735 if {[$m entrycget $i -command] eq $cmd} {
1736 eval $m $op $i $argv
1737 break
1738 }
1739 }
1740}
1741
1742proc allviewmenus {n op args} {
1743 global viewhlmenu
1744
1745 doviewmenu .bar.view 5 [list showview $n] $op $args
1746 doviewmenu $viewhlmenu 1 [list addvhighlight $n] $op $args
1747}
1748
1749proc newviewok {top n} {
1750 global nextviewnum newviewperm newviewname newishighlight
1751 global viewname viewfiles viewperm selectedview curview
1752 global viewargs newviewargs viewhlmenu
1753
1754 if {[catch {
1755 set newargs [shellsplit $newviewargs($n)]
1756 } err]} {
1757 error_popup "Error in commit selection arguments: $err"
1758 wm raise $top
1759 focus $top
1760 return
1761 }
1762 set files {}
1763 foreach f [split [$top.t get 0.0 end] "\n"] {
1764 set ft [string trim $f]
1765 if {$ft ne {}} {
1766 lappend files $ft
1767 }
1768 }
1769 if {![info exists viewfiles($n)]} {
1770 # creating a new view
1771 incr nextviewnum
1772 set viewname($n) $newviewname($n)
1773 set viewperm($n) $newviewperm($n)
1774 set viewfiles($n) $files
1775 set viewargs($n) $newargs
1776 addviewmenu $n
1777 if {!$newishighlight} {
1778 run showview $n
1779 } else {
1780 run addvhighlight $n
1781 }
1782 } else {
1783 # editing an existing view
1784 set viewperm($n) $newviewperm($n)
1785 if {$newviewname($n) ne $viewname($n)} {
1786 set viewname($n) $newviewname($n)
1787 doviewmenu .bar.view 5 [list showview $n] \
1788 entryconf [list -label $viewname($n)]
1789 doviewmenu $viewhlmenu 1 [list addvhighlight $n] \
1790 entryconf [list -label $viewname($n) -value $viewname($n)]
1791 }
1792 if {$files ne $viewfiles($n) || $newargs ne $viewargs($n)} {
1793 set viewfiles($n) $files
1794 set viewargs($n) $newargs
1795 if {$curview == $n} {
1796 run updatecommits
1797 }
1798 }
1799 }
1800 catch {destroy $top}
1801}
1802
1803proc delview {} {
1804 global curview viewdata viewperm hlview selectedhlview
1805
1806 if {$curview == 0} return
1807 if {[info exists hlview] && $hlview == $curview} {
1808 set selectedhlview None
1809 unset hlview
1810 }
1811 allviewmenus $curview delete
1812 set viewdata($curview) {}
1813 set viewperm($curview) 0
1814 showview 0
1815}
1816
1817proc addviewmenu {n} {
1818 global viewname viewhlmenu
1819
1820 .bar.view add radiobutton -label $viewname($n) \
1821 -command [list showview $n] -variable selectedview -value $n
1822 $viewhlmenu add radiobutton -label $viewname($n) \
1823 -command [list addvhighlight $n] -variable selectedhlview
1824}
1825
1826proc flatten {var} {
1827 global $var
1828
1829 set ret {}
1830 foreach i [array names $var] {
1831 lappend ret $i [set $var\($i\)]
1832 }
1833 return $ret
1834}
1835
1836proc unflatten {var l} {
1837 global $var
1838
1839 catch {unset $var}
1840 foreach {i v} $l {
1841 set $var\($i\) $v
1842 }
1843}
1844
1845proc showview {n} {
1846 global curview viewdata viewfiles
1847 global displayorder parentlist rowidlist rowoffsets
1848 global colormap rowtextx commitrow nextcolor canvxmax
1849 global numcommits rowrangelist commitlisted idrowranges rowchk
1850 global selectedline currentid canv canvy0
1851 global treediffs
1852 global pending_select phase
1853 global commitidx rowlaidout rowoptim
1854 global commfd
1855 global selectedview selectfirst
1856 global vparentlist vdisporder vcmitlisted
1857 global hlview selectedhlview
1858
1859 if {$n == $curview} return
1860 set selid {}
1861 if {[info exists selectedline]} {
1862 set selid $currentid
1863 set y [yc $selectedline]
1864 set ymax [lindex [$canv cget -scrollregion] 3]
1865 set span [$canv yview]
1866 set ytop [expr {[lindex $span 0] * $ymax}]
1867 set ybot [expr {[lindex $span 1] * $ymax}]
1868 if {$ytop < $y && $y < $ybot} {
1869 set yscreen [expr {$y - $ytop}]
1870 } else {
1871 set yscreen [expr {($ybot - $ytop) / 2}]
1872 }
1873 } elseif {[info exists pending_select]} {
1874 set selid $pending_select
1875 unset pending_select
1876 }
1877 unselectline
1878 normalline
1879 if {$curview >= 0} {
1880 set vparentlist($curview) $parentlist
1881 set vdisporder($curview) $displayorder
1882 set vcmitlisted($curview) $commitlisted
1883 if {$phase ne {}} {
1884 set viewdata($curview) \
1885 [list $phase $rowidlist $rowoffsets $rowrangelist \
1886 [flatten idrowranges] [flatten idinlist] \
1887 $rowlaidout $rowoptim $numcommits]
1888 } elseif {![info exists viewdata($curview)]
1889 || [lindex $viewdata($curview) 0] ne {}} {
1890 set viewdata($curview) \
1891 [list {} $rowidlist $rowoffsets $rowrangelist]
1892 }
1893 }
1894 catch {unset treediffs}
1895 clear_display
1896 if {[info exists hlview] && $hlview == $n} {
1897 unset hlview
1898 set selectedhlview None
1899 }
1900
1901 set curview $n
1902 set selectedview $n
1903 .bar.view entryconf Edit* -state [expr {$n == 0? "disabled": "normal"}]
1904 .bar.view entryconf Delete* -state [expr {$n == 0? "disabled": "normal"}]
1905
1906 if {![info exists viewdata($n)]} {
1907 if {$selid ne {}} {
1908 set pending_select $selid
1909 }
1910 getcommits
1911 return
1912 }
1913
1914 set v $viewdata($n)
1915 set phase [lindex $v 0]
1916 set displayorder $vdisporder($n)
1917 set parentlist $vparentlist($n)
1918 set commitlisted $vcmitlisted($n)
1919 set rowidlist [lindex $v 1]
1920 set rowoffsets [lindex $v 2]
1921 set rowrangelist [lindex $v 3]
1922 if {$phase eq {}} {
1923 set numcommits [llength $displayorder]
1924 catch {unset idrowranges}
1925 } else {
1926 unflatten idrowranges [lindex $v 4]
1927 unflatten idinlist [lindex $v 5]
1928 set rowlaidout [lindex $v 6]
1929 set rowoptim [lindex $v 7]
1930 set numcommits [lindex $v 8]
1931 catch {unset rowchk}
1932 }
1933
1934 catch {unset colormap}
1935 catch {unset rowtextx}
1936 set nextcolor 0
1937 set canvxmax [$canv cget -width]
1938 set curview $n
1939 set row 0
1940 setcanvscroll
1941 set yf 0
1942 set row {}
1943 set selectfirst 0
1944 if {$selid ne {} && [info exists commitrow($n,$selid)]} {
1945 set row $commitrow($n,$selid)
1946 # try to get the selected row in the same position on the screen
1947 set ymax [lindex [$canv cget -scrollregion] 3]
1948 set ytop [expr {[yc $row] - $yscreen}]
1949 if {$ytop < 0} {
1950 set ytop 0
1951 }
1952 set yf [expr {$ytop * 1.0 / $ymax}]
1953 }
1954 allcanvs yview moveto $yf
1955 drawvisible
1956 if {$row ne {}} {
1957 selectline $row 0
1958 } elseif {$selid ne {}} {
1959 set pending_select $selid
1960 } else {
1961 set row [first_real_row]
1962 if {$row < $numcommits} {
1963 selectline $row 0
1964 } else {
1965 set selectfirst 1
1966 }
1967 }
1968 if {$phase ne {}} {
1969 if {$phase eq "getcommits"} {
1970 show_status "Reading commits..."
1971 }
1972 run chewcommits $n
1973 } elseif {$numcommits == 0} {
1974 show_status "No commits selected"
1975 }
1976}
1977
1978# Stuff relating to the highlighting facility
1979
1980proc ishighlighted {row} {
1981 global vhighlights fhighlights nhighlights rhighlights
1982
1983 if {[info exists nhighlights($row)] && $nhighlights($row) > 0} {
1984 return $nhighlights($row)
1985 }
1986 if {[info exists vhighlights($row)] && $vhighlights($row) > 0} {
1987 return $vhighlights($row)
1988 }
1989 if {[info exists fhighlights($row)] && $fhighlights($row) > 0} {
1990 return $fhighlights($row)
1991 }
1992 if {[info exists rhighlights($row)] && $rhighlights($row) > 0} {
1993 return $rhighlights($row)
1994 }
1995 return 0
1996}
1997
1998proc bolden {row font} {
1999 global canv linehtag selectedline boldrows
2000
2001 lappend boldrows $row
2002 $canv itemconf $linehtag($row) -font $font
2003 if {[info exists selectedline] && $row == $selectedline} {
2004 $canv delete secsel
2005 set t [eval $canv create rect [$canv bbox $linehtag($row)] \
2006 -outline {{}} -tags secsel \
2007 -fill [$canv cget -selectbackground]]
2008 $canv lower $t
2009 }
2010}
2011
2012proc bolden_name {row font} {
2013 global canv2 linentag selectedline boldnamerows
2014
2015 lappend boldnamerows $row
2016 $canv2 itemconf $linentag($row) -font $font
2017 if {[info exists selectedline] && $row == $selectedline} {
2018 $canv2 delete secsel
2019 set t [eval $canv2 create rect [$canv2 bbox $linentag($row)] \
2020 -outline {{}} -tags secsel \
2021 -fill [$canv2 cget -selectbackground]]
2022 $canv2 lower $t
2023 }
2024}
2025
2026proc unbolden {} {
2027 global mainfont boldrows
2028
2029 set stillbold {}
2030 foreach row $boldrows {
2031 if {![ishighlighted $row]} {
2032 bolden $row $mainfont
2033 } else {
2034 lappend stillbold $row
2035 }
2036 }
2037 set boldrows $stillbold
2038}
2039
2040proc addvhighlight {n} {
2041 global hlview curview viewdata vhl_done vhighlights commitidx
2042
2043 if {[info exists hlview]} {
2044 delvhighlight
2045 }
2046 set hlview $n
2047 if {$n != $curview && ![info exists viewdata($n)]} {
2048 set viewdata($n) [list getcommits {{}} {{}} {} {} {} 0 0 0 {}]
2049 set vparentlist($n) {}
2050 set vdisporder($n) {}
2051 set vcmitlisted($n) {}
2052 start_rev_list $n
2053 }
2054 set vhl_done $commitidx($hlview)
2055 if {$vhl_done > 0} {
2056 drawvisible
2057 }
2058}
2059
2060proc delvhighlight {} {
2061 global hlview vhighlights
2062
2063 if {![info exists hlview]} return
2064 unset hlview
2065 catch {unset vhighlights}
2066 unbolden
2067}
2068
2069proc vhighlightmore {} {
2070 global hlview vhl_done commitidx vhighlights
2071 global displayorder vdisporder curview mainfont
2072
2073 set font [concat $mainfont bold]
2074 set max $commitidx($hlview)
2075 if {$hlview == $curview} {
2076 set disp $displayorder
2077 } else {
2078 set disp $vdisporder($hlview)
2079 }
2080 set vr [visiblerows]
2081 set r0 [lindex $vr 0]
2082 set r1 [lindex $vr 1]
2083 for {set i $vhl_done} {$i < $max} {incr i} {
2084 set id [lindex $disp $i]
2085 if {[info exists commitrow($curview,$id)]} {
2086 set row $commitrow($curview,$id)
2087 if {$r0 <= $row && $row <= $r1} {
2088 if {![highlighted $row]} {
2089 bolden $row $font
2090 }
2091 set vhighlights($row) 1
2092 }
2093 }
2094 }
2095 set vhl_done $max
2096}
2097
2098proc askvhighlight {row id} {
2099 global hlview vhighlights commitrow iddrawn mainfont
2100
2101 if {[info exists commitrow($hlview,$id)]} {
2102 if {[info exists iddrawn($id)] && ![ishighlighted $row]} {
2103 bolden $row [concat $mainfont bold]
2104 }
2105 set vhighlights($row) 1
2106 } else {
2107 set vhighlights($row) 0
2108 }
2109}
2110
2111proc hfiles_change {name ix op} {
2112 global highlight_files filehighlight fhighlights fh_serial
2113 global mainfont highlight_paths
2114
2115 if {[info exists filehighlight]} {
2116 # delete previous highlights
2117 catch {close $filehighlight}
2118 unset filehighlight
2119 catch {unset fhighlights}
2120 unbolden
2121 unhighlight_filelist
2122 }
2123 set highlight_paths {}
2124 after cancel do_file_hl $fh_serial
2125 incr fh_serial
2126 if {$highlight_files ne {}} {
2127 after 300 do_file_hl $fh_serial
2128 }
2129}
2130
2131proc makepatterns {l} {
2132 set ret {}
2133 foreach e $l {
2134 set ee [string map {"*" "\\*" "?" "\\?" "\[" "\\\[" "\\" "\\\\"} $e]
2135 if {[string index $ee end] eq "/"} {
2136 lappend ret "$ee*"
2137 } else {
2138 lappend ret $ee
2139 lappend ret "$ee/*"
2140 }
2141 }
2142 return $ret
2143}
2144
2145proc do_file_hl {serial} {
2146 global highlight_files filehighlight highlight_paths gdttype fhl_list
2147
2148 if {$gdttype eq "touching paths:"} {
2149 if {[catch {set paths [shellsplit $highlight_files]}]} return
2150 set highlight_paths [makepatterns $paths]
2151 highlight_filelist
2152 set gdtargs [concat -- $paths]
2153 } else {
2154 set gdtargs [list "-S$highlight_files"]
2155 }
2156 set cmd [concat | git diff-tree -r -s --stdin $gdtargs]
2157 set filehighlight [open $cmd r+]
2158 fconfigure $filehighlight -blocking 0
2159 filerun $filehighlight readfhighlight
2160 set fhl_list {}
2161 drawvisible
2162 flushhighlights
2163}
2164
2165proc flushhighlights {} {
2166 global filehighlight fhl_list
2167
2168 if {[info exists filehighlight]} {
2169 lappend fhl_list {}
2170 puts $filehighlight ""
2171 flush $filehighlight
2172 }
2173}
2174
2175proc askfilehighlight {row id} {
2176 global filehighlight fhighlights fhl_list
2177
2178 lappend fhl_list $id
2179 set fhighlights($row) -1
2180 puts $filehighlight $id
2181}
2182
2183proc readfhighlight {} {
2184 global filehighlight fhighlights commitrow curview mainfont iddrawn
2185 global fhl_list
2186
2187 if {![info exists filehighlight]} {
2188 return 0
2189 }
2190 set nr 0
2191 while {[incr nr] <= 100 && [gets $filehighlight line] >= 0} {
2192 set line [string trim $line]
2193 set i [lsearch -exact $fhl_list $line]
2194 if {$i < 0} continue
2195 for {set j 0} {$j < $i} {incr j} {
2196 set id [lindex $fhl_list $j]
2197 if {[info exists commitrow($curview,$id)]} {
2198 set fhighlights($commitrow($curview,$id)) 0
2199 }
2200 }
2201 set fhl_list [lrange $fhl_list [expr {$i+1}] end]
2202 if {$line eq {}} continue
2203 if {![info exists commitrow($curview,$line)]} continue
2204 set row $commitrow($curview,$line)
2205 if {[info exists iddrawn($line)] && ![ishighlighted $row]} {
2206 bolden $row [concat $mainfont bold]
2207 }
2208 set fhighlights($row) 1
2209 }
2210 if {[eof $filehighlight]} {
2211 # strange...
2212 puts "oops, git diff-tree died"
2213 catch {close $filehighlight}
2214 unset filehighlight
2215 return 0
2216 }
2217 next_hlcont
2218 return 1
2219}
2220
2221proc find_change {name ix op} {
2222 global nhighlights mainfont boldnamerows
2223 global findstring findpattern findtype
2224
2225 # delete previous highlights, if any
2226 foreach row $boldnamerows {
2227 bolden_name $row $mainfont
2228 }
2229 set boldnamerows {}
2230 catch {unset nhighlights}
2231 unbolden
2232 unmarkmatches
2233 if {$findtype ne "Regexp"} {
2234 set e [string map {"*" "\\*" "?" "\\?" "\[" "\\\[" "\\" "\\\\"} \
2235 $findstring]
2236 set findpattern "*$e*"
2237 }
2238 drawvisible
2239}
2240
2241proc doesmatch {f} {
2242 global findtype findstring findpattern
2243
2244 if {$findtype eq "Regexp"} {
2245 return [regexp $findstring $f]
2246 } elseif {$findtype eq "IgnCase"} {
2247 return [string match -nocase $findpattern $f]
2248 } else {
2249 return [string match $findpattern $f]
2250 }
2251}
2252
2253proc askfindhighlight {row id} {
2254 global nhighlights commitinfo iddrawn mainfont
2255 global findloc
2256 global markingmatches
2257
2258 if {![info exists commitinfo($id)]} {
2259 getcommit $id
2260 }
2261 set info $commitinfo($id)
2262 set isbold 0
2263 set fldtypes {Headline Author Date Committer CDate Comments}
2264 foreach f $info ty $fldtypes {
2265 if {($findloc eq "All fields" || $findloc eq $ty) &&
2266 [doesmatch $f]} {
2267 if {$ty eq "Author"} {
2268 set isbold 2
2269 break
2270 }
2271 set isbold 1
2272 }
2273 }
2274 if {$isbold && [info exists iddrawn($id)]} {
2275 set f [concat $mainfont bold]
2276 if {![ishighlighted $row]} {
2277 bolden $row $f
2278 if {$isbold > 1} {
2279 bolden_name $row $f
2280 }
2281 }
2282 if {$markingmatches} {
2283 markrowmatches $row $id
2284 }
2285 }
2286 set nhighlights($row) $isbold
2287}
2288
2289proc markrowmatches {row id} {
2290 global canv canv2 linehtag linentag commitinfo findloc
2291
2292 set headline [lindex $commitinfo($id) 0]
2293 set author [lindex $commitinfo($id) 1]
2294 $canv delete match$row
2295 $canv2 delete match$row
2296 if {$findloc eq "All fields" || $findloc eq "Headline"} {
2297 set m [findmatches $headline]
2298 if {$m ne {}} {
2299 markmatches $canv $row $headline $linehtag($row) $m \
2300 [$canv itemcget $linehtag($row) -font] $row
2301 }
2302 }
2303 if {$findloc eq "All fields" || $findloc eq "Author"} {
2304 set m [findmatches $author]
2305 if {$m ne {}} {
2306 markmatches $canv2 $row $author $linentag($row) $m \
2307 [$canv2 itemcget $linentag($row) -font] $row
2308 }
2309 }
2310}
2311
2312proc vrel_change {name ix op} {
2313 global highlight_related
2314
2315 rhighlight_none
2316 if {$highlight_related ne "None"} {
2317 run drawvisible
2318 }
2319}
2320
2321# prepare for testing whether commits are descendents or ancestors of a
2322proc rhighlight_sel {a} {
2323 global descendent desc_todo ancestor anc_todo
2324 global highlight_related rhighlights
2325
2326 catch {unset descendent}
2327 set desc_todo [list $a]
2328 catch {unset ancestor}
2329 set anc_todo [list $a]
2330 if {$highlight_related ne "None"} {
2331 rhighlight_none
2332 run drawvisible
2333 }
2334}
2335
2336proc rhighlight_none {} {
2337 global rhighlights
2338
2339 catch {unset rhighlights}
2340 unbolden
2341}
2342
2343proc is_descendent {a} {
2344 global curview children commitrow descendent desc_todo
2345
2346 set v $curview
2347 set la $commitrow($v,$a)
2348 set todo $desc_todo
2349 set leftover {}
2350 set done 0
2351 for {set i 0} {$i < [llength $todo]} {incr i} {
2352 set do [lindex $todo $i]
2353 if {$commitrow($v,$do) < $la} {
2354 lappend leftover $do
2355 continue
2356 }
2357 foreach nk $children($v,$do) {
2358 if {![info exists descendent($nk)]} {
2359 set descendent($nk) 1
2360 lappend todo $nk
2361 if {$nk eq $a} {
2362 set done 1
2363 }
2364 }
2365 }
2366 if {$done} {
2367 set desc_todo [concat $leftover [lrange $todo [expr {$i+1}] end]]
2368 return
2369 }
2370 }
2371 set descendent($a) 0
2372 set desc_todo $leftover
2373}
2374
2375proc is_ancestor {a} {
2376 global curview parentlist commitrow ancestor anc_todo
2377
2378 set v $curview
2379 set la $commitrow($v,$a)
2380 set todo $anc_todo
2381 set leftover {}
2382 set done 0
2383 for {set i 0} {$i < [llength $todo]} {incr i} {
2384 set do [lindex $todo $i]
2385 if {![info exists commitrow($v,$do)] || $commitrow($v,$do) > $la} {
2386 lappend leftover $do
2387 continue
2388 }
2389 foreach np [lindex $parentlist $commitrow($v,$do)] {
2390 if {![info exists ancestor($np)]} {
2391 set ancestor($np) 1
2392 lappend todo $np
2393 if {$np eq $a} {
2394 set done 1
2395 }
2396 }
2397 }
2398 if {$done} {
2399 set anc_todo [concat $leftover [lrange $todo [expr {$i+1}] end]]
2400 return
2401 }
2402 }
2403 set ancestor($a) 0
2404 set anc_todo $leftover
2405}
2406
2407proc askrelhighlight {row id} {
2408 global descendent highlight_related iddrawn mainfont rhighlights
2409 global selectedline ancestor
2410
2411 if {![info exists selectedline]} return
2412 set isbold 0
2413 if {$highlight_related eq "Descendent" ||
2414 $highlight_related eq "Not descendent"} {
2415 if {![info exists descendent($id)]} {
2416 is_descendent $id
2417 }
2418 if {$descendent($id) == ($highlight_related eq "Descendent")} {
2419 set isbold 1
2420 }
2421 } elseif {$highlight_related eq "Ancestor" ||
2422 $highlight_related eq "Not ancestor"} {
2423 if {![info exists ancestor($id)]} {
2424 is_ancestor $id
2425 }
2426 if {$ancestor($id) == ($highlight_related eq "Ancestor")} {
2427 set isbold 1
2428 }
2429 }
2430 if {[info exists iddrawn($id)]} {
2431 if {$isbold && ![ishighlighted $row]} {
2432 bolden $row [concat $mainfont bold]
2433 }
2434 }
2435 set rhighlights($row) $isbold
2436}
2437
2438proc next_hlcont {} {
2439 global fhl_row fhl_dirn displayorder numcommits
2440 global vhighlights fhighlights nhighlights rhighlights
2441 global hlview filehighlight findstring highlight_related
2442
2443 if {![info exists fhl_dirn] || $fhl_dirn == 0} return
2444 set row $fhl_row
2445 while {1} {
2446 if {$row < 0 || $row >= $numcommits} {
2447 bell
2448 set fhl_dirn 0
2449 return
2450 }
2451 set id [lindex $displayorder $row]
2452 if {[info exists hlview]} {
2453 if {![info exists vhighlights($row)]} {
2454 askvhighlight $row $id
2455 }
2456 if {$vhighlights($row) > 0} break
2457 }
2458 if {$findstring ne {}} {
2459 if {![info exists nhighlights($row)]} {
2460 askfindhighlight $row $id
2461 }
2462 if {$nhighlights($row) > 0} break
2463 }
2464 if {$highlight_related ne "None"} {
2465 if {![info exists rhighlights($row)]} {
2466 askrelhighlight $row $id
2467 }
2468 if {$rhighlights($row) > 0} break
2469 }
2470 if {[info exists filehighlight]} {
2471 if {![info exists fhighlights($row)]} {
2472 # ask for a few more while we're at it...
2473 set r $row
2474 for {set n 0} {$n < 100} {incr n} {
2475 if {![info exists fhighlights($r)]} {
2476 askfilehighlight $r [lindex $displayorder $r]
2477 }
2478 incr r $fhl_dirn
2479 if {$r < 0 || $r >= $numcommits} break
2480 }
2481 flushhighlights
2482 }
2483 if {$fhighlights($row) < 0} {
2484 set fhl_row $row
2485 return
2486 }
2487 if {$fhighlights($row) > 0} break
2488 }
2489 incr row $fhl_dirn
2490 }
2491 set fhl_dirn 0
2492 selectline $row 1
2493}
2494
2495proc next_highlight {dirn} {
2496 global selectedline fhl_row fhl_dirn
2497 global hlview filehighlight findstring highlight_related
2498
2499 if {![info exists selectedline]} return
2500 if {!([info exists hlview] || $findstring ne {} ||
2501 $highlight_related ne "None" || [info exists filehighlight])} return
2502 set fhl_row [expr {$selectedline + $dirn}]
2503 set fhl_dirn $dirn
2504 next_hlcont
2505}
2506
2507proc cancel_next_highlight {} {
2508 global fhl_dirn
2509
2510 set fhl_dirn 0
2511}
2512
2513# Graph layout functions
2514
2515proc shortids {ids} {
2516 set res {}
2517 foreach id $ids {
2518 if {[llength $id] > 1} {
2519 lappend res [shortids $id]
2520 } elseif {[regexp {^[0-9a-f]{40}$} $id]} {
2521 lappend res [string range $id 0 7]
2522 } else {
2523 lappend res $id
2524 }
2525 }
2526 return $res
2527}
2528
2529proc incrange {l x o} {
2530 set n [llength $l]
2531 while {$x < $n} {
2532 set e [lindex $l $x]
2533 if {$e ne {}} {
2534 lset l $x [expr {$e + $o}]
2535 }
2536 incr x
2537 }
2538 return $l
2539}
2540
2541proc ntimes {n o} {
2542 set ret {}
2543 for {} {$n > 0} {incr n -1} {
2544 lappend ret $o
2545 }
2546 return $ret
2547}
2548
2549proc usedinrange {id l1 l2} {
2550 global children commitrow curview
2551
2552 if {[info exists commitrow($curview,$id)]} {
2553 set r $commitrow($curview,$id)
2554 if {$l1 <= $r && $r <= $l2} {
2555 return [expr {$r - $l1 + 1}]
2556 }
2557 }
2558 set kids $children($curview,$id)
2559 foreach c $kids {
2560 set r $commitrow($curview,$c)
2561 if {$l1 <= $r && $r <= $l2} {
2562 return [expr {$r - $l1 + 1}]
2563 }
2564 }
2565 return 0
2566}
2567
2568proc sanity {row {full 0}} {
2569 global rowidlist rowoffsets
2570
2571 set col -1
2572 set ids [lindex $rowidlist $row]
2573 foreach id $ids {
2574 incr col
2575 if {$id eq {}} continue
2576 if {$col < [llength $ids] - 1 &&
2577 [lsearch -exact -start [expr {$col+1}] $ids $id] >= 0} {
2578 puts "oops: [shortids $id] repeated in row $row col $col: {[shortids [lindex $rowidlist $row]]}"
2579 }
2580 set o [lindex $rowoffsets $row $col]
2581 set y $row
2582 set x $col
2583 while {$o ne {}} {
2584 incr y -1
2585 incr x $o
2586 if {[lindex $rowidlist $y $x] != $id} {
2587 puts "oops: rowoffsets wrong at row [expr {$y+1}] col [expr {$x-$o}]"
2588 puts " id=[shortids $id] check started at row $row"
2589 for {set i $row} {$i >= $y} {incr i -1} {
2590 puts " row $i ids={[shortids [lindex $rowidlist $i]]} offs={[lindex $rowoffsets $i]}"
2591 }
2592 break
2593 }
2594 if {!$full} break
2595 set o [lindex $rowoffsets $y $x]
2596 }
2597 }
2598}
2599
2600proc makeuparrow {oid x y z} {
2601 global rowidlist rowoffsets uparrowlen idrowranges displayorder
2602
2603 for {set i 1} {$i < $uparrowlen && $y > 1} {incr i} {
2604 incr y -1
2605 incr x $z
2606 set off0 [lindex $rowoffsets $y]
2607 for {set x0 $x} {1} {incr x0} {
2608 if {$x0 >= [llength $off0]} {
2609 set x0 [llength [lindex $rowoffsets [expr {$y-1}]]]
2610 break
2611 }
2612 set z [lindex $off0 $x0]
2613 if {$z ne {}} {
2614 incr x0 $z
2615 break
2616 }
2617 }
2618 set z [expr {$x0 - $x}]
2619 lset rowidlist $y [linsert [lindex $rowidlist $y] $x $oid]
2620 lset rowoffsets $y [linsert [lindex $rowoffsets $y] $x $z]
2621 }
2622 set tmp [lreplace [lindex $rowoffsets $y] $x $x {}]
2623 lset rowoffsets $y [incrange $tmp [expr {$x+1}] -1]
2624 lappend idrowranges($oid) [lindex $displayorder $y]
2625}
2626
2627proc initlayout {} {
2628 global rowidlist rowoffsets displayorder commitlisted
2629 global rowlaidout rowoptim
2630 global idinlist rowchk rowrangelist idrowranges
2631 global numcommits canvxmax canv
2632 global nextcolor
2633 global parentlist
2634 global colormap rowtextx
2635 global selectfirst
2636
2637 set numcommits 0
2638 set displayorder {}
2639 set commitlisted {}
2640 set parentlist {}
2641 set rowrangelist {}
2642 set nextcolor 0
2643 set rowidlist {{}}
2644 set rowoffsets {{}}
2645 catch {unset idinlist}
2646 catch {unset rowchk}
2647 set rowlaidout 0
2648 set rowoptim 0
2649 set canvxmax [$canv cget -width]
2650 catch {unset colormap}
2651 catch {unset rowtextx}
2652 catch {unset idrowranges}
2653 set selectfirst 1
2654}
2655
2656proc setcanvscroll {} {
2657 global canv canv2 canv3 numcommits linespc canvxmax canvy0
2658
2659 set ymax [expr {$canvy0 + ($numcommits - 0.5) * $linespc + 2}]
2660 $canv conf -scrollregion [list 0 0 $canvxmax $ymax]
2661 $canv2 conf -scrollregion [list 0 0 0 $ymax]
2662 $canv3 conf -scrollregion [list 0 0 0 $ymax]
2663}
2664
2665proc visiblerows {} {
2666 global canv numcommits linespc
2667
2668 set ymax [lindex [$canv cget -scrollregion] 3]
2669 if {$ymax eq {} || $ymax == 0} return
2670 set f [$canv yview]
2671 set y0 [expr {int([lindex $f 0] * $ymax)}]
2672 set r0 [expr {int(($y0 - 3) / $linespc) - 1}]
2673 if {$r0 < 0} {
2674 set r0 0
2675 }
2676 set y1 [expr {int([lindex $f 1] * $ymax)}]
2677 set r1 [expr {int(($y1 - 3) / $linespc) + 1}]
2678 if {$r1 >= $numcommits} {
2679 set r1 [expr {$numcommits - 1}]
2680 }
2681 return [list $r0 $r1]
2682}
2683
2684proc layoutmore {tmax allread} {
2685 global rowlaidout rowoptim commitidx numcommits optim_delay
2686 global uparrowlen curview rowidlist idinlist
2687
2688 set showlast 0
2689 set showdelay $optim_delay
2690 set optdelay [expr {$uparrowlen + 1}]
2691 while {1} {
2692 if {$rowoptim - $showdelay > $numcommits} {
2693 showstuff [expr {$rowoptim - $showdelay}] $showlast
2694 } elseif {$rowlaidout - $optdelay > $rowoptim} {
2695 set nr [expr {$rowlaidout - $optdelay - $rowoptim}]
2696 if {$nr > 100} {
2697 set nr 100
2698 }
2699 optimize_rows $rowoptim 0 [expr {$rowoptim + $nr}]
2700 incr rowoptim $nr
2701 } elseif {$commitidx($curview) > $rowlaidout} {
2702 set nr [expr {$commitidx($curview) - $rowlaidout}]
2703 # may need to increase this threshold if uparrowlen or
2704 # mingaplen are increased...
2705 if {$nr > 150} {
2706 set nr 150
2707 }
2708 set row $rowlaidout
2709 set rowlaidout [layoutrows $row [expr {$row + $nr}] $allread]
2710 if {$rowlaidout == $row} {
2711 return 0
2712 }
2713 } elseif {$allread} {
2714 set optdelay 0
2715 set nrows $commitidx($curview)
2716 if {[lindex $rowidlist $nrows] ne {} ||
2717 [array names idinlist] ne {}} {
2718 layouttail
2719 set rowlaidout $commitidx($curview)
2720 } elseif {$rowoptim == $nrows} {
2721 set showdelay 0
2722 set showlast 1
2723 if {$numcommits == $nrows} {
2724 return 0
2725 }
2726 }
2727 } else {
2728 return 0
2729 }
2730 if {$tmax ne {} && [clock clicks -milliseconds] >= $tmax} {
2731 return 1
2732 }
2733 }
2734}
2735
2736proc showstuff {canshow last} {
2737 global numcommits commitrow pending_select selectedline curview
2738 global lookingforhead mainheadid displayorder selectfirst
2739 global lastscrollset
2740
2741 if {$numcommits == 0} {
2742 global phase
2743 set phase "incrdraw"
2744 allcanvs delete all
2745 }
2746 set r0 $numcommits
2747 set prev $numcommits
2748 set numcommits $canshow
2749 set t [clock clicks -milliseconds]
2750 if {$prev < 100 || $last || $t - $lastscrollset > 500} {
2751 set lastscrollset $t
2752 setcanvscroll
2753 }
2754 set rows [visiblerows]
2755 set r1 [lindex $rows 1]
2756 if {$r1 >= $canshow} {
2757 set r1 [expr {$canshow - 1}]
2758 }
2759 if {$r0 <= $r1} {
2760 drawcommits $r0 $r1
2761 }
2762 if {[info exists pending_select] &&
2763 [info exists commitrow($curview,$pending_select)] &&
2764 $commitrow($curview,$pending_select) < $numcommits} {
2765 selectline $commitrow($curview,$pending_select) 1
2766 }
2767 if {$selectfirst} {
2768 if {[info exists selectedline] || [info exists pending_select]} {
2769 set selectfirst 0
2770 } else {
2771 set l [first_real_row]
2772 selectline $l 1
2773 set selectfirst 0
2774 }
2775 }
2776 if {$lookingforhead && [info exists commitrow($curview,$mainheadid)]
2777 && ($last || $commitrow($curview,$mainheadid) < $numcommits - 1)} {
2778 set lookingforhead 0
2779 dodiffindex
2780 }
2781}
2782
2783proc doshowlocalchanges {} {
2784 global lookingforhead curview mainheadid phase commitrow
2785
2786 if {[info exists commitrow($curview,$mainheadid)] &&
2787 ($phase eq {} || $commitrow($curview,$mainheadid) < $numcommits - 1)} {
2788 dodiffindex
2789 } elseif {$phase ne {}} {
2790 set lookingforhead 1
2791 }
2792}
2793
2794proc dohidelocalchanges {} {
2795 global lookingforhead localfrow localirow lserial
2796
2797 set lookingforhead 0
2798 if {$localfrow >= 0} {
2799 removerow $localfrow
2800 set localfrow -1
2801 if {$localirow > 0} {
2802 incr localirow -1
2803 }
2804 }
2805 if {$localirow >= 0} {
2806 removerow $localirow
2807 set localirow -1
2808 }
2809 incr lserial
2810}
2811
2812# spawn off a process to do git diff-index --cached HEAD
2813proc dodiffindex {} {
2814 global localirow localfrow lserial
2815
2816 incr lserial
2817 set localfrow -1
2818 set localirow -1
2819 set fd [open "|git diff-index --cached HEAD" r]
2820 fconfigure $fd -blocking 0
2821 filerun $fd [list readdiffindex $fd $lserial]
2822}
2823
2824proc readdiffindex {fd serial} {
2825 global localirow commitrow mainheadid nullid2 curview
2826 global commitinfo commitdata lserial
2827
2828 set isdiff 1
2829 if {[gets $fd line] < 0} {
2830 if {![eof $fd]} {
2831 return 1
2832 }
2833 set isdiff 0
2834 }
2835 # we only need to see one line and we don't really care what it says...
2836 close $fd
2837
2838 # now see if there are any local changes not checked in to the index
2839 if {$serial == $lserial} {
2840 set fd [open "|git diff-files" r]
2841 fconfigure $fd -blocking 0
2842 filerun $fd [list readdifffiles $fd $serial]
2843 }
2844
2845 if {$isdiff && $serial == $lserial && $localirow == -1} {
2846 # add the line for the changes in the index to the graph
2847 set localirow $commitrow($curview,$mainheadid)
2848 set hl "Local changes checked in to index but not committed"
2849 set commitinfo($nullid2) [list $hl {} {} {} {} " $hl\n"]
2850 set commitdata($nullid2) "\n $hl\n"
2851 insertrow $localirow $nullid2
2852 }
2853 return 0
2854}
2855
2856proc readdifffiles {fd serial} {
2857 global localirow localfrow commitrow mainheadid nullid curview
2858 global commitinfo commitdata lserial
2859
2860 set isdiff 1
2861 if {[gets $fd line] < 0} {
2862 if {![eof $fd]} {
2863 return 1
2864 }
2865 set isdiff 0
2866 }
2867 # we only need to see one line and we don't really care what it says...
2868 close $fd
2869
2870 if {$isdiff && $serial == $lserial && $localfrow == -1} {
2871 # add the line for the local diff to the graph
2872 if {$localirow >= 0} {
2873 set localfrow $localirow
2874 incr localirow
2875 } else {
2876 set localfrow $commitrow($curview,$mainheadid)
2877 }
2878 set hl "Local uncommitted changes, not checked in to index"
2879 set commitinfo($nullid) [list $hl {} {} {} {} " $hl\n"]
2880 set commitdata($nullid) "\n $hl\n"
2881 insertrow $localfrow $nullid
2882 }
2883 return 0
2884}
2885
2886proc layoutrows {row endrow last} {
2887 global rowidlist rowoffsets displayorder
2888 global uparrowlen downarrowlen maxwidth mingaplen
2889 global children parentlist
2890 global idrowranges
2891 global commitidx curview
2892 global idinlist rowchk rowrangelist
2893
2894 set idlist [lindex $rowidlist $row]
2895 set offs [lindex $rowoffsets $row]
2896 while {$row < $endrow} {
2897 set id [lindex $displayorder $row]
2898 set oldolds {}
2899 set newolds {}
2900 foreach p [lindex $parentlist $row] {
2901 if {![info exists idinlist($p)]} {
2902 lappend newolds $p
2903 } elseif {!$idinlist($p)} {
2904 lappend oldolds $p
2905 }
2906 set idinlist($p) 1
2907 }
2908 set nev [expr {[llength $idlist] + [llength $newolds]
2909 + [llength $oldolds] - $maxwidth + 1}]
2910 if {$nev > 0} {
2911 if {!$last &&
2912 $row + $uparrowlen + $mingaplen >= $commitidx($curview)} break
2913 for {set x [llength $idlist]} {[incr x -1] >= 0} {} {
2914 set i [lindex $idlist $x]
2915 if {![info exists rowchk($i)] || $row >= $rowchk($i)} {
2916 set r [usedinrange $i [expr {$row - $downarrowlen}] \
2917 [expr {$row + $uparrowlen + $mingaplen}]]
2918 if {$r == 0} {
2919 set idlist [lreplace $idlist $x $x]
2920 set offs [lreplace $offs $x $x]
2921 set offs [incrange $offs $x 1]
2922 set idinlist($i) 0
2923 set rm1 [expr {$row - 1}]
2924 lappend idrowranges($i) [lindex $displayorder $rm1]
2925 if {[incr nev -1] <= 0} break
2926 continue
2927 }
2928 set rowchk($id) [expr {$row + $r}]
2929 }
2930 }
2931 lset rowidlist $row $idlist
2932 lset rowoffsets $row $offs
2933 }
2934 set col [lsearch -exact $idlist $id]
2935 if {$col < 0} {
2936 set col [llength $idlist]
2937 lappend idlist $id
2938 lset rowidlist $row $idlist
2939 set z {}
2940 if {$children($curview,$id) ne {}} {
2941 set z [expr {[llength [lindex $rowidlist [expr {$row-1}]]] - $col}]
2942 unset idinlist($id)
2943 }
2944 lappend offs $z
2945 lset rowoffsets $row $offs
2946 if {$z ne {}} {
2947 makeuparrow $id $col $row $z
2948 }
2949 } else {
2950 unset idinlist($id)
2951 }
2952 set ranges {}
2953 if {[info exists idrowranges($id)]} {
2954 set ranges $idrowranges($id)
2955 lappend ranges $id
2956 unset idrowranges($id)
2957 }
2958 lappend rowrangelist $ranges
2959 incr row
2960 set offs [ntimes [llength $idlist] 0]
2961 set l [llength $newolds]
2962 set idlist [eval lreplace \$idlist $col $col $newolds]
2963 set o 0
2964 if {$l != 1} {
2965 set offs [lrange $offs 0 [expr {$col - 1}]]
2966 foreach x $newolds {
2967 lappend offs {}
2968 incr o -1
2969 }
2970 incr o
2971 set tmp [expr {[llength $idlist] - [llength $offs]}]
2972 if {$tmp > 0} {
2973 set offs [concat $offs [ntimes $tmp $o]]
2974 }
2975 } else {
2976 lset offs $col {}
2977 }
2978 foreach i $newolds {
2979 set idrowranges($i) $id
2980 }
2981 incr col $l
2982 foreach oid $oldolds {
2983 set idlist [linsert $idlist $col $oid]
2984 set offs [linsert $offs $col $o]
2985 makeuparrow $oid $col $row $o
2986 incr col
2987 }
2988 lappend rowidlist $idlist
2989 lappend rowoffsets $offs
2990 }
2991 return $row
2992}
2993
2994proc addextraid {id row} {
2995 global displayorder commitrow commitinfo
2996 global commitidx commitlisted
2997 global parentlist children curview
2998
2999 incr commitidx($curview)
3000 lappend displayorder $id
3001 lappend commitlisted 0
3002 lappend parentlist {}
3003 set commitrow($curview,$id) $row
3004 readcommit $id
3005 if {![info exists commitinfo($id)]} {
3006 set commitinfo($id) {"No commit information available"}
3007 }
3008 if {![info exists children($curview,$id)]} {
3009 set children($curview,$id) {}
3010 }
3011}
3012
3013proc layouttail {} {
3014 global rowidlist rowoffsets idinlist commitidx curview
3015 global idrowranges rowrangelist
3016
3017 set row $commitidx($curview)
3018 set idlist [lindex $rowidlist $row]
3019 while {$idlist ne {}} {
3020 set col [expr {[llength $idlist] - 1}]
3021 set id [lindex $idlist $col]
3022 addextraid $id $row
3023 catch {unset idinlist($id)}
3024 lappend idrowranges($id) $id
3025 lappend rowrangelist $idrowranges($id)
3026 unset idrowranges($id)
3027 incr row
3028 set offs [ntimes $col 0]
3029 set idlist [lreplace $idlist $col $col]
3030 lappend rowidlist $idlist
3031 lappend rowoffsets $offs
3032 }
3033
3034 foreach id [array names idinlist] {
3035 unset idinlist($id)
3036 addextraid $id $row
3037 lset rowidlist $row [list $id]
3038 lset rowoffsets $row 0
3039 makeuparrow $id 0 $row 0
3040 lappend idrowranges($id) $id
3041 lappend rowrangelist $idrowranges($id)
3042 unset idrowranges($id)
3043 incr row
3044 lappend rowidlist {}
3045 lappend rowoffsets {}
3046 }
3047}
3048
3049proc insert_pad {row col npad} {
3050 global rowidlist rowoffsets
3051
3052 set pad [ntimes $npad {}]
3053 lset rowidlist $row [eval linsert [list [lindex $rowidlist $row]] $col $pad]
3054 set tmp [eval linsert [list [lindex $rowoffsets $row]] $col $pad]
3055 lset rowoffsets $row [incrange $tmp [expr {$col + $npad}] [expr {-$npad}]]
3056}
3057
3058proc optimize_rows {row col endrow} {
3059 global rowidlist rowoffsets displayorder
3060
3061 for {} {$row < $endrow} {incr row} {
3062 set idlist [lindex $rowidlist $row]
3063 set offs [lindex $rowoffsets $row]
3064 set haspad 0
3065 for {} {$col < [llength $offs]} {incr col} {
3066 if {[lindex $idlist $col] eq {}} {
3067 set haspad 1
3068 continue
3069 }
3070 set z [lindex $offs $col]
3071 if {$z eq {}} continue
3072 set isarrow 0
3073 set x0 [expr {$col + $z}]
3074 set y0 [expr {$row - 1}]
3075 set z0 [lindex $rowoffsets $y0 $x0]
3076 if {$z0 eq {}} {
3077 set id [lindex $idlist $col]
3078 set ranges [rowranges $id]
3079 if {$ranges ne {} && $y0 > [lindex $ranges 0]} {
3080 set isarrow 1
3081 }
3082 }
3083 # Looking at lines from this row to the previous row,
3084 # make them go straight up if they end in an arrow on
3085 # the previous row; otherwise make them go straight up
3086 # or at 45 degrees.
3087 if {$z < -1 || ($z < 0 && $isarrow)} {
3088 # Line currently goes left too much;
3089 # insert pads in the previous row, then optimize it
3090 set npad [expr {-1 - $z + $isarrow}]
3091 set offs [incrange $offs $col $npad]
3092 insert_pad $y0 $x0 $npad
3093 if {$y0 > 0} {
3094 optimize_rows $y0 $x0 $row
3095 }
3096 set z [lindex $offs $col]
3097 set x0 [expr {$col + $z}]
3098 set z0 [lindex $rowoffsets $y0 $x0]
3099 } elseif {$z > 1 || ($z > 0 && $isarrow)} {
3100 # Line currently goes right too much;
3101 # insert pads in this line and adjust the next's rowoffsets
3102 set npad [expr {$z - 1 + $isarrow}]
3103 set y1 [expr {$row + 1}]
3104 set offs2 [lindex $rowoffsets $y1]
3105 set x1 -1
3106 foreach z $offs2 {
3107 incr x1
3108 if {$z eq {} || $x1 + $z < $col} continue
3109 if {$x1 + $z > $col} {
3110 incr npad
3111 }
3112 lset rowoffsets $y1 [incrange $offs2 $x1 $npad]
3113 break
3114 }
3115 set pad [ntimes $npad {}]
3116 set idlist [eval linsert \$idlist $col $pad]
3117 set tmp [eval linsert \$offs $col $pad]
3118 incr col $npad
3119 set offs [incrange $tmp $col [expr {-$npad}]]
3120 set z [lindex $offs $col]
3121 set haspad 1
3122 }
3123 if {$z0 eq {} && !$isarrow} {
3124 # this line links to its first child on row $row-2
3125 set rm2 [expr {$row - 2}]
3126 set id [lindex $displayorder $rm2]
3127 set xc [lsearch -exact [lindex $rowidlist $rm2] $id]
3128 if {$xc >= 0} {
3129 set z0 [expr {$xc - $x0}]
3130 }
3131 }
3132 # avoid lines jigging left then immediately right
3133 if {$z0 ne {} && $z < 0 && $z0 > 0} {
3134 insert_pad $y0 $x0 1
3135 set offs [incrange $offs $col 1]
3136 optimize_rows $y0 [expr {$x0 + 1}] $row
3137 }
3138 }
3139 if {!$haspad} {
3140 set o {}
3141 # Find the first column that doesn't have a line going right
3142 for {set col [llength $idlist]} {[incr col -1] >= 0} {} {
3143 set o [lindex $offs $col]
3144 if {$o eq {}} {
3145 # check if this is the link to the first child
3146 set id [lindex $idlist $col]
3147 set ranges [rowranges $id]
3148 if {$ranges ne {} && $row == [lindex $ranges 0]} {
3149 # it is, work out offset to child
3150 set y0 [expr {$row - 1}]
3151 set id [lindex $displayorder $y0]
3152 set x0 [lsearch -exact [lindex $rowidlist $y0] $id]
3153 if {$x0 >= 0} {
3154 set o [expr {$x0 - $col}]
3155 }
3156 }
3157 }
3158 if {$o eq {} || $o <= 0} break
3159 }
3160 # Insert a pad at that column as long as it has a line and
3161 # isn't the last column, and adjust the next row' offsets
3162 if {$o ne {} && [incr col] < [llength $idlist]} {
3163 set y1 [expr {$row + 1}]
3164 set offs2 [lindex $rowoffsets $y1]
3165 set x1 -1
3166 foreach z $offs2 {
3167 incr x1
3168 if {$z eq {} || $x1 + $z < $col} continue
3169 lset rowoffsets $y1 [incrange $offs2 $x1 1]
3170 break
3171 }
3172 set idlist [linsert $idlist $col {}]
3173 set tmp [linsert $offs $col {}]
3174 incr col
3175 set offs [incrange $tmp $col -1]
3176 }
3177 }
3178 lset rowidlist $row $idlist
3179 lset rowoffsets $row $offs
3180 set col 0
3181 }
3182}
3183
3184proc xc {row col} {
3185 global canvx0 linespc
3186 return [expr {$canvx0 + $col * $linespc}]
3187}
3188
3189proc yc {row} {
3190 global canvy0 linespc
3191 return [expr {$canvy0 + $row * $linespc}]
3192}
3193
3194proc linewidth {id} {
3195 global thickerline lthickness
3196
3197 set wid $lthickness
3198 if {[info exists thickerline] && $id eq $thickerline} {
3199 set wid [expr {2 * $lthickness}]
3200 }
3201 return $wid
3202}
3203
3204proc rowranges {id} {
3205 global phase idrowranges commitrow rowlaidout rowrangelist curview
3206
3207 set ranges {}
3208 if {$phase eq {} ||
3209 ([info exists commitrow($curview,$id)]
3210 && $commitrow($curview,$id) < $rowlaidout)} {
3211 set ranges [lindex $rowrangelist $commitrow($curview,$id)]
3212 } elseif {[info exists idrowranges($id)]} {
3213 set ranges $idrowranges($id)
3214 }
3215 set linenos {}
3216 foreach rid $ranges {
3217 lappend linenos $commitrow($curview,$rid)
3218 }
3219 if {$linenos ne {}} {
3220 lset linenos 0 [expr {[lindex $linenos 0] + 1}]
3221 }
3222 return $linenos
3223}
3224
3225# work around tk8.4 refusal to draw arrows on diagonal segments
3226proc adjarrowhigh {coords} {
3227 global linespc
3228
3229 set x0 [lindex $coords 0]
3230 set x1 [lindex $coords 2]
3231 if {$x0 != $x1} {
3232 set y0 [lindex $coords 1]
3233 set y1 [lindex $coords 3]
3234 if {$y0 - $y1 <= 2 * $linespc && $x1 == [lindex $coords 4]} {
3235 # we have a nearby vertical segment, just trim off the diag bit
3236 set coords [lrange $coords 2 end]
3237 } else {
3238 set slope [expr {($x0 - $x1) / ($y0 - $y1)}]
3239 set xi [expr {$x0 - $slope * $linespc / 2}]
3240 set yi [expr {$y0 - $linespc / 2}]
3241 set coords [lreplace $coords 0 1 $xi $y0 $xi $yi]
3242 }
3243 }
3244 return $coords
3245}
3246
3247proc drawlineseg {id row endrow arrowlow} {
3248 global rowidlist displayorder iddrawn linesegs
3249 global canv colormap linespc curview maxlinelen
3250
3251 set cols [list [lsearch -exact [lindex $rowidlist $row] $id]]
3252 set le [expr {$row + 1}]
3253 set arrowhigh 1
3254 while {1} {
3255 set c [lsearch -exact [lindex $rowidlist $le] $id]
3256 if {$c < 0} {
3257 incr le -1
3258 break
3259 }
3260 lappend cols $c
3261 set x [lindex $displayorder $le]
3262 if {$x eq $id} {
3263 set arrowhigh 0
3264 break
3265 }
3266 if {[info exists iddrawn($x)] || $le == $endrow} {
3267 set c [lsearch -exact [lindex $rowidlist [expr {$le+1}]] $id]
3268 if {$c >= 0} {
3269 lappend cols $c
3270 set arrowhigh 0
3271 }
3272 break
3273 }
3274 incr le
3275 }
3276 if {$le <= $row} {
3277 return $row
3278 }
3279
3280 set lines {}
3281 set i 0
3282 set joinhigh 0
3283 if {[info exists linesegs($id)]} {
3284 set lines $linesegs($id)
3285 foreach li $lines {
3286 set r0 [lindex $li 0]
3287 if {$r0 > $row} {
3288 if {$r0 == $le && [lindex $li 1] - $row <= $maxlinelen} {
3289 set joinhigh 1
3290 }
3291 break
3292 }
3293 incr i
3294 }
3295 }
3296 set joinlow 0
3297 if {$i > 0} {
3298 set li [lindex $lines [expr {$i-1}]]
3299 set r1 [lindex $li 1]
3300 if {$r1 == $row && $le - [lindex $li 0] <= $maxlinelen} {
3301 set joinlow 1
3302 }
3303 }
3304
3305 set x [lindex $cols [expr {$le - $row}]]
3306 set xp [lindex $cols [expr {$le - 1 - $row}]]
3307 set dir [expr {$xp - $x}]
3308 if {$joinhigh} {
3309 set ith [lindex $lines $i 2]
3310 set coords [$canv coords $ith]
3311 set ah [$canv itemcget $ith -arrow]
3312 set arrowhigh [expr {$ah eq "first" || $ah eq "both"}]
3313 set x2 [lindex $cols [expr {$le + 1 - $row}]]
3314 if {$x2 ne {} && $x - $x2 == $dir} {
3315 set coords [lrange $coords 0 end-2]
3316 }
3317 } else {
3318 set coords [list [xc $le $x] [yc $le]]
3319 }
3320 if {$joinlow} {
3321 set itl [lindex $lines [expr {$i-1}] 2]
3322 set al [$canv itemcget $itl -arrow]
3323 set arrowlow [expr {$al eq "last" || $al eq "both"}]
3324 } elseif {$arrowlow &&
3325 [lsearch -exact [lindex $rowidlist [expr {$row-1}]] $id] >= 0} {
3326 set arrowlow 0
3327 }
3328 set arrow [lindex {none first last both} [expr {$arrowhigh + 2*$arrowlow}]]
3329 for {set y $le} {[incr y -1] > $row} {} {
3330 set x $xp
3331 set xp [lindex $cols [expr {$y - 1 - $row}]]
3332 set ndir [expr {$xp - $x}]
3333 if {$dir != $ndir || $xp < 0} {
3334 lappend coords [xc $y $x] [yc $y]
3335 }
3336 set dir $ndir
3337 }
3338 if {!$joinlow} {
3339 if {$xp < 0} {
3340 # join parent line to first child
3341 set ch [lindex $displayorder $row]
3342 set xc [lsearch -exact [lindex $rowidlist $row] $ch]
3343 if {$xc < 0} {
3344 puts "oops: drawlineseg: child $ch not on row $row"
3345 } else {
3346 if {$xc < $x - 1} {
3347 lappend coords [xc $row [expr {$x-1}]] [yc $row]
3348 } elseif {$xc > $x + 1} {
3349 lappend coords [xc $row [expr {$x+1}]] [yc $row]
3350 }
3351 set x $xc
3352 }
3353 lappend coords [xc $row $x] [yc $row]
3354 } else {
3355 set xn [xc $row $xp]
3356 set yn [yc $row]
3357 # work around tk8.4 refusal to draw arrows on diagonal segments
3358 if {$arrowlow && $xn != [lindex $coords end-1]} {
3359 if {[llength $coords] < 4 ||
3360 [lindex $coords end-3] != [lindex $coords end-1] ||
3361 [lindex $coords end] - $yn > 2 * $linespc} {
3362 set xn [xc $row [expr {$xp - 0.5 * $dir}]]
3363 set yo [yc [expr {$row + 0.5}]]
3364 lappend coords $xn $yo $xn $yn
3365 }
3366 } else {
3367 lappend coords $xn $yn
3368 }
3369 }
3370 if {!$joinhigh} {
3371 if {$arrowhigh} {
3372 set coords [adjarrowhigh $coords]
3373 }
3374 assigncolor $id
3375 set t [$canv create line $coords -width [linewidth $id] \
3376 -fill $colormap($id) -tags lines.$id -arrow $arrow]
3377 $canv lower $t
3378 bindline $t $id
3379 set lines [linsert $lines $i [list $row $le $t]]
3380 } else {
3381 $canv coords $ith $coords
3382 if {$arrow ne $ah} {
3383 $canv itemconf $ith -arrow $arrow
3384 }
3385 lset lines $i 0 $row
3386 }
3387 } else {
3388 set xo [lsearch -exact [lindex $rowidlist [expr {$row - 1}]] $id]
3389 set ndir [expr {$xo - $xp}]
3390 set clow [$canv coords $itl]
3391 if {$dir == $ndir} {
3392 set clow [lrange $clow 2 end]
3393 }
3394 set coords [concat $coords $clow]
3395 if {!$joinhigh} {
3396 lset lines [expr {$i-1}] 1 $le
3397 if {$arrowhigh} {
3398 set coords [adjarrowhigh $coords]
3399 }
3400 } else {
3401 # coalesce two pieces
3402 $canv delete $ith
3403 set b [lindex $lines [expr {$i-1}] 0]
3404 set e [lindex $lines $i 1]
3405 set lines [lreplace $lines [expr {$i-1}] $i [list $b $e $itl]]
3406 }
3407 $canv coords $itl $coords
3408 if {$arrow ne $al} {
3409 $canv itemconf $itl -arrow $arrow
3410 }
3411 }
3412
3413 set linesegs($id) $lines
3414 return $le
3415}
3416
3417proc drawparentlinks {id row} {
3418 global rowidlist canv colormap curview parentlist
3419 global idpos
3420
3421 set rowids [lindex $rowidlist $row]
3422 set col [lsearch -exact $rowids $id]
3423 if {$col < 0} return
3424 set olds [lindex $parentlist $row]
3425 set row2 [expr {$row + 1}]
3426 set x [xc $row $col]
3427 set y [yc $row]
3428 set y2 [yc $row2]
3429 set ids [lindex $rowidlist $row2]
3430 # rmx = right-most X coord used
3431 set rmx 0
3432 foreach p $olds {
3433 set i [lsearch -exact $ids $p]
3434 if {$i < 0} {
3435 puts "oops, parent $p of $id not in list"
3436 continue
3437 }
3438 set x2 [xc $row2 $i]
3439 if {$x2 > $rmx} {
3440 set rmx $x2
3441 }
3442 if {[lsearch -exact $rowids $p] < 0} {
3443 # drawlineseg will do this one for us
3444 continue
3445 }
3446 assigncolor $p
3447 # should handle duplicated parents here...
3448 set coords [list $x $y]
3449 if {$i < $col - 1} {
3450 lappend coords [xc $row [expr {$i + 1}]] $y
3451 } elseif {$i > $col + 1} {
3452 lappend coords [xc $row [expr {$i - 1}]] $y
3453 }
3454 lappend coords $x2 $y2
3455 set t [$canv create line $coords -width [linewidth $p] \
3456 -fill $colormap($p) -tags lines.$p]
3457 $canv lower $t
3458 bindline $t $p
3459 }
3460 if {$rmx > [lindex $idpos($id) 1]} {
3461 lset idpos($id) 1 $rmx
3462 redrawtags $id
3463 }
3464}
3465
3466proc drawlines {id} {
3467 global canv
3468
3469 $canv itemconf lines.$id -width [linewidth $id]
3470}
3471
3472proc drawcmittext {id row col} {
3473 global linespc canv canv2 canv3 canvy0 fgcolor curview
3474 global commitlisted commitinfo rowidlist parentlist
3475 global rowtextx idpos idtags idheads idotherrefs
3476 global linehtag linentag linedtag
3477 global mainfont canvxmax boldrows boldnamerows fgcolor nullid nullid2
3478
3479 # listed is 0 for boundary, 1 for normal, 2 for left, 3 for right
3480 set listed [lindex $commitlisted $row]
3481 if {$id eq $nullid} {
3482 set ofill red
3483 } elseif {$id eq $nullid2} {
3484 set ofill green
3485 } else {
3486 set ofill [expr {$listed != 0? "blue": "white"}]
3487 }
3488 set x [xc $row $col]
3489 set y [yc $row]
3490 set orad [expr {$linespc / 3}]
3491 if {$listed <= 1} {
3492 set t [$canv create oval [expr {$x - $orad}] [expr {$y - $orad}] \
3493 [expr {$x + $orad - 1}] [expr {$y + $orad - 1}] \
3494 -fill $ofill -outline $fgcolor -width 1 -tags circle]
3495 } elseif {$listed == 2} {
3496 # triangle pointing left for left-side commits
3497 set t [$canv create polygon \
3498 [expr {$x - $orad}] $y \
3499 [expr {$x + $orad - 1}] [expr {$y - $orad}] \
3500 [expr {$x + $orad - 1}] [expr {$y + $orad - 1}] \
3501 -fill $ofill -outline $fgcolor -width 1 -tags circle]
3502 } else {
3503 # triangle pointing right for right-side commits
3504 set t [$canv create polygon \
3505 [expr {$x + $orad - 1}] $y \
3506 [expr {$x - $orad}] [expr {$y - $orad}] \
3507 [expr {$x - $orad}] [expr {$y + $orad - 1}] \
3508 -fill $ofill -outline $fgcolor -width 1 -tags circle]
3509 }
3510 $canv raise $t
3511 $canv bind $t <1> {selcanvline {} %x %y}
3512 set rmx [llength [lindex $rowidlist $row]]
3513 set olds [lindex $parentlist $row]
3514 if {$olds ne {}} {
3515 set nextids [lindex $rowidlist [expr {$row + 1}]]
3516 foreach p $olds {
3517 set i [lsearch -exact $nextids $p]
3518 if {$i > $rmx} {
3519 set rmx $i
3520 }
3521 }
3522 }
3523 set xt [xc $row $rmx]
3524 set rowtextx($row) $xt
3525 set idpos($id) [list $x $xt $y]
3526 if {[info exists idtags($id)] || [info exists idheads($id)]
3527 || [info exists idotherrefs($id)]} {
3528 set xt [drawtags $id $x $xt $y]
3529 }
3530 set headline [lindex $commitinfo($id) 0]
3531 set name [lindex $commitinfo($id) 1]
3532 set date [lindex $commitinfo($id) 2]
3533 set date [formatdate $date]
3534 set font $mainfont
3535 set nfont $mainfont
3536 set isbold [ishighlighted $row]
3537 if {$isbold > 0} {
3538 lappend boldrows $row
3539 lappend font bold
3540 if {$isbold > 1} {
3541 lappend boldnamerows $row
3542 lappend nfont bold
3543 }
3544 }
3545 set linehtag($row) [$canv create text $xt $y -anchor w -fill $fgcolor \
3546 -text $headline -font $font -tags text]
3547 $canv bind $linehtag($row) <Button-3> "rowmenu %X %Y $id"
3548 set linentag($row) [$canv2 create text 3 $y -anchor w -fill $fgcolor \
3549 -text $name -font $nfont -tags text]
3550 set linedtag($row) [$canv3 create text 3 $y -anchor w -fill $fgcolor \
3551 -text $date -font $mainfont -tags text]
3552 set xr [expr {$xt + [font measure $mainfont $headline]}]
3553 if {$xr > $canvxmax} {
3554 set canvxmax $xr
3555 setcanvscroll
3556 }
3557}
3558
3559proc drawcmitrow {row} {
3560 global displayorder rowidlist
3561 global iddrawn markingmatches
3562 global commitinfo parentlist numcommits
3563 global filehighlight fhighlights findstring nhighlights
3564 global hlview vhighlights
3565 global highlight_related rhighlights
3566
3567 if {$row >= $numcommits} return
3568
3569 set id [lindex $displayorder $row]
3570 if {[info exists hlview] && ![info exists vhighlights($row)]} {
3571 askvhighlight $row $id
3572 }
3573 if {[info exists filehighlight] && ![info exists fhighlights($row)]} {
3574 askfilehighlight $row $id
3575 }
3576 if {$findstring ne {} && ![info exists nhighlights($row)]} {
3577 askfindhighlight $row $id
3578 }
3579 if {$highlight_related ne "None" && ![info exists rhighlights($row)]} {
3580 askrelhighlight $row $id
3581 }
3582 if {![info exists iddrawn($id)]} {
3583 set col [lsearch -exact [lindex $rowidlist $row] $id]
3584 if {$col < 0} {
3585 puts "oops, row $row id $id not in list"
3586 return
3587 }
3588 if {![info exists commitinfo($id)]} {
3589 getcommit $id
3590 }
3591 assigncolor $id
3592 drawcmittext $id $row $col
3593 set iddrawn($id) 1
3594 }
3595 if {$markingmatches} {
3596 markrowmatches $row $id
3597 }
3598}
3599
3600proc drawcommits {row {endrow {}}} {
3601 global numcommits iddrawn displayorder curview
3602 global parentlist rowidlist
3603
3604 if {$row < 0} {
3605 set row 0
3606 }
3607 if {$endrow eq {}} {
3608 set endrow $row
3609 }
3610 if {$endrow >= $numcommits} {
3611 set endrow [expr {$numcommits - 1}]
3612 }
3613
3614 # make the lines join to already-drawn rows either side
3615 set r [expr {$row - 1}]
3616 if {$r < 0 || ![info exists iddrawn([lindex $displayorder $r])]} {
3617 set r $row
3618 }
3619 set er [expr {$endrow + 1}]
3620 if {$er >= $numcommits ||
3621 ![info exists iddrawn([lindex $displayorder $er])]} {
3622 set er $endrow
3623 }
3624 for {} {$r <= $er} {incr r} {
3625 set id [lindex $displayorder $r]
3626 set wasdrawn [info exists iddrawn($id)]
3627 drawcmitrow $r
3628 if {$r == $er} break
3629 set nextid [lindex $displayorder [expr {$r + 1}]]
3630 if {$wasdrawn && [info exists iddrawn($nextid)]} {
3631 catch {unset prevlines}
3632 continue
3633 }
3634 drawparentlinks $id $r
3635
3636 if {[info exists lineends($r)]} {
3637 foreach lid $lineends($r) {
3638 unset prevlines($lid)
3639 }
3640 }
3641 set rowids [lindex $rowidlist $r]
3642 foreach lid $rowids {
3643 if {$lid eq {}} continue
3644 if {$lid eq $id} {
3645 # see if this is the first child of any of its parents
3646 foreach p [lindex $parentlist $r] {
3647 if {[lsearch -exact $rowids $p] < 0} {
3648 # make this line extend up to the child
3649 set le [drawlineseg $p $r $er 0]
3650 lappend lineends($le) $p
3651 set prevlines($p) 1
3652 }
3653 }
3654 } elseif {![info exists prevlines($lid)]} {
3655 set le [drawlineseg $lid $r $er 1]
3656 lappend lineends($le) $lid
3657 set prevlines($lid) 1
3658 }
3659 }
3660 }
3661}
3662
3663proc drawfrac {f0 f1} {
3664 global canv linespc
3665
3666 set ymax [lindex [$canv cget -scrollregion] 3]
3667 if {$ymax eq {} || $ymax == 0} return
3668 set y0 [expr {int($f0 * $ymax)}]
3669 set row [expr {int(($y0 - 3) / $linespc) - 1}]
3670 set y1 [expr {int($f1 * $ymax)}]
3671 set endrow [expr {int(($y1 - 3) / $linespc) + 1}]
3672 drawcommits $row $endrow
3673}
3674
3675proc drawvisible {} {
3676 global canv
3677 eval drawfrac [$canv yview]
3678}
3679
3680proc clear_display {} {
3681 global iddrawn linesegs
3682 global vhighlights fhighlights nhighlights rhighlights
3683
3684 allcanvs delete all
3685 catch {unset iddrawn}
3686 catch {unset linesegs}
3687 catch {unset vhighlights}
3688 catch {unset fhighlights}
3689 catch {unset nhighlights}
3690 catch {unset rhighlights}
3691}
3692
3693proc findcrossings {id} {
3694 global rowidlist parentlist numcommits rowoffsets displayorder
3695
3696 set cross {}
3697 set ccross {}
3698 foreach {s e} [rowranges $id] {
3699 if {$e >= $numcommits} {
3700 set e [expr {$numcommits - 1}]
3701 }
3702 if {$e <= $s} continue
3703 set x [lsearch -exact [lindex $rowidlist $e] $id]
3704 if {$x < 0} {
3705 puts "findcrossings: oops, no [shortids $id] in row $e"
3706 continue
3707 }
3708 for {set row $e} {[incr row -1] >= $s} {} {
3709 set olds [lindex $parentlist $row]
3710 set kid [lindex $displayorder $row]
3711 set kidx [lsearch -exact [lindex $rowidlist $row] $kid]
3712 if {$kidx < 0} continue
3713 set nextrow [lindex $rowidlist [expr {$row + 1}]]
3714 foreach p $olds {
3715 set px [lsearch -exact $nextrow $p]
3716 if {$px < 0} continue
3717 if {($kidx < $x && $x < $px) || ($px < $x && $x < $kidx)} {
3718 if {[lsearch -exact $ccross $p] >= 0} continue
3719 if {$x == $px + ($kidx < $px? -1: 1)} {
3720 lappend ccross $p
3721 } elseif {[lsearch -exact $cross $p] < 0} {
3722 lappend cross $p
3723 }
3724 }
3725 }
3726 set inc [lindex $rowoffsets $row $x]
3727 if {$inc eq {}} break
3728 incr x $inc
3729 }
3730 }
3731 return [concat $ccross {{}} $cross]
3732}
3733
3734proc assigncolor {id} {
3735 global colormap colors nextcolor
3736 global commitrow parentlist children children curview
3737
3738 if {[info exists colormap($id)]} return
3739 set ncolors [llength $colors]
3740 if {[info exists children($curview,$id)]} {
3741 set kids $children($curview,$id)
3742 } else {
3743 set kids {}
3744 }
3745 if {[llength $kids] == 1} {
3746 set child [lindex $kids 0]
3747 if {[info exists colormap($child)]
3748 && [llength [lindex $parentlist $commitrow($curview,$child)]] == 1} {
3749 set colormap($id) $colormap($child)
3750 return
3751 }
3752 }
3753 set badcolors {}
3754 set origbad {}
3755 foreach x [findcrossings $id] {
3756 if {$x eq {}} {
3757 # delimiter between corner crossings and other crossings
3758 if {[llength $badcolors] >= $ncolors - 1} break
3759 set origbad $badcolors
3760 }
3761 if {[info exists colormap($x)]
3762 && [lsearch -exact $badcolors $colormap($x)] < 0} {
3763 lappend badcolors $colormap($x)
3764 }
3765 }
3766 if {[llength $badcolors] >= $ncolors} {
3767 set badcolors $origbad
3768 }
3769 set origbad $badcolors
3770 if {[llength $badcolors] < $ncolors - 1} {
3771 foreach child $kids {
3772 if {[info exists colormap($child)]
3773 && [lsearch -exact $badcolors $colormap($child)] < 0} {
3774 lappend badcolors $colormap($child)
3775 }
3776 foreach p [lindex $parentlist $commitrow($curview,$child)] {
3777 if {[info exists colormap($p)]
3778 && [lsearch -exact $badcolors $colormap($p)] < 0} {
3779 lappend badcolors $colormap($p)
3780 }
3781 }
3782 }
3783 if {[llength $badcolors] >= $ncolors} {
3784 set badcolors $origbad
3785 }
3786 }
3787 for {set i 0} {$i <= $ncolors} {incr i} {
3788 set c [lindex $colors $nextcolor]
3789 if {[incr nextcolor] >= $ncolors} {
3790 set nextcolor 0
3791 }
3792 if {[lsearch -exact $badcolors $c]} break
3793 }
3794 set colormap($id) $c
3795}
3796
3797proc bindline {t id} {
3798 global canv
3799
3800 $canv bind $t <Enter> "lineenter %x %y $id"
3801 $canv bind $t <Motion> "linemotion %x %y $id"
3802 $canv bind $t <Leave> "lineleave $id"
3803 $canv bind $t <Button-1> "lineclick %x %y $id 1"
3804}
3805
3806proc drawtags {id x xt y1} {
3807 global idtags idheads idotherrefs mainhead
3808 global linespc lthickness
3809 global canv mainfont commitrow rowtextx curview fgcolor bgcolor
3810
3811 set marks {}
3812 set ntags 0
3813 set nheads 0
3814 if {[info exists idtags($id)]} {
3815 set marks $idtags($id)
3816 set ntags [llength $marks]
3817 }
3818 if {[info exists idheads($id)]} {
3819 set marks [concat $marks $idheads($id)]
3820 set nheads [llength $idheads($id)]
3821 }
3822 if {[info exists idotherrefs($id)]} {
3823 set marks [concat $marks $idotherrefs($id)]
3824 }
3825 if {$marks eq {}} {
3826 return $xt
3827 }
3828
3829 set delta [expr {int(0.5 * ($linespc - $lthickness))}]
3830 set yt [expr {$y1 - 0.5 * $linespc}]
3831 set yb [expr {$yt + $linespc - 1}]
3832 set xvals {}
3833 set wvals {}
3834 set i -1
3835 foreach tag $marks {
3836 incr i
3837 if {$i >= $ntags && $i < $ntags + $nheads && $tag eq $mainhead} {
3838 set wid [font measure [concat $mainfont bold] $tag]
3839 } else {
3840 set wid [font measure $mainfont $tag]
3841 }
3842 lappend xvals $xt
3843 lappend wvals $wid
3844 set xt [expr {$xt + $delta + $wid + $lthickness + $linespc}]
3845 }
3846 set t [$canv create line $x $y1 [lindex $xvals end] $y1 \
3847 -width $lthickness -fill black -tags tag.$id]
3848 $canv lower $t
3849 foreach tag $marks x $xvals wid $wvals {
3850 set xl [expr {$x + $delta}]
3851 set xr [expr {$x + $delta + $wid + $lthickness}]
3852 set font $mainfont
3853 if {[incr ntags -1] >= 0} {
3854 # draw a tag
3855 set t [$canv create polygon $x [expr {$yt + $delta}] $xl $yt \
3856 $xr $yt $xr $yb $xl $yb $x [expr {$yb - $delta}] \
3857 -width 1 -outline black -fill yellow -tags tag.$id]
3858 $canv bind $t <1> [list showtag $tag 1]
3859 set rowtextx($commitrow($curview,$id)) [expr {$xr + $linespc}]
3860 } else {
3861 # draw a head or other ref
3862 if {[incr nheads -1] >= 0} {
3863 set col green
3864 if {$tag eq $mainhead} {
3865 lappend font bold
3866 }
3867 } else {
3868 set col "#ddddff"
3869 }
3870 set xl [expr {$xl - $delta/2}]
3871 $canv create polygon $x $yt $xr $yt $xr $yb $x $yb \
3872 -width 1 -outline black -fill $col -tags tag.$id
3873 if {[regexp {^(remotes/.*/|remotes/)} $tag match remoteprefix]} {
3874 set rwid [font measure $mainfont $remoteprefix]
3875 set xi [expr {$x + 1}]
3876 set yti [expr {$yt + 1}]
3877 set xri [expr {$x + $rwid}]
3878 $canv create polygon $xi $yti $xri $yti $xri $yb $xi $yb \
3879 -width 0 -fill "#ffddaa" -tags tag.$id
3880 }
3881 }
3882 set t [$canv create text $xl $y1 -anchor w -text $tag -fill $fgcolor \
3883 -font $font -tags [list tag.$id text]]
3884 if {$ntags >= 0} {
3885 $canv bind $t <1> [list showtag $tag 1]
3886 } elseif {$nheads >= 0} {
3887 $canv bind $t <Button-3> [list headmenu %X %Y $id $tag]
3888 }
3889 }
3890 return $xt
3891}
3892
3893proc xcoord {i level ln} {
3894 global canvx0 xspc1 xspc2
3895
3896 set x [expr {$canvx0 + $i * $xspc1($ln)}]
3897 if {$i > 0 && $i == $level} {
3898 set x [expr {$x + 0.5 * ($xspc2 - $xspc1($ln))}]
3899 } elseif {$i > $level} {
3900 set x [expr {$x + $xspc2 - $xspc1($ln)}]
3901 }
3902 return $x
3903}
3904
3905proc show_status {msg} {
3906 global canv mainfont fgcolor
3907
3908 clear_display
3909 $canv create text 3 3 -anchor nw -text $msg -font $mainfont \
3910 -tags text -fill $fgcolor
3911}
3912
3913# Insert a new commit as the child of the commit on row $row.
3914# The new commit will be displayed on row $row and the commits
3915# on that row and below will move down one row.
3916proc insertrow {row newcmit} {
3917 global displayorder parentlist commitlisted children
3918 global commitrow curview rowidlist rowoffsets numcommits
3919 global rowrangelist rowlaidout rowoptim numcommits
3920 global selectedline rowchk commitidx
3921
3922 if {$row >= $numcommits} {
3923 puts "oops, inserting new row $row but only have $numcommits rows"
3924 return
3925 }
3926 set p [lindex $displayorder $row]
3927 set displayorder [linsert $displayorder $row $newcmit]
3928 set parentlist [linsert $parentlist $row $p]
3929 set kids $children($curview,$p)
3930 lappend kids $newcmit
3931 set children($curview,$p) $kids
3932 set children($curview,$newcmit) {}
3933 set commitlisted [linsert $commitlisted $row 1]
3934 set l [llength $displayorder]
3935 for {set r $row} {$r < $l} {incr r} {
3936 set id [lindex $displayorder $r]
3937 set commitrow($curview,$id) $r
3938 }
3939 incr commitidx($curview)
3940
3941 set idlist [lindex $rowidlist $row]
3942 set offs [lindex $rowoffsets $row]
3943 set newoffs {}
3944 foreach x $idlist {
3945 if {$x eq {} || ($x eq $p && [llength $kids] == 1)} {
3946 lappend newoffs {}
3947 } else {
3948 lappend newoffs 0
3949 }
3950 }
3951 if {[llength $kids] == 1} {
3952 set col [lsearch -exact $idlist $p]
3953 lset idlist $col $newcmit
3954 } else {
3955 set col [llength $idlist]
3956 lappend idlist $newcmit
3957 lappend offs {}
3958 lset rowoffsets $row $offs
3959 }
3960 set rowidlist [linsert $rowidlist $row $idlist]
3961 set rowoffsets [linsert $rowoffsets [expr {$row+1}] $newoffs]
3962
3963 set rowrangelist [linsert $rowrangelist $row {}]
3964 if {[llength $kids] > 1} {
3965 set rp1 [expr {$row + 1}]
3966 set ranges [lindex $rowrangelist $rp1]
3967 if {$ranges eq {}} {
3968 set ranges [list $newcmit $p]
3969 } elseif {[lindex $ranges end-1] eq $p} {
3970 lset ranges end-1 $newcmit
3971 }
3972 lset rowrangelist $rp1 $ranges
3973 }
3974
3975 catch {unset rowchk}
3976
3977 incr rowlaidout
3978 incr rowoptim
3979 incr numcommits
3980
3981 if {[info exists selectedline] && $selectedline >= $row} {
3982 incr selectedline
3983 }
3984 redisplay
3985}
3986
3987# Remove a commit that was inserted with insertrow on row $row.
3988proc removerow {row} {
3989 global displayorder parentlist commitlisted children
3990 global commitrow curview rowidlist rowoffsets numcommits
3991 global rowrangelist idrowranges rowlaidout rowoptim numcommits
3992 global linesegends selectedline rowchk commitidx
3993
3994 if {$row >= $numcommits} {
3995 puts "oops, removing row $row but only have $numcommits rows"
3996 return
3997 }
3998 set rp1 [expr {$row + 1}]
3999 set id [lindex $displayorder $row]
4000 set p [lindex $parentlist $row]
4001 set displayorder [lreplace $displayorder $row $row]
4002 set parentlist [lreplace $parentlist $row $row]
4003 set commitlisted [lreplace $commitlisted $row $row]
4004 set kids $children($curview,$p)
4005 set i [lsearch -exact $kids $id]
4006 if {$i >= 0} {
4007 set kids [lreplace $kids $i $i]
4008 set children($curview,$p) $kids
4009 }
4010 set l [llength $displayorder]
4011 for {set r $row} {$r < $l} {incr r} {
4012 set id [lindex $displayorder $r]
4013 set commitrow($curview,$id) $r
4014 }
4015 incr commitidx($curview) -1
4016
4017 set rowidlist [lreplace $rowidlist $row $row]
4018 set rowoffsets [lreplace $rowoffsets $rp1 $rp1]
4019 if {$kids ne {}} {
4020 set offs [lindex $rowoffsets $row]
4021 set offs [lreplace $offs end end]
4022 lset rowoffsets $row $offs
4023 }
4024
4025 set rowrangelist [lreplace $rowrangelist $row $row]
4026 if {[llength $kids] > 0} {
4027 set ranges [lindex $rowrangelist $row]
4028 if {[lindex $ranges end-1] eq $id} {
4029 set ranges [lreplace $ranges end-1 end]
4030 lset rowrangelist $row $ranges
4031 }
4032 }
4033
4034 catch {unset rowchk}
4035
4036 incr rowlaidout -1
4037 incr rowoptim -1
4038 incr numcommits -1
4039
4040 if {[info exists selectedline] && $selectedline > $row} {
4041 incr selectedline -1
4042 }
4043 redisplay
4044}
4045
4046# Don't change the text pane cursor if it is currently the hand cursor,
4047# showing that we are over a sha1 ID link.
4048proc settextcursor {c} {
4049 global ctext curtextcursor
4050
4051 if {[$ctext cget -cursor] == $curtextcursor} {
4052 $ctext config -cursor $c
4053 }
4054 set curtextcursor $c
4055}
4056
4057proc nowbusy {what} {
4058 global isbusy
4059
4060 if {[array names isbusy] eq {}} {
4061 . config -cursor watch
4062 settextcursor watch
4063 }
4064 set isbusy($what) 1
4065}
4066
4067proc notbusy {what} {
4068 global isbusy maincursor textcursor
4069
4070 catch {unset isbusy($what)}
4071 if {[array names isbusy] eq {}} {
4072 . config -cursor $maincursor
4073 settextcursor $textcursor
4074 }
4075}
4076
4077proc findmatches {f} {
4078 global findtype findstring
4079 if {$findtype == "Regexp"} {
4080 set matches [regexp -indices -all -inline $findstring $f]
4081 } else {
4082 set fs $findstring
4083 if {$findtype == "IgnCase"} {
4084 set f [string tolower $f]
4085 set fs [string tolower $fs]
4086 }
4087 set matches {}
4088 set i 0
4089 set l [string length $fs]
4090 while {[set j [string first $fs $f $i]] >= 0} {
4091 lappend matches [list $j [expr {$j+$l-1}]]
4092 set i [expr {$j + $l}]
4093 }
4094 }
4095 return $matches
4096}
4097
4098proc dofind {{rev 0}} {
4099 global findstring findstartline findcurline selectedline numcommits
4100
4101 unmarkmatches
4102 cancel_next_highlight
4103 focus .
4104 if {$findstring eq {} || $numcommits == 0} return
4105 if {![info exists selectedline]} {
4106 set findstartline [lindex [visiblerows] $rev]
4107 } else {
4108 set findstartline $selectedline
4109 }
4110 set findcurline $findstartline
4111 nowbusy finding
4112 if {!$rev} {
4113 run findmore
4114 } else {
4115 if {$findcurline == 0} {
4116 set findcurline $numcommits
4117 }
4118 incr findcurline -1
4119 run findmorerev
4120 }
4121}
4122
4123proc findnext {restart} {
4124 global findcurline
4125 if {![info exists findcurline]} {
4126 if {$restart} {
4127 dofind
4128 } else {
4129 bell
4130 }
4131 } else {
4132 run findmore
4133 nowbusy finding
4134 }
4135}
4136
4137proc findprev {} {
4138 global findcurline
4139 if {![info exists findcurline]} {
4140 dofind 1
4141 } else {
4142 run findmorerev
4143 nowbusy finding
4144 }
4145}
4146
4147proc findmore {} {
4148 global commitdata commitinfo numcommits findstring findpattern findloc
4149 global findstartline findcurline displayorder
4150
4151 set fldtypes {Headline Author Date Committer CDate Comments}
4152 set l [expr {$findcurline + 1}]
4153 if {$l >= $numcommits} {
4154 set l 0
4155 }
4156 if {$l <= $findstartline} {
4157 set lim [expr {$findstartline + 1}]
4158 } else {
4159 set lim $numcommits
4160 }
4161 if {$lim - $l > 500} {
4162 set lim [expr {$l + 500}]
4163 }
4164 set last 0
4165 for {} {$l < $lim} {incr l} {
4166 set id [lindex $displayorder $l]
4167 # shouldn't happen unless git log doesn't give all the commits...
4168 if {![info exists commitdata($id)]} continue
4169 if {![doesmatch $commitdata($id)]} continue
4170 if {![info exists commitinfo($id)]} {
4171 getcommit $id
4172 }
4173 set info $commitinfo($id)
4174 foreach f $info ty $fldtypes {
4175 if {($findloc eq "All fields" || $findloc eq $ty) &&
4176 [doesmatch $f]} {
4177 findselectline $l
4178 notbusy finding
4179 return 0
4180 }
4181 }
4182 }
4183 if {$l == $findstartline + 1} {
4184 bell
4185 unset findcurline
4186 notbusy finding
4187 return 0
4188 }
4189 set findcurline [expr {$l - 1}]
4190 return 1
4191}
4192
4193proc findmorerev {} {
4194 global commitdata commitinfo numcommits findstring findpattern findloc
4195 global findstartline findcurline displayorder
4196
4197 set fldtypes {Headline Author Date Committer CDate Comments}
4198 set l $findcurline
4199 if {$l == 0} {
4200 set l $numcommits
4201 }
4202 incr l -1
4203 if {$l >= $findstartline} {
4204 set lim [expr {$findstartline - 1}]
4205 } else {
4206 set lim -1
4207 }
4208 if {$l - $lim > 500} {
4209 set lim [expr {$l - 500}]
4210 }
4211 set last 0
4212 for {} {$l > $lim} {incr l -1} {
4213 set id [lindex $displayorder $l]
4214 if {![doesmatch $commitdata($id)]} continue
4215 if {![info exists commitinfo($id)]} {
4216 getcommit $id
4217 }
4218 set info $commitinfo($id)
4219 foreach f $info ty $fldtypes {
4220 if {($findloc eq "All fields" || $findloc eq $ty) &&
4221 [doesmatch $f]} {
4222 findselectline $l
4223 notbusy finding
4224 return 0
4225 }
4226 }
4227 }
4228 if {$l == -1} {
4229 bell
4230 unset findcurline
4231 notbusy finding
4232 return 0
4233 }
4234 set findcurline [expr {$l + 1}]
4235 return 1
4236}
4237
4238proc findselectline {l} {
4239 global findloc commentend ctext findcurline markingmatches
4240
4241 set markingmatches 1
4242 set findcurline $l
4243 selectline $l 1
4244 if {$findloc == "All fields" || $findloc == "Comments"} {
4245 # highlight the matches in the comments
4246 set f [$ctext get 1.0 $commentend]
4247 set matches [findmatches $f]
4248 foreach match $matches {
4249 set start [lindex $match 0]
4250 set end [expr {[lindex $match 1] + 1}]
4251 $ctext tag add found "1.0 + $start c" "1.0 + $end c"
4252 }
4253 }
4254 drawvisible
4255}
4256
4257# mark the bits of a headline or author that match a find string
4258proc markmatches {canv l str tag matches font row} {
4259 global selectedline
4260
4261 set bbox [$canv bbox $tag]
4262 set x0 [lindex $bbox 0]
4263 set y0 [lindex $bbox 1]
4264 set y1 [lindex $bbox 3]
4265 foreach match $matches {
4266 set start [lindex $match 0]
4267 set end [lindex $match 1]
4268 if {$start > $end} continue
4269 set xoff [font measure $font [string range $str 0 [expr {$start-1}]]]
4270 set xlen [font measure $font [string range $str 0 [expr {$end}]]]
4271 set t [$canv create rect [expr {$x0+$xoff}] $y0 \
4272 [expr {$x0+$xlen+2}] $y1 \
4273 -outline {} -tags [list match$l matches] -fill yellow]
4274 $canv lower $t
4275 if {[info exists selectedline] && $row == $selectedline} {
4276 $canv raise $t secsel
4277 }
4278 }
4279}
4280
4281proc unmarkmatches {} {
4282 global findids markingmatches findcurline
4283
4284 allcanvs delete matches
4285 catch {unset findids}
4286 set markingmatches 0
4287 catch {unset findcurline}
4288}
4289
4290proc selcanvline {w x y} {
4291 global canv canvy0 ctext linespc
4292 global rowtextx
4293 set ymax [lindex [$canv cget -scrollregion] 3]
4294 if {$ymax == {}} return
4295 set yfrac [lindex [$canv yview] 0]
4296 set y [expr {$y + $yfrac * $ymax}]
4297 set l [expr {int(($y - $canvy0) / $linespc + 0.5)}]
4298 if {$l < 0} {
4299 set l 0
4300 }
4301 if {$w eq $canv} {
4302 if {![info exists rowtextx($l)] || $x < $rowtextx($l)} return
4303 }
4304 unmarkmatches
4305 selectline $l 1
4306}
4307
4308proc commit_descriptor {p} {
4309 global commitinfo
4310 if {![info exists commitinfo($p)]} {
4311 getcommit $p
4312 }
4313 set l "..."
4314 if {[llength $commitinfo($p)] > 1} {
4315 set l [lindex $commitinfo($p) 0]
4316 }
4317 return "$p ($l)\n"
4318}
4319
4320# append some text to the ctext widget, and make any SHA1 ID
4321# that we know about be a clickable link.
4322proc appendwithlinks {text tags} {
4323 global ctext commitrow linknum curview
4324
4325 set start [$ctext index "end - 1c"]
4326 $ctext insert end $text $tags
4327 set links [regexp -indices -all -inline {[0-9a-f]{40}} $text]
4328 foreach l $links {
4329 set s [lindex $l 0]
4330 set e [lindex $l 1]
4331 set linkid [string range $text $s $e]
4332 if {![info exists commitrow($curview,$linkid)]} continue
4333 incr e
4334 $ctext tag add link "$start + $s c" "$start + $e c"
4335 $ctext tag add link$linknum "$start + $s c" "$start + $e c"
4336 $ctext tag bind link$linknum <1> \
4337 [list selectline $commitrow($curview,$linkid) 1]
4338 incr linknum
4339 }
4340 $ctext tag conf link -foreground blue -underline 1
4341 $ctext tag bind link <Enter> { %W configure -cursor hand2 }
4342 $ctext tag bind link <Leave> { %W configure -cursor $curtextcursor }
4343}
4344
4345proc viewnextline {dir} {
4346 global canv linespc
4347
4348 $canv delete hover
4349 set ymax [lindex [$canv cget -scrollregion] 3]
4350 set wnow [$canv yview]
4351 set wtop [expr {[lindex $wnow 0] * $ymax}]
4352 set newtop [expr {$wtop + $dir * $linespc}]
4353 if {$newtop < 0} {
4354 set newtop 0
4355 } elseif {$newtop > $ymax} {
4356 set newtop $ymax
4357 }
4358 allcanvs yview moveto [expr {$newtop * 1.0 / $ymax}]
4359}
4360
4361# add a list of tag or branch names at position pos
4362# returns the number of names inserted
4363proc appendrefs {pos ids var} {
4364 global ctext commitrow linknum curview $var maxrefs
4365
4366 if {[catch {$ctext index $pos}]} {
4367 return 0
4368 }
4369 $ctext conf -state normal
4370 $ctext delete $pos "$pos lineend"
4371 set tags {}
4372 foreach id $ids {
4373 foreach tag [set $var\($id\)] {
4374 lappend tags [list $tag $id]
4375 }
4376 }
4377 if {[llength $tags] > $maxrefs} {
4378 $ctext insert $pos "many ([llength $tags])"
4379 } else {
4380 set tags [lsort -index 0 -decreasing $tags]
4381 set sep {}
4382 foreach ti $tags {
4383 set id [lindex $ti 1]
4384 set lk link$linknum
4385 incr linknum
4386 $ctext tag delete $lk
4387 $ctext insert $pos $sep
4388 $ctext insert $pos [lindex $ti 0] $lk
4389 if {[info exists commitrow($curview,$id)]} {
4390 $ctext tag conf $lk -foreground blue
4391 $ctext tag bind $lk <1> \
4392 [list selectline $commitrow($curview,$id) 1]
4393 $ctext tag conf $lk -underline 1
4394 $ctext tag bind $lk <Enter> { %W configure -cursor hand2 }
4395 $ctext tag bind $lk <Leave> \
4396 { %W configure -cursor $curtextcursor }
4397 }
4398 set sep ", "
4399 }
4400 }
4401 $ctext conf -state disabled
4402 return [llength $tags]
4403}
4404
4405# called when we have finished computing the nearby tags
4406proc dispneartags {delay} {
4407 global selectedline currentid showneartags tagphase
4408
4409 if {![info exists selectedline] || !$showneartags} return
4410 after cancel dispnexttag
4411 if {$delay} {
4412 after 200 dispnexttag
4413 set tagphase -1
4414 } else {
4415 after idle dispnexttag
4416 set tagphase 0
4417 }
4418}
4419
4420proc dispnexttag {} {
4421 global selectedline currentid showneartags tagphase ctext
4422
4423 if {![info exists selectedline] || !$showneartags} return
4424 switch -- $tagphase {
4425 0 {
4426 set dtags [desctags $currentid]
4427 if {$dtags ne {}} {
4428 appendrefs precedes $dtags idtags
4429 }
4430 }
4431 1 {
4432 set atags [anctags $currentid]
4433 if {$atags ne {}} {
4434 appendrefs follows $atags idtags
4435 }
4436 }
4437 2 {
4438 set dheads [descheads $currentid]
4439 if {$dheads ne {}} {
4440 if {[appendrefs branch $dheads idheads] > 1
4441 && [$ctext get "branch -3c"] eq "h"} {
4442 # turn "Branch" into "Branches"
4443 $ctext conf -state normal
4444 $ctext insert "branch -2c" "es"
4445 $ctext conf -state disabled
4446 }
4447 }
4448 }
4449 }
4450 if {[incr tagphase] <= 2} {
4451 after idle dispnexttag
4452 }
4453}
4454
4455proc selectline {l isnew} {
4456 global canv canv2 canv3 ctext commitinfo selectedline
4457 global displayorder linehtag linentag linedtag
4458 global canvy0 linespc parentlist children curview
4459 global currentid sha1entry
4460 global commentend idtags linknum
4461 global mergemax numcommits pending_select
4462 global cmitmode showneartags allcommits
4463
4464 catch {unset pending_select}
4465 $canv delete hover
4466 normalline
4467 cancel_next_highlight
4468 if {$l < 0 || $l >= $numcommits} return
4469 set y [expr {$canvy0 + $l * $linespc}]
4470 set ymax [lindex [$canv cget -scrollregion] 3]
4471 set ytop [expr {$y - $linespc - 1}]
4472 set ybot [expr {$y + $linespc + 1}]
4473 set wnow [$canv yview]
4474 set wtop [expr {[lindex $wnow 0] * $ymax}]
4475 set wbot [expr {[lindex $wnow 1] * $ymax}]
4476 set wh [expr {$wbot - $wtop}]
4477 set newtop $wtop
4478 if {$ytop < $wtop} {
4479 if {$ybot < $wtop} {
4480 set newtop [expr {$y - $wh / 2.0}]
4481 } else {
4482 set newtop $ytop
4483 if {$newtop > $wtop - $linespc} {
4484 set newtop [expr {$wtop - $linespc}]
4485 }
4486 }
4487 } elseif {$ybot > $wbot} {
4488 if {$ytop > $wbot} {
4489 set newtop [expr {$y - $wh / 2.0}]
4490 } else {
4491 set newtop [expr {$ybot - $wh}]
4492 if {$newtop < $wtop + $linespc} {
4493 set newtop [expr {$wtop + $linespc}]
4494 }
4495 }
4496 }
4497 if {$newtop != $wtop} {
4498 if {$newtop < 0} {
4499 set newtop 0
4500 }
4501 allcanvs yview moveto [expr {$newtop * 1.0 / $ymax}]
4502 drawvisible
4503 }
4504
4505 if {![info exists linehtag($l)]} return
4506 $canv delete secsel
4507 set t [eval $canv create rect [$canv bbox $linehtag($l)] -outline {{}} \
4508 -tags secsel -fill [$canv cget -selectbackground]]
4509 $canv lower $t
4510 $canv2 delete secsel
4511 set t [eval $canv2 create rect [$canv2 bbox $linentag($l)] -outline {{}} \
4512 -tags secsel -fill [$canv2 cget -selectbackground]]
4513 $canv2 lower $t
4514 $canv3 delete secsel
4515 set t [eval $canv3 create rect [$canv3 bbox $linedtag($l)] -outline {{}} \
4516 -tags secsel -fill [$canv3 cget -selectbackground]]
4517 $canv3 lower $t
4518
4519 if {$isnew} {
4520 addtohistory [list selectline $l 0]
4521 }
4522
4523 set selectedline $l
4524
4525 set id [lindex $displayorder $l]
4526 set currentid $id
4527 $sha1entry delete 0 end
4528 $sha1entry insert 0 $id
4529 $sha1entry selection from 0
4530 $sha1entry selection to end
4531 rhighlight_sel $id
4532
4533 $ctext conf -state normal
4534 clear_ctext
4535 set linknum 0
4536 set info $commitinfo($id)
4537 set date [formatdate [lindex $info 2]]
4538 $ctext insert end "Author: [lindex $info 1] $date\n"
4539 set date [formatdate [lindex $info 4]]
4540 $ctext insert end "Committer: [lindex $info 3] $date\n"
4541 if {[info exists idtags($id)]} {
4542 $ctext insert end "Tags:"
4543 foreach tag $idtags($id) {
4544 $ctext insert end " $tag"
4545 }
4546 $ctext insert end "\n"
4547 }
4548
4549 set headers {}
4550 set olds [lindex $parentlist $l]
4551 if {[llength $olds] > 1} {
4552 set np 0
4553 foreach p $olds {
4554 if {$np >= $mergemax} {
4555 set tag mmax
4556 } else {
4557 set tag m$np
4558 }
4559 $ctext insert end "Parent: " $tag
4560 appendwithlinks [commit_descriptor $p] {}
4561 incr np
4562 }
4563 } else {
4564 foreach p $olds {
4565 append headers "Parent: [commit_descriptor $p]"
4566 }
4567 }
4568
4569 foreach c $children($curview,$id) {
4570 append headers "Child: [commit_descriptor $c]"
4571 }
4572
4573 # make anything that looks like a SHA1 ID be a clickable link
4574 appendwithlinks $headers {}
4575 if {$showneartags} {
4576 if {![info exists allcommits]} {
4577 getallcommits
4578 }
4579 $ctext insert end "Branch: "
4580 $ctext mark set branch "end -1c"
4581 $ctext mark gravity branch left
4582 $ctext insert end "\nFollows: "
4583 $ctext mark set follows "end -1c"
4584 $ctext mark gravity follows left
4585 $ctext insert end "\nPrecedes: "
4586 $ctext mark set precedes "end -1c"
4587 $ctext mark gravity precedes left
4588 $ctext insert end "\n"
4589 dispneartags 1
4590 }
4591 $ctext insert end "\n"
4592 set comment [lindex $info 5]
4593 if {[string first "\r" $comment] >= 0} {
4594 set comment [string map {"\r" "\n "} $comment]
4595 }
4596 appendwithlinks $comment {comment}
4597
4598 $ctext tag remove found 1.0 end
4599 $ctext conf -state disabled
4600 set commentend [$ctext index "end - 1c"]
4601
4602 init_flist "Comments"
4603 if {$cmitmode eq "tree"} {
4604 gettree $id
4605 } elseif {[llength $olds] <= 1} {
4606 startdiff $id
4607 } else {
4608 mergediff $id $l
4609 }
4610}
4611
4612proc selfirstline {} {
4613 unmarkmatches
4614 selectline 0 1
4615}
4616
4617proc sellastline {} {
4618 global numcommits
4619 unmarkmatches
4620 set l [expr {$numcommits - 1}]
4621 selectline $l 1
4622}
4623
4624proc selnextline {dir} {
4625 global selectedline
4626 focus .
4627 if {![info exists selectedline]} return
4628 set l [expr {$selectedline + $dir}]
4629 unmarkmatches
4630 selectline $l 1
4631}
4632
4633proc selnextpage {dir} {
4634 global canv linespc selectedline numcommits
4635
4636 set lpp [expr {([winfo height $canv] - 2) / $linespc}]
4637 if {$lpp < 1} {
4638 set lpp 1
4639 }
4640 allcanvs yview scroll [expr {$dir * $lpp}] units
4641 drawvisible
4642 if {![info exists selectedline]} return
4643 set l [expr {$selectedline + $dir * $lpp}]
4644 if {$l < 0} {
4645 set l 0
4646 } elseif {$l >= $numcommits} {
4647 set l [expr $numcommits - 1]
4648 }
4649 unmarkmatches
4650 selectline $l 1
4651}
4652
4653proc unselectline {} {
4654 global selectedline currentid
4655
4656 catch {unset selectedline}
4657 catch {unset currentid}
4658 allcanvs delete secsel
4659 rhighlight_none
4660 cancel_next_highlight
4661}
4662
4663proc reselectline {} {
4664 global selectedline
4665
4666 if {[info exists selectedline]} {
4667 selectline $selectedline 0
4668 }
4669}
4670
4671proc addtohistory {cmd} {
4672 global history historyindex curview
4673
4674 set elt [list $curview $cmd]
4675 if {$historyindex > 0
4676 && [lindex $history [expr {$historyindex - 1}]] == $elt} {
4677 return
4678 }
4679
4680 if {$historyindex < [llength $history]} {
4681 set history [lreplace $history $historyindex end $elt]
4682 } else {
4683 lappend history $elt
4684 }
4685 incr historyindex
4686 if {$historyindex > 1} {
4687 .tf.bar.leftbut conf -state normal
4688 } else {
4689 .tf.bar.leftbut conf -state disabled
4690 }
4691 .tf.bar.rightbut conf -state disabled
4692}
4693
4694proc godo {elt} {
4695 global curview
4696
4697 set view [lindex $elt 0]
4698 set cmd [lindex $elt 1]
4699 if {$curview != $view} {
4700 showview $view
4701 }
4702 eval $cmd
4703}
4704
4705proc goback {} {
4706 global history historyindex
4707 focus .
4708
4709 if {$historyindex > 1} {
4710 incr historyindex -1
4711 godo [lindex $history [expr {$historyindex - 1}]]
4712 .tf.bar.rightbut conf -state normal
4713 }
4714 if {$historyindex <= 1} {
4715 .tf.bar.leftbut conf -state disabled
4716 }
4717}
4718
4719proc goforw {} {
4720 global history historyindex
4721 focus .
4722
4723 if {$historyindex < [llength $history]} {
4724 set cmd [lindex $history $historyindex]
4725 incr historyindex
4726 godo $cmd
4727 .tf.bar.leftbut conf -state normal
4728 }
4729 if {$historyindex >= [llength $history]} {
4730 .tf.bar.rightbut conf -state disabled
4731 }
4732}
4733
4734proc gettree {id} {
4735 global treefilelist treeidlist diffids diffmergeid treepending
4736 global nullid nullid2
4737
4738 set diffids $id
4739 catch {unset diffmergeid}
4740 if {![info exists treefilelist($id)]} {
4741 if {![info exists treepending]} {
4742 if {$id eq $nullid} {
4743 set cmd [list | git ls-files]
4744 } elseif {$id eq $nullid2} {
4745 set cmd [list | git ls-files --stage -t]
4746 } else {
4747 set cmd [list | git ls-tree -r $id]
4748 }
4749 if {[catch {set gtf [open $cmd r]}]} {
4750 return
4751 }
4752 set treepending $id
4753 set treefilelist($id) {}
4754 set treeidlist($id) {}
4755 fconfigure $gtf -blocking 0
4756 filerun $gtf [list gettreeline $gtf $id]
4757 }
4758 } else {
4759 setfilelist $id
4760 }
4761}
4762
4763proc gettreeline {gtf id} {
4764 global treefilelist treeidlist treepending cmitmode diffids nullid nullid2
4765
4766 set nl 0
4767 while {[incr nl] <= 1000 && [gets $gtf line] >= 0} {
4768 if {$diffids eq $nullid} {
4769 set fname $line
4770 } else {
4771 if {$diffids ne $nullid2 && [lindex $line 1] ne "blob"} continue
4772 set i [string first "\t" $line]
4773 if {$i < 0} continue
4774 set sha1 [lindex $line 2]
4775 set fname [string range $line [expr {$i+1}] end]
4776 if {[string index $fname 0] eq "\""} {
4777 set fname [lindex $fname 0]
4778 }
4779 lappend treeidlist($id) $sha1
4780 }
4781 lappend treefilelist($id) $fname
4782 }
4783 if {![eof $gtf]} {
4784 return [expr {$nl >= 1000? 2: 1}]
4785 }
4786 close $gtf
4787 unset treepending
4788 if {$cmitmode ne "tree"} {
4789 if {![info exists diffmergeid]} {
4790 gettreediffs $diffids
4791 }
4792 } elseif {$id ne $diffids} {
4793 gettree $diffids
4794 } else {
4795 setfilelist $id
4796 }
4797 return 0
4798}
4799
4800proc showfile {f} {
4801 global treefilelist treeidlist diffids nullid nullid2
4802 global ctext commentend
4803
4804 set i [lsearch -exact $treefilelist($diffids) $f]
4805 if {$i < 0} {
4806 puts "oops, $f not in list for id $diffids"
4807 return
4808 }
4809 if {$diffids eq $nullid} {
4810 if {[catch {set bf [open $f r]} err]} {
4811 puts "oops, can't read $f: $err"
4812 return
4813 }
4814 } else {
4815 set blob [lindex $treeidlist($diffids) $i]
4816 if {[catch {set bf [open [concat | git cat-file blob $blob] r]} err]} {
4817 puts "oops, error reading blob $blob: $err"
4818 return
4819 }
4820 }
4821 fconfigure $bf -blocking 0
4822 filerun $bf [list getblobline $bf $diffids]
4823 $ctext config -state normal
4824 clear_ctext $commentend
4825 $ctext insert end "\n"
4826 $ctext insert end "$f\n" filesep
4827 $ctext config -state disabled
4828 $ctext yview $commentend
4829}
4830
4831proc getblobline {bf id} {
4832 global diffids cmitmode ctext
4833
4834 if {$id ne $diffids || $cmitmode ne "tree"} {
4835 catch {close $bf}
4836 return 0
4837 }
4838 $ctext config -state normal
4839 set nl 0
4840 while {[incr nl] <= 1000 && [gets $bf line] >= 0} {
4841 $ctext insert end "$line\n"
4842 }
4843 if {[eof $bf]} {
4844 # delete last newline
4845 $ctext delete "end - 2c" "end - 1c"
4846 close $bf
4847 return 0
4848 }
4849 $ctext config -state disabled
4850 return [expr {$nl >= 1000? 2: 1}]
4851}
4852
4853proc mergediff {id l} {
4854 global diffmergeid diffopts mdifffd
4855 global diffids
4856 global parentlist
4857
4858 set diffmergeid $id
4859 set diffids $id
4860 # this doesn't seem to actually affect anything...
4861 set env(GIT_DIFF_OPTS) $diffopts
4862 set cmd [concat | git diff-tree --no-commit-id --cc $id]
4863 if {[catch {set mdf [open $cmd r]} err]} {
4864 error_popup "Error getting merge diffs: $err"
4865 return
4866 }
4867 fconfigure $mdf -blocking 0
4868 set mdifffd($id) $mdf
4869 set np [llength [lindex $parentlist $l]]
4870 filerun $mdf [list getmergediffline $mdf $id $np]
4871}
4872
4873proc getmergediffline {mdf id np} {
4874 global diffmergeid ctext cflist mergemax
4875 global difffilestart mdifffd
4876
4877 $ctext conf -state normal
4878 set nr 0
4879 while {[incr nr] <= 1000 && [gets $mdf line] >= 0} {
4880 if {![info exists diffmergeid] || $id != $diffmergeid
4881 || $mdf != $mdifffd($id)} {
4882 close $mdf
4883 return 0
4884 }
4885 if {[regexp {^diff --cc (.*)} $line match fname]} {
4886 # start of a new file
4887 $ctext insert end "\n"
4888 set here [$ctext index "end - 1c"]
4889 lappend difffilestart $here
4890 add_flist [list $fname]
4891 set l [expr {(78 - [string length $fname]) / 2}]
4892 set pad [string range "----------------------------------------" 1 $l]
4893 $ctext insert end "$pad $fname $pad\n" filesep
4894 } elseif {[regexp {^@@} $line]} {
4895 $ctext insert end "$line\n" hunksep
4896 } elseif {[regexp {^[0-9a-f]{40}$} $line] || [regexp {^index} $line]} {
4897 # do nothing
4898 } else {
4899 # parse the prefix - one ' ', '-' or '+' for each parent
4900 set spaces {}
4901 set minuses {}
4902 set pluses {}
4903 set isbad 0
4904 for {set j 0} {$j < $np} {incr j} {
4905 set c [string range $line $j $j]
4906 if {$c == " "} {
4907 lappend spaces $j
4908 } elseif {$c == "-"} {
4909 lappend minuses $j
4910 } elseif {$c == "+"} {
4911 lappend pluses $j
4912 } else {
4913 set isbad 1
4914 break
4915 }
4916 }
4917 set tags {}
4918 set num {}
4919 if {!$isbad && $minuses ne {} && $pluses eq {}} {
4920 # line doesn't appear in result, parents in $minuses have the line
4921 set num [lindex $minuses 0]
4922 } elseif {!$isbad && $pluses ne {} && $minuses eq {}} {
4923 # line appears in result, parents in $pluses don't have the line
4924 lappend tags mresult
4925 set num [lindex $spaces 0]
4926 }
4927 if {$num ne {}} {
4928 if {$num >= $mergemax} {
4929 set num "max"
4930 }
4931 lappend tags m$num
4932 }
4933 $ctext insert end "$line\n" $tags
4934 }
4935 }
4936 $ctext conf -state disabled
4937 if {[eof $mdf]} {
4938 close $mdf
4939 return 0
4940 }
4941 return [expr {$nr >= 1000? 2: 1}]
4942}
4943
4944proc startdiff {ids} {
4945 global treediffs diffids treepending diffmergeid nullid nullid2
4946
4947 set diffids $ids
4948 catch {unset diffmergeid}
4949 if {![info exists treediffs($ids)] ||
4950 [lsearch -exact $ids $nullid] >= 0 ||
4951 [lsearch -exact $ids $nullid2] >= 0} {
4952 if {![info exists treepending]} {
4953 gettreediffs $ids
4954 }
4955 } else {
4956 addtocflist $ids
4957 }
4958}
4959
4960proc addtocflist {ids} {
4961 global treediffs cflist
4962 add_flist $treediffs($ids)
4963 getblobdiffs $ids
4964}
4965
4966proc diffcmd {ids flags} {
4967 global nullid nullid2
4968
4969 set i [lsearch -exact $ids $nullid]
4970 set j [lsearch -exact $ids $nullid2]
4971 if {$i >= 0} {
4972 if {[llength $ids] > 1 && $j < 0} {
4973 # comparing working directory with some specific revision
4974 set cmd [concat | git diff-index $flags]
4975 if {$i == 0} {
4976 lappend cmd -R [lindex $ids 1]
4977 } else {
4978 lappend cmd [lindex $ids 0]
4979 }
4980 } else {
4981 # comparing working directory with index
4982 set cmd [concat | git diff-files $flags]
4983 if {$j == 1} {
4984 lappend cmd -R
4985 }
4986 }
4987 } elseif {$j >= 0} {
4988 set cmd [concat | git diff-index --cached $flags]
4989 if {[llength $ids] > 1} {
4990 # comparing index with specific revision
4991 if {$i == 0} {
4992 lappend cmd -R [lindex $ids 1]
4993 } else {
4994 lappend cmd [lindex $ids 0]
4995 }
4996 } else {
4997 # comparing index with HEAD
4998 lappend cmd HEAD
4999 }
5000 } else {
5001 set cmd [concat | git diff-tree -r $flags $ids]
5002 }
5003 return $cmd
5004}
5005
5006proc gettreediffs {ids} {
5007 global treediff treepending
5008
5009 set treepending $ids
5010 set treediff {}
5011 if {[catch {set gdtf [open [diffcmd $ids {--no-commit-id}] r]}]} return
5012 fconfigure $gdtf -blocking 0
5013 filerun $gdtf [list gettreediffline $gdtf $ids]
5014}
5015
5016proc gettreediffline {gdtf ids} {
5017 global treediff treediffs treepending diffids diffmergeid
5018 global cmitmode
5019
5020 set nr 0
5021 while {[incr nr] <= 1000 && [gets $gdtf line] >= 0} {
5022 set i [string first "\t" $line]
5023 if {$i >= 0} {
5024 set file [string range $line [expr {$i+1}] end]
5025 if {[string index $file 0] eq "\""} {
5026 set file [lindex $file 0]
5027 }
5028 lappend treediff $file
5029 }
5030 }
5031 if {![eof $gdtf]} {
5032 return [expr {$nr >= 1000? 2: 1}]
5033 }
5034 close $gdtf
5035 set treediffs($ids) $treediff
5036 unset treepending
5037 if {$cmitmode eq "tree"} {
5038 gettree $diffids
5039 } elseif {$ids != $diffids} {
5040 if {![info exists diffmergeid]} {
5041 gettreediffs $diffids
5042 }
5043 } else {
5044 addtocflist $ids
5045 }
5046 return 0
5047}
5048
5049proc getblobdiffs {ids} {
5050 global diffopts blobdifffd diffids env
5051 global diffinhdr treediffs
5052
5053 set env(GIT_DIFF_OPTS) $diffopts
5054 if {[catch {set bdf [open [diffcmd $ids {-p -C --no-commit-id}] r]} err]} {
5055 puts "error getting diffs: $err"
5056 return
5057 }
5058 set diffinhdr 0
5059 fconfigure $bdf -blocking 0
5060 set blobdifffd($ids) $bdf
5061 filerun $bdf [list getblobdiffline $bdf $diffids]
5062}
5063
5064proc setinlist {var i val} {
5065 global $var
5066
5067 while {[llength [set $var]] < $i} {
5068 lappend $var {}
5069 }
5070 if {[llength [set $var]] == $i} {
5071 lappend $var $val
5072 } else {
5073 lset $var $i $val
5074 }
5075}
5076
5077proc makediffhdr {fname ids} {
5078 global ctext curdiffstart treediffs
5079
5080 set i [lsearch -exact $treediffs($ids) $fname]
5081 if {$i >= 0} {
5082 setinlist difffilestart $i $curdiffstart
5083 }
5084 set l [expr {(78 - [string length $fname]) / 2}]
5085 set pad [string range "----------------------------------------" 1 $l]
5086 $ctext insert $curdiffstart "$pad $fname $pad" filesep
5087}
5088
5089proc getblobdiffline {bdf ids} {
5090 global diffids blobdifffd ctext curdiffstart
5091 global diffnexthead diffnextnote difffilestart
5092 global diffinhdr treediffs
5093
5094 set nr 0
5095 $ctext conf -state normal
5096 while {[incr nr] <= 1000 && [gets $bdf line] >= 0} {
5097 if {$ids != $diffids || $bdf != $blobdifffd($ids)} {
5098 close $bdf
5099 return 0
5100 }
5101 if {![string compare -length 11 "diff --git " $line]} {
5102 # trim off "diff --git "
5103 set line [string range $line 11 end]
5104 set diffinhdr 1
5105 # start of a new file
5106 $ctext insert end "\n"
5107 set curdiffstart [$ctext index "end - 1c"]
5108 $ctext insert end "\n" filesep
5109 # If the name hasn't changed the length will be odd,
5110 # the middle char will be a space, and the two bits either
5111 # side will be a/name and b/name, or "a/name" and "b/name".
5112 # If the name has changed we'll get "rename from" and
5113 # "rename to" lines following this, and we'll use them
5114 # to get the filenames.
5115 # This complexity is necessary because spaces in the filename(s)
5116 # don't get escaped.
5117 set l [string length $line]
5118 set i [expr {$l / 2}]
5119 if {!(($l & 1) && [string index $line $i] eq " " &&
5120 [string range $line 2 [expr {$i - 1}]] eq \
5121 [string range $line [expr {$i + 3}] end])} {
5122 continue
5123 }
5124 # unescape if quoted and chop off the a/ from the front
5125 if {[string index $line 0] eq "\""} {
5126 set fname [string range [lindex $line 0] 2 end]
5127 } else {
5128 set fname [string range $line 2 [expr {$i - 1}]]
5129 }
5130 makediffhdr $fname $ids
5131
5132 } elseif {[regexp {^@@ -([0-9]+)(,[0-9]+)? \+([0-9]+)(,[0-9]+)? @@(.*)} \
5133 $line match f1l f1c f2l f2c rest]} {
5134 $ctext insert end "$line\n" hunksep
5135 set diffinhdr 0
5136
5137 } elseif {$diffinhdr} {
5138 if {![string compare -length 12 "rename from " $line]} {
5139 set fname [string range $line 12 end]
5140 if {[string index $fname 0] eq "\""} {
5141 set fname [lindex $fname 0]
5142 }
5143 set i [lsearch -exact $treediffs($ids) $fname]
5144 if {$i >= 0} {
5145 setinlist difffilestart $i $curdiffstart
5146 }
5147 } elseif {![string compare -length 10 $line "rename to "]} {
5148 set fname [string range $line 10 end]
5149 if {[string index $fname 0] eq "\""} {
5150 set fname [lindex $fname 0]
5151 }
5152 makediffhdr $fname $ids
5153 } elseif {[string compare -length 3 $line "---"] == 0} {
5154 # do nothing
5155 continue
5156 } elseif {[string compare -length 3 $line "+++"] == 0} {
5157 set diffinhdr 0
5158 continue
5159 }
5160 $ctext insert end "$line\n" filesep
5161
5162 } else {
5163 set x [string range $line 0 0]
5164 if {$x == "-" || $x == "+"} {
5165 set tag [expr {$x == "+"}]
5166 $ctext insert end "$line\n" d$tag
5167 } elseif {$x == " "} {
5168 $ctext insert end "$line\n"
5169 } else {
5170 # "\ No newline at end of file",
5171 # or something else we don't recognize
5172 $ctext insert end "$line\n" hunksep
5173 }
5174 }
5175 }
5176 $ctext conf -state disabled
5177 if {[eof $bdf]} {
5178 close $bdf
5179 return 0
5180 }
5181 return [expr {$nr >= 1000? 2: 1}]
5182}
5183
5184proc changediffdisp {} {
5185 global ctext diffelide
5186
5187 $ctext tag conf d0 -elide [lindex $diffelide 0]
5188 $ctext tag conf d1 -elide [lindex $diffelide 1]
5189}
5190
5191proc prevfile {} {
5192 global difffilestart ctext
5193 set prev [lindex $difffilestart 0]
5194 set here [$ctext index @0,0]
5195 foreach loc $difffilestart {
5196 if {[$ctext compare $loc >= $here]} {
5197 $ctext yview $prev
5198 return
5199 }
5200 set prev $loc
5201 }
5202 $ctext yview $prev
5203}
5204
5205proc nextfile {} {
5206 global difffilestart ctext
5207 set here [$ctext index @0,0]
5208 foreach loc $difffilestart {
5209 if {[$ctext compare $loc > $here]} {
5210 $ctext yview $loc
5211 return
5212 }
5213 }
5214}
5215
5216proc clear_ctext {{first 1.0}} {
5217 global ctext smarktop smarkbot
5218
5219 set l [lindex [split $first .] 0]
5220 if {![info exists smarktop] || [$ctext compare $first < $smarktop.0]} {
5221 set smarktop $l
5222 }
5223 if {![info exists smarkbot] || [$ctext compare $first < $smarkbot.0]} {
5224 set smarkbot $l
5225 }
5226 $ctext delete $first end
5227}
5228
5229proc incrsearch {name ix op} {
5230 global ctext searchstring searchdirn
5231
5232 $ctext tag remove found 1.0 end
5233 if {[catch {$ctext index anchor}]} {
5234 # no anchor set, use start of selection, or of visible area
5235 set sel [$ctext tag ranges sel]
5236 if {$sel ne {}} {
5237 $ctext mark set anchor [lindex $sel 0]
5238 } elseif {$searchdirn eq "-forwards"} {
5239 $ctext mark set anchor @0,0
5240 } else {
5241 $ctext mark set anchor @0,[winfo height $ctext]
5242 }
5243 }
5244 if {$searchstring ne {}} {
5245 set here [$ctext search $searchdirn -- $searchstring anchor]
5246 if {$here ne {}} {
5247 $ctext see $here
5248 }
5249 searchmarkvisible 1
5250 }
5251}
5252
5253proc dosearch {} {
5254 global sstring ctext searchstring searchdirn
5255
5256 focus $sstring
5257 $sstring icursor end
5258 set searchdirn -forwards
5259 if {$searchstring ne {}} {
5260 set sel [$ctext tag ranges sel]
5261 if {$sel ne {}} {
5262 set start "[lindex $sel 0] + 1c"
5263 } elseif {[catch {set start [$ctext index anchor]}]} {
5264 set start "@0,0"
5265 }
5266 set match [$ctext search -count mlen -- $searchstring $start]
5267 $ctext tag remove sel 1.0 end
5268 if {$match eq {}} {
5269 bell
5270 return
5271 }
5272 $ctext see $match
5273 set mend "$match + $mlen c"
5274 $ctext tag add sel $match $mend
5275 $ctext mark unset anchor
5276 }
5277}
5278
5279proc dosearchback {} {
5280 global sstring ctext searchstring searchdirn
5281
5282 focus $sstring
5283 $sstring icursor end
5284 set searchdirn -backwards
5285 if {$searchstring ne {}} {
5286 set sel [$ctext tag ranges sel]
5287 if {$sel ne {}} {
5288 set start [lindex $sel 0]
5289 } elseif {[catch {set start [$ctext index anchor]}]} {
5290 set start @0,[winfo height $ctext]
5291 }
5292 set match [$ctext search -backwards -count ml -- $searchstring $start]
5293 $ctext tag remove sel 1.0 end
5294 if {$match eq {}} {
5295 bell
5296 return
5297 }
5298 $ctext see $match
5299 set mend "$match + $ml c"
5300 $ctext tag add sel $match $mend
5301 $ctext mark unset anchor
5302 }
5303}
5304
5305proc searchmark {first last} {
5306 global ctext searchstring
5307
5308 set mend $first.0
5309 while {1} {
5310 set match [$ctext search -count mlen -- $searchstring $mend $last.end]
5311 if {$match eq {}} break
5312 set mend "$match + $mlen c"
5313 $ctext tag add found $match $mend
5314 }
5315}
5316
5317proc searchmarkvisible {doall} {
5318 global ctext smarktop smarkbot
5319
5320 set topline [lindex [split [$ctext index @0,0] .] 0]
5321 set botline [lindex [split [$ctext index @0,[winfo height $ctext]] .] 0]
5322 if {$doall || $botline < $smarktop || $topline > $smarkbot} {
5323 # no overlap with previous
5324 searchmark $topline $botline
5325 set smarktop $topline
5326 set smarkbot $botline
5327 } else {
5328 if {$topline < $smarktop} {
5329 searchmark $topline [expr {$smarktop-1}]
5330 set smarktop $topline
5331 }
5332 if {$botline > $smarkbot} {
5333 searchmark [expr {$smarkbot+1}] $botline
5334 set smarkbot $botline
5335 }
5336 }
5337}
5338
5339proc scrolltext {f0 f1} {
5340 global searchstring
5341
5342 .bleft.sb set $f0 $f1
5343 if {$searchstring ne {}} {
5344 searchmarkvisible 0
5345 }
5346}
5347
5348proc setcoords {} {
5349 global linespc charspc canvx0 canvy0 mainfont
5350 global xspc1 xspc2 lthickness
5351
5352 set linespc [font metrics $mainfont -linespace]
5353 set charspc [font measure $mainfont "m"]
5354 set canvy0 [expr {int(3 + 0.5 * $linespc)}]
5355 set canvx0 [expr {int(3 + 0.5 * $linespc)}]
5356 set lthickness [expr {int($linespc / 9) + 1}]
5357 set xspc1(0) $linespc
5358 set xspc2 $linespc
5359}
5360
5361proc redisplay {} {
5362 global canv
5363 global selectedline
5364
5365 set ymax [lindex [$canv cget -scrollregion] 3]
5366 if {$ymax eq {} || $ymax == 0} return
5367 set span [$canv yview]
5368 clear_display
5369 setcanvscroll
5370 allcanvs yview moveto [lindex $span 0]
5371 drawvisible
5372 if {[info exists selectedline]} {
5373 selectline $selectedline 0
5374 allcanvs yview moveto [lindex $span 0]
5375 }
5376}
5377
5378proc incrfont {inc} {
5379 global mainfont textfont ctext canv phase cflist
5380 global charspc tabstop
5381 global stopped entries
5382 unmarkmatches
5383 set mainfont [lreplace $mainfont 1 1 [expr {[lindex $mainfont 1] + $inc}]]
5384 set textfont [lreplace $textfont 1 1 [expr {[lindex $textfont 1] + $inc}]]
5385 setcoords
5386 $ctext conf -font $textfont -tabs "[expr {$tabstop * $charspc}]"
5387 $cflist conf -font $textfont
5388 $ctext tag conf filesep -font [concat $textfont bold]
5389 foreach e $entries {
5390 $e conf -font $mainfont
5391 }
5392 if {$phase eq "getcommits"} {
5393 $canv itemconf textitems -font $mainfont
5394 }
5395 redisplay
5396}
5397
5398proc clearsha1 {} {
5399 global sha1entry sha1string
5400 if {[string length $sha1string] == 40} {
5401 $sha1entry delete 0 end
5402 }
5403}
5404
5405proc sha1change {n1 n2 op} {
5406 global sha1string currentid sha1but
5407 if {$sha1string == {}
5408 || ([info exists currentid] && $sha1string == $currentid)} {
5409 set state disabled
5410 } else {
5411 set state normal
5412 }
5413 if {[$sha1but cget -state] == $state} return
5414 if {$state == "normal"} {
5415 $sha1but conf -state normal -relief raised -text "Goto: "
5416 } else {
5417 $sha1but conf -state disabled -relief flat -text "SHA1 ID: "
5418 }
5419}
5420
5421proc gotocommit {} {
5422 global sha1string currentid commitrow tagids headids
5423 global displayorder numcommits curview
5424
5425 if {$sha1string == {}
5426 || ([info exists currentid] && $sha1string == $currentid)} return
5427 if {[info exists tagids($sha1string)]} {
5428 set id $tagids($sha1string)
5429 } elseif {[info exists headids($sha1string)]} {
5430 set id $headids($sha1string)
5431 } else {
5432 set id [string tolower $sha1string]
5433 if {[regexp {^[0-9a-f]{4,39}$} $id]} {
5434 set matches {}
5435 foreach i $displayorder {
5436 if {[string match $id* $i]} {
5437 lappend matches $i
5438 }
5439 }
5440 if {$matches ne {}} {
5441 if {[llength $matches] > 1} {
5442 error_popup "Short SHA1 id $id is ambiguous"
5443 return
5444 }
5445 set id [lindex $matches 0]
5446 }
5447 }
5448 }
5449 if {[info exists commitrow($curview,$id)]} {
5450 selectline $commitrow($curview,$id) 1
5451 return
5452 }
5453 if {[regexp {^[0-9a-fA-F]{4,}$} $sha1string]} {
5454 set type "SHA1 id"
5455 } else {
5456 set type "Tag/Head"
5457 }
5458 error_popup "$type $sha1string is not known"
5459}
5460
5461proc lineenter {x y id} {
5462 global hoverx hovery hoverid hovertimer
5463 global commitinfo canv
5464
5465 if {![info exists commitinfo($id)] && ![getcommit $id]} return
5466 set hoverx $x
5467 set hovery $y
5468 set hoverid $id
5469 if {[info exists hovertimer]} {
5470 after cancel $hovertimer
5471 }
5472 set hovertimer [after 500 linehover]
5473 $canv delete hover
5474}
5475
5476proc linemotion {x y id} {
5477 global hoverx hovery hoverid hovertimer
5478
5479 if {[info exists hoverid] && $id == $hoverid} {
5480 set hoverx $x
5481 set hovery $y
5482 if {[info exists hovertimer]} {
5483 after cancel $hovertimer
5484 }
5485 set hovertimer [after 500 linehover]
5486 }
5487}
5488
5489proc lineleave {id} {
5490 global hoverid hovertimer canv
5491
5492 if {[info exists hoverid] && $id == $hoverid} {
5493 $canv delete hover
5494 if {[info exists hovertimer]} {
5495 after cancel $hovertimer
5496 unset hovertimer
5497 }
5498 unset hoverid
5499 }
5500}
5501
5502proc linehover {} {
5503 global hoverx hovery hoverid hovertimer
5504 global canv linespc lthickness
5505 global commitinfo mainfont
5506
5507 set text [lindex $commitinfo($hoverid) 0]
5508 set ymax [lindex [$canv cget -scrollregion] 3]
5509 if {$ymax == {}} return
5510 set yfrac [lindex [$canv yview] 0]
5511 set x [expr {$hoverx + 2 * $linespc}]
5512 set y [expr {$hovery + $yfrac * $ymax - $linespc / 2}]
5513 set x0 [expr {$x - 2 * $lthickness}]
5514 set y0 [expr {$y - 2 * $lthickness}]
5515 set x1 [expr {$x + [font measure $mainfont $text] + 2 * $lthickness}]
5516 set y1 [expr {$y + $linespc + 2 * $lthickness}]
5517 set t [$canv create rectangle $x0 $y0 $x1 $y1 \
5518 -fill \#ffff80 -outline black -width 1 -tags hover]
5519 $canv raise $t
5520 set t [$canv create text $x $y -anchor nw -text $text -tags hover \
5521 -font $mainfont]
5522 $canv raise $t
5523}
5524
5525proc clickisonarrow {id y} {
5526 global lthickness
5527
5528 set ranges [rowranges $id]
5529 set thresh [expr {2 * $lthickness + 6}]
5530 set n [expr {[llength $ranges] - 1}]
5531 for {set i 1} {$i < $n} {incr i} {
5532 set row [lindex $ranges $i]
5533 if {abs([yc $row] - $y) < $thresh} {
5534 return $i
5535 }
5536 }
5537 return {}
5538}
5539
5540proc arrowjump {id n y} {
5541 global canv
5542
5543 # 1 <-> 2, 3 <-> 4, etc...
5544 set n [expr {(($n - 1) ^ 1) + 1}]
5545 set row [lindex [rowranges $id] $n]
5546 set yt [yc $row]
5547 set ymax [lindex [$canv cget -scrollregion] 3]
5548 if {$ymax eq {} || $ymax <= 0} return
5549 set view [$canv yview]
5550 set yspan [expr {[lindex $view 1] - [lindex $view 0]}]
5551 set yfrac [expr {$yt / $ymax - $yspan / 2}]
5552 if {$yfrac < 0} {
5553 set yfrac 0
5554 }
5555 allcanvs yview moveto $yfrac
5556}
5557
5558proc lineclick {x y id isnew} {
5559 global ctext commitinfo children canv thickerline curview
5560
5561 if {![info exists commitinfo($id)] && ![getcommit $id]} return
5562 unmarkmatches
5563 unselectline
5564 normalline
5565 $canv delete hover
5566 # draw this line thicker than normal
5567 set thickerline $id
5568 drawlines $id
5569 if {$isnew} {
5570 set ymax [lindex [$canv cget -scrollregion] 3]
5571 if {$ymax eq {}} return
5572 set yfrac [lindex [$canv yview] 0]
5573 set y [expr {$y + $yfrac * $ymax}]
5574 }
5575 set dirn [clickisonarrow $id $y]
5576 if {$dirn ne {}} {
5577 arrowjump $id $dirn $y
5578 return
5579 }
5580
5581 if {$isnew} {
5582 addtohistory [list lineclick $x $y $id 0]
5583 }
5584 # fill the details pane with info about this line
5585 $ctext conf -state normal
5586 clear_ctext
5587 $ctext tag conf link -foreground blue -underline 1
5588 $ctext tag bind link <Enter> { %W configure -cursor hand2 }
5589 $ctext tag bind link <Leave> { %W configure -cursor $curtextcursor }
5590 $ctext insert end "Parent:\t"
5591 $ctext insert end $id [list link link0]
5592 $ctext tag bind link0 <1> [list selbyid $id]
5593 set info $commitinfo($id)
5594 $ctext insert end "\n\t[lindex $info 0]\n"
5595 $ctext insert end "\tAuthor:\t[lindex $info 1]\n"
5596 set date [formatdate [lindex $info 2]]
5597 $ctext insert end "\tDate:\t$date\n"
5598 set kids $children($curview,$id)
5599 if {$kids ne {}} {
5600 $ctext insert end "\nChildren:"
5601 set i 0
5602 foreach child $kids {
5603 incr i
5604 if {![info exists commitinfo($child)] && ![getcommit $child]} continue
5605 set info $commitinfo($child)
5606 $ctext insert end "\n\t"
5607 $ctext insert end $child [list link link$i]
5608 $ctext tag bind link$i <1> [list selbyid $child]
5609 $ctext insert end "\n\t[lindex $info 0]"
5610 $ctext insert end "\n\tAuthor:\t[lindex $info 1]"
5611 set date [formatdate [lindex $info 2]]
5612 $ctext insert end "\n\tDate:\t$date\n"
5613 }
5614 }
5615 $ctext conf -state disabled
5616 init_flist {}
5617}
5618
5619proc normalline {} {
5620 global thickerline
5621 if {[info exists thickerline]} {
5622 set id $thickerline
5623 unset thickerline
5624 drawlines $id
5625 }
5626}
5627
5628proc selbyid {id} {
5629 global commitrow curview
5630 if {[info exists commitrow($curview,$id)]} {
5631 selectline $commitrow($curview,$id) 1
5632 }
5633}
5634
5635proc mstime {} {
5636 global startmstime
5637 if {![info exists startmstime]} {
5638 set startmstime [clock clicks -milliseconds]
5639 }
5640 return [format "%.3f" [expr {([clock click -milliseconds] - $startmstime) / 1000.0}]]
5641}
5642
5643proc rowmenu {x y id} {
5644 global rowctxmenu commitrow selectedline rowmenuid curview
5645 global nullid nullid2 fakerowmenu mainhead
5646
5647 set rowmenuid $id
5648 if {![info exists selectedline]
5649 || $commitrow($curview,$id) eq $selectedline} {
5650 set state disabled
5651 } else {
5652 set state normal
5653 }
5654 if {$id ne $nullid && $id ne $nullid2} {
5655 set menu $rowctxmenu
5656 $menu entryconfigure 7 -label "Reset $mainhead branch to here"
5657 } else {
5658 set menu $fakerowmenu
5659 }
5660 $menu entryconfigure "Diff this*" -state $state
5661 $menu entryconfigure "Diff selected*" -state $state
5662 $menu entryconfigure "Make patch" -state $state
5663 tk_popup $menu $x $y
5664}
5665
5666proc diffvssel {dirn} {
5667 global rowmenuid selectedline displayorder
5668
5669 if {![info exists selectedline]} return
5670 if {$dirn} {
5671 set oldid [lindex $displayorder $selectedline]
5672 set newid $rowmenuid
5673 } else {
5674 set oldid $rowmenuid
5675 set newid [lindex $displayorder $selectedline]
5676 }
5677 addtohistory [list doseldiff $oldid $newid]
5678 doseldiff $oldid $newid
5679}
5680
5681proc doseldiff {oldid newid} {
5682 global ctext
5683 global commitinfo
5684
5685 $ctext conf -state normal
5686 clear_ctext
5687 init_flist "Top"
5688 $ctext insert end "From "
5689 $ctext tag conf link -foreground blue -underline 1
5690 $ctext tag bind link <Enter> { %W configure -cursor hand2 }
5691 $ctext tag bind link <Leave> { %W configure -cursor $curtextcursor }
5692 $ctext tag bind link0 <1> [list selbyid $oldid]
5693 $ctext insert end $oldid [list link link0]
5694 $ctext insert end "\n "
5695 $ctext insert end [lindex $commitinfo($oldid) 0]
5696 $ctext insert end "\n\nTo "
5697 $ctext tag bind link1 <1> [list selbyid $newid]
5698 $ctext insert end $newid [list link link1]
5699 $ctext insert end "\n "
5700 $ctext insert end [lindex $commitinfo($newid) 0]
5701 $ctext insert end "\n"
5702 $ctext conf -state disabled
5703 $ctext tag remove found 1.0 end
5704 startdiff [list $oldid $newid]
5705}
5706
5707proc mkpatch {} {
5708 global rowmenuid currentid commitinfo patchtop patchnum
5709
5710 if {![info exists currentid]} return
5711 set oldid $currentid
5712 set oldhead [lindex $commitinfo($oldid) 0]
5713 set newid $rowmenuid
5714 set newhead [lindex $commitinfo($newid) 0]
5715 set top .patch
5716 set patchtop $top
5717 catch {destroy $top}
5718 toplevel $top
5719 label $top.title -text "Generate patch"
5720 grid $top.title - -pady 10
5721 label $top.from -text "From:"
5722 entry $top.fromsha1 -width 40 -relief flat
5723 $top.fromsha1 insert 0 $oldid
5724 $top.fromsha1 conf -state readonly
5725 grid $top.from $top.fromsha1 -sticky w
5726 entry $top.fromhead -width 60 -relief flat
5727 $top.fromhead insert 0 $oldhead
5728 $top.fromhead conf -state readonly
5729 grid x $top.fromhead -sticky w
5730 label $top.to -text "To:"
5731 entry $top.tosha1 -width 40 -relief flat
5732 $top.tosha1 insert 0 $newid
5733 $top.tosha1 conf -state readonly
5734 grid $top.to $top.tosha1 -sticky w
5735 entry $top.tohead -width 60 -relief flat
5736 $top.tohead insert 0 $newhead
5737 $top.tohead conf -state readonly
5738 grid x $top.tohead -sticky w
5739 button $top.rev -text "Reverse" -command mkpatchrev -padx 5
5740 grid $top.rev x -pady 10
5741 label $top.flab -text "Output file:"
5742 entry $top.fname -width 60
5743 $top.fname insert 0 [file normalize "patch$patchnum.patch"]
5744 incr patchnum
5745 grid $top.flab $top.fname -sticky w
5746 frame $top.buts
5747 button $top.buts.gen -text "Generate" -command mkpatchgo
5748 button $top.buts.can -text "Cancel" -command mkpatchcan
5749 grid $top.buts.gen $top.buts.can
5750 grid columnconfigure $top.buts 0 -weight 1 -uniform a
5751 grid columnconfigure $top.buts 1 -weight 1 -uniform a
5752 grid $top.buts - -pady 10 -sticky ew
5753 focus $top.fname
5754}
5755
5756proc mkpatchrev {} {
5757 global patchtop
5758
5759 set oldid [$patchtop.fromsha1 get]
5760 set oldhead [$patchtop.fromhead get]
5761 set newid [$patchtop.tosha1 get]
5762 set newhead [$patchtop.tohead get]
5763 foreach e [list fromsha1 fromhead tosha1 tohead] \
5764 v [list $newid $newhead $oldid $oldhead] {
5765 $patchtop.$e conf -state normal
5766 $patchtop.$e delete 0 end
5767 $patchtop.$e insert 0 $v
5768 $patchtop.$e conf -state readonly
5769 }
5770}
5771
5772proc mkpatchgo {} {
5773 global patchtop nullid nullid2
5774
5775 set oldid [$patchtop.fromsha1 get]
5776 set newid [$patchtop.tosha1 get]
5777 set fname [$patchtop.fname get]
5778 set cmd [diffcmd [list $oldid $newid] -p]
5779 lappend cmd >$fname &
5780 if {[catch {eval exec $cmd} err]} {
5781 error_popup "Error creating patch: $err"
5782 }
5783 catch {destroy $patchtop}
5784 unset patchtop
5785}
5786
5787proc mkpatchcan {} {
5788 global patchtop
5789
5790 catch {destroy $patchtop}
5791 unset patchtop
5792}
5793
5794proc mktag {} {
5795 global rowmenuid mktagtop commitinfo
5796
5797 set top .maketag
5798 set mktagtop $top
5799 catch {destroy $top}
5800 toplevel $top
5801 label $top.title -text "Create tag"
5802 grid $top.title - -pady 10
5803 label $top.id -text "ID:"
5804 entry $top.sha1 -width 40 -relief flat
5805 $top.sha1 insert 0 $rowmenuid
5806 $top.sha1 conf -state readonly
5807 grid $top.id $top.sha1 -sticky w
5808 entry $top.head -width 60 -relief flat
5809 $top.head insert 0 [lindex $commitinfo($rowmenuid) 0]
5810 $top.head conf -state readonly
5811 grid x $top.head -sticky w
5812 label $top.tlab -text "Tag name:"
5813 entry $top.tag -width 60
5814 grid $top.tlab $top.tag -sticky w
5815 frame $top.buts
5816 button $top.buts.gen -text "Create" -command mktaggo
5817 button $top.buts.can -text "Cancel" -command mktagcan
5818 grid $top.buts.gen $top.buts.can
5819 grid columnconfigure $top.buts 0 -weight 1 -uniform a
5820 grid columnconfigure $top.buts 1 -weight 1 -uniform a
5821 grid $top.buts - -pady 10 -sticky ew
5822 focus $top.tag
5823}
5824
5825proc domktag {} {
5826 global mktagtop env tagids idtags
5827
5828 set id [$mktagtop.sha1 get]
5829 set tag [$mktagtop.tag get]
5830 if {$tag == {}} {
5831 error_popup "No tag name specified"
5832 return
5833 }
5834 if {[info exists tagids($tag)]} {
5835 error_popup "Tag \"$tag\" already exists"
5836 return
5837 }
5838 if {[catch {
5839 set dir [gitdir]
5840 set fname [file join $dir "refs/tags" $tag]
5841 set f [open $fname w]
5842 puts $f $id
5843 close $f
5844 } err]} {
5845 error_popup "Error creating tag: $err"
5846 return
5847 }
5848
5849 set tagids($tag) $id
5850 lappend idtags($id) $tag
5851 redrawtags $id
5852 addedtag $id
5853}
5854
5855proc redrawtags {id} {
5856 global canv linehtag commitrow idpos selectedline curview
5857 global mainfont canvxmax iddrawn
5858
5859 if {![info exists commitrow($curview,$id)]} return
5860 if {![info exists iddrawn($id)]} return
5861 drawcommits $commitrow($curview,$id)
5862 $canv delete tag.$id
5863 set xt [eval drawtags $id $idpos($id)]
5864 $canv coords $linehtag($commitrow($curview,$id)) $xt [lindex $idpos($id) 2]
5865 set text [$canv itemcget $linehtag($commitrow($curview,$id)) -text]
5866 set xr [expr {$xt + [font measure $mainfont $text]}]
5867 if {$xr > $canvxmax} {
5868 set canvxmax $xr
5869 setcanvscroll
5870 }
5871 if {[info exists selectedline]
5872 && $selectedline == $commitrow($curview,$id)} {
5873 selectline $selectedline 0
5874 }
5875}
5876
5877proc mktagcan {} {
5878 global mktagtop
5879
5880 catch {destroy $mktagtop}
5881 unset mktagtop
5882}
5883
5884proc mktaggo {} {
5885 domktag
5886 mktagcan
5887}
5888
5889proc writecommit {} {
5890 global rowmenuid wrcomtop commitinfo wrcomcmd
5891
5892 set top .writecommit
5893 set wrcomtop $top
5894 catch {destroy $top}
5895 toplevel $top
5896 label $top.title -text "Write commit to file"
5897 grid $top.title - -pady 10
5898 label $top.id -text "ID:"
5899 entry $top.sha1 -width 40 -relief flat
5900 $top.sha1 insert 0 $rowmenuid
5901 $top.sha1 conf -state readonly
5902 grid $top.id $top.sha1 -sticky w
5903 entry $top.head -width 60 -relief flat
5904 $top.head insert 0 [lindex $commitinfo($rowmenuid) 0]
5905 $top.head conf -state readonly
5906 grid x $top.head -sticky w
5907 label $top.clab -text "Command:"
5908 entry $top.cmd -width 60 -textvariable wrcomcmd
5909 grid $top.clab $top.cmd -sticky w -pady 10
5910 label $top.flab -text "Output file:"
5911 entry $top.fname -width 60
5912 $top.fname insert 0 [file normalize "commit-[string range $rowmenuid 0 6]"]
5913 grid $top.flab $top.fname -sticky w
5914 frame $top.buts
5915 button $top.buts.gen -text "Write" -command wrcomgo
5916 button $top.buts.can -text "Cancel" -command wrcomcan
5917 grid $top.buts.gen $top.buts.can
5918 grid columnconfigure $top.buts 0 -weight 1 -uniform a
5919 grid columnconfigure $top.buts 1 -weight 1 -uniform a
5920 grid $top.buts - -pady 10 -sticky ew
5921 focus $top.fname
5922}
5923
5924proc wrcomgo {} {
5925 global wrcomtop
5926
5927 set id [$wrcomtop.sha1 get]
5928 set cmd "echo $id | [$wrcomtop.cmd get]"
5929 set fname [$wrcomtop.fname get]
5930 if {[catch {exec sh -c $cmd >$fname &} err]} {
5931 error_popup "Error writing commit: $err"
5932 }
5933 catch {destroy $wrcomtop}
5934 unset wrcomtop
5935}
5936
5937proc wrcomcan {} {
5938 global wrcomtop
5939
5940 catch {destroy $wrcomtop}
5941 unset wrcomtop
5942}
5943
5944proc mkbranch {} {
5945 global rowmenuid mkbrtop
5946
5947 set top .makebranch
5948 catch {destroy $top}
5949 toplevel $top
5950 label $top.title -text "Create new branch"
5951 grid $top.title - -pady 10
5952 label $top.id -text "ID:"
5953 entry $top.sha1 -width 40 -relief flat
5954 $top.sha1 insert 0 $rowmenuid
5955 $top.sha1 conf -state readonly
5956 grid $top.id $top.sha1 -sticky w
5957 label $top.nlab -text "Name:"
5958 entry $top.name -width 40
5959 grid $top.nlab $top.name -sticky w
5960 frame $top.buts
5961 button $top.buts.go -text "Create" -command [list mkbrgo $top]
5962 button $top.buts.can -text "Cancel" -command "catch {destroy $top}"
5963 grid $top.buts.go $top.buts.can
5964 grid columnconfigure $top.buts 0 -weight 1 -uniform a
5965 grid columnconfigure $top.buts 1 -weight 1 -uniform a
5966 grid $top.buts - -pady 10 -sticky ew
5967 focus $top.name
5968}
5969
5970proc mkbrgo {top} {
5971 global headids idheads
5972
5973 set name [$top.name get]
5974 set id [$top.sha1 get]
5975 if {$name eq {}} {
5976 error_popup "Please specify a name for the new branch"
5977 return
5978 }
5979 catch {destroy $top}
5980 nowbusy newbranch
5981 update
5982 if {[catch {
5983 exec git branch $name $id
5984 } err]} {
5985 notbusy newbranch
5986 error_popup $err
5987 } else {
5988 set headids($name) $id
5989 lappend idheads($id) $name
5990 addedhead $id $name
5991 notbusy newbranch
5992 redrawtags $id
5993 dispneartags 0
5994 }
5995}
5996
5997proc cherrypick {} {
5998 global rowmenuid curview commitrow
5999 global mainhead
6000
6001 set oldhead [exec git rev-parse HEAD]
6002 set dheads [descheads $rowmenuid]
6003 if {$dheads ne {} && [lsearch -exact $dheads $oldhead] >= 0} {
6004 set ok [confirm_popup "Commit [string range $rowmenuid 0 7] is already\
6005 included in branch $mainhead -- really re-apply it?"]
6006 if {!$ok} return
6007 }
6008 nowbusy cherrypick
6009 update
6010 # Unfortunately git-cherry-pick writes stuff to stderr even when
6011 # no error occurs, and exec takes that as an indication of error...
6012 if {[catch {exec sh -c "git cherry-pick -r $rowmenuid 2>&1"} err]} {
6013 notbusy cherrypick
6014 error_popup $err
6015 return
6016 }
6017 set newhead [exec git rev-parse HEAD]
6018 if {$newhead eq $oldhead} {
6019 notbusy cherrypick
6020 error_popup "No changes committed"
6021 return
6022 }
6023 addnewchild $newhead $oldhead
6024 if {[info exists commitrow($curview,$oldhead)]} {
6025 insertrow $commitrow($curview,$oldhead) $newhead
6026 if {$mainhead ne {}} {
6027 movehead $newhead $mainhead
6028 movedhead $newhead $mainhead
6029 }
6030 redrawtags $oldhead
6031 redrawtags $newhead
6032 }
6033 notbusy cherrypick
6034}
6035
6036proc resethead {} {
6037 global mainheadid mainhead rowmenuid confirm_ok resettype
6038 global showlocalchanges
6039
6040 set confirm_ok 0
6041 set w ".confirmreset"
6042 toplevel $w
6043 wm transient $w .
6044 wm title $w "Confirm reset"
6045 message $w.m -text \
6046 "Reset branch $mainhead to [string range $rowmenuid 0 7]?" \
6047 -justify center -aspect 1000
6048 pack $w.m -side top -fill x -padx 20 -pady 20
6049 frame $w.f -relief sunken -border 2
6050 message $w.f.rt -text "Reset type:" -aspect 1000
6051 grid $w.f.rt -sticky w
6052 set resettype mixed
6053 radiobutton $w.f.soft -value soft -variable resettype -justify left \
6054 -text "Soft: Leave working tree and index untouched"
6055 grid $w.f.soft -sticky w
6056 radiobutton $w.f.mixed -value mixed -variable resettype -justify left \
6057 -text "Mixed: Leave working tree untouched, reset index"
6058 grid $w.f.mixed -sticky w
6059 radiobutton $w.f.hard -value hard -variable resettype -justify left \
6060 -text "Hard: Reset working tree and index\n(discard ALL local changes)"
6061 grid $w.f.hard -sticky w
6062 pack $w.f -side top -fill x
6063 button $w.ok -text OK -command "set confirm_ok 1; destroy $w"
6064 pack $w.ok -side left -fill x -padx 20 -pady 20
6065 button $w.cancel -text Cancel -command "destroy $w"
6066 pack $w.cancel -side right -fill x -padx 20 -pady 20
6067 bind $w <Visibility> "grab $w; focus $w"
6068 tkwait window $w
6069 if {!$confirm_ok} return
6070 if {[catch {set fd [open \
6071 [list | sh -c "git reset --$resettype $rowmenuid 2>&1"] r]} err]} {
6072 error_popup $err
6073 } else {
6074 dohidelocalchanges
6075 set w ".resetprogress"
6076 filerun $fd [list readresetstat $fd $w]
6077 toplevel $w
6078 wm transient $w
6079 wm title $w "Reset progress"
6080 message $w.m -text "Reset in progress, please wait..." \
6081 -justify center -aspect 1000
6082 pack $w.m -side top -fill x -padx 20 -pady 5
6083 canvas $w.c -width 150 -height 20 -bg white
6084 $w.c create rect 0 0 0 20 -fill green -tags rect
6085 pack $w.c -side top -fill x -padx 20 -pady 5 -expand 1
6086 nowbusy reset
6087 }
6088}
6089
6090proc readresetstat {fd w} {
6091 global mainhead mainheadid showlocalchanges
6092
6093 if {[gets $fd line] >= 0} {
6094 if {[regexp {([0-9]+)% \(([0-9]+)/([0-9]+)\)} $line match p m n]} {
6095 set x [expr {($m * 150) / $n}]
6096 $w.c coords rect 0 0 $x 20
6097 }
6098 return 1
6099 }
6100 destroy $w
6101 notbusy reset
6102 if {[catch {close $fd} err]} {
6103 error_popup $err
6104 }
6105 set oldhead $mainheadid
6106 set newhead [exec git rev-parse HEAD]
6107 if {$newhead ne $oldhead} {
6108 movehead $newhead $mainhead
6109 movedhead $newhead $mainhead
6110 set mainheadid $newhead
6111 redrawtags $oldhead
6112 redrawtags $newhead
6113 }
6114 if {$showlocalchanges} {
6115 doshowlocalchanges
6116 }
6117 return 0
6118}
6119
6120# context menu for a head
6121proc headmenu {x y id head} {
6122 global headmenuid headmenuhead headctxmenu mainhead
6123
6124 set headmenuid $id
6125 set headmenuhead $head
6126 set state normal
6127 if {$head eq $mainhead} {
6128 set state disabled
6129 }
6130 $headctxmenu entryconfigure 0 -state $state
6131 $headctxmenu entryconfigure 1 -state $state
6132 tk_popup $headctxmenu $x $y
6133}
6134
6135proc cobranch {} {
6136 global headmenuid headmenuhead mainhead headids
6137 global showlocalchanges mainheadid
6138
6139 # check the tree is clean first??
6140 set oldmainhead $mainhead
6141 nowbusy checkout
6142 update
6143 dohidelocalchanges
6144 if {[catch {
6145 exec git checkout -q $headmenuhead
6146 } err]} {
6147 notbusy checkout
6148 error_popup $err
6149 } else {
6150 notbusy checkout
6151 set mainhead $headmenuhead
6152 set mainheadid $headmenuid
6153 if {[info exists headids($oldmainhead)]} {
6154 redrawtags $headids($oldmainhead)
6155 }
6156 redrawtags $headmenuid
6157 }
6158 if {$showlocalchanges} {
6159 dodiffindex
6160 }
6161}
6162
6163proc rmbranch {} {
6164 global headmenuid headmenuhead mainhead
6165 global headids idheads
6166
6167 set head $headmenuhead
6168 set id $headmenuid
6169 # this check shouldn't be needed any more...
6170 if {$head eq $mainhead} {
6171 error_popup "Cannot delete the currently checked-out branch"
6172 return
6173 }
6174 set dheads [descheads $id]
6175 if {$dheads eq $headids($head)} {
6176 # the stuff on this branch isn't on any other branch
6177 if {![confirm_popup "The commits on branch $head aren't on any other\
6178 branch.\nReally delete branch $head?"]} return
6179 }
6180 nowbusy rmbranch
6181 update
6182 if {[catch {exec git branch -D $head} err]} {
6183 notbusy rmbranch
6184 error_popup $err
6185 return
6186 }
6187 removehead $id $head
6188 removedhead $id $head
6189 redrawtags $id
6190 notbusy rmbranch
6191 dispneartags 0
6192}
6193
6194# Stuff for finding nearby tags
6195proc getallcommits {} {
6196 global allcommits allids nbmp nextarc seeds
6197
6198 set allids {}
6199 set nbmp 0
6200 set nextarc 0
6201 set allcommits 0
6202 set seeds {}
6203 regetallcommits
6204}
6205
6206# Called when the graph might have changed
6207proc regetallcommits {} {
6208 global allcommits seeds
6209
6210 set cmd [concat | git rev-list --all --parents]
6211 foreach id $seeds {
6212 lappend cmd "^$id"
6213 }
6214 set fd [open $cmd r]
6215 fconfigure $fd -blocking 0
6216 incr allcommits
6217 nowbusy allcommits
6218 filerun $fd [list getallclines $fd]
6219}
6220
6221# Since most commits have 1 parent and 1 child, we group strings of
6222# such commits into "arcs" joining branch/merge points (BMPs), which
6223# are commits that either don't have 1 parent or don't have 1 child.
6224#
6225# arcnos(id) - incoming arcs for BMP, arc we're on for other nodes
6226# arcout(id) - outgoing arcs for BMP
6227# arcids(a) - list of IDs on arc including end but not start
6228# arcstart(a) - BMP ID at start of arc
6229# arcend(a) - BMP ID at end of arc
6230# growing(a) - arc a is still growing
6231# arctags(a) - IDs out of arcids (excluding end) that have tags
6232# archeads(a) - IDs out of arcids (excluding end) that have heads
6233# The start of an arc is at the descendent end, so "incoming" means
6234# coming from descendents, and "outgoing" means going towards ancestors.
6235
6236proc getallclines {fd} {
6237 global allids allparents allchildren idtags idheads nextarc nbmp
6238 global arcnos arcids arctags arcout arcend arcstart archeads growing
6239 global seeds allcommits
6240
6241 set nid 0
6242 while {[incr nid] <= 1000 && [gets $fd line] >= 0} {
6243 set id [lindex $line 0]
6244 if {[info exists allparents($id)]} {
6245 # seen it already
6246 continue
6247 }
6248 lappend allids $id
6249 set olds [lrange $line 1 end]
6250 set allparents($id) $olds
6251 if {![info exists allchildren($id)]} {
6252 set allchildren($id) {}
6253 set arcnos($id) {}
6254 lappend seeds $id
6255 } else {
6256 set a $arcnos($id)
6257 if {[llength $olds] == 1 && [llength $a] == 1} {
6258 lappend arcids($a) $id
6259 if {[info exists idtags($id)]} {
6260 lappend arctags($a) $id
6261 }
6262 if {[info exists idheads($id)]} {
6263 lappend archeads($a) $id
6264 }
6265 if {[info exists allparents($olds)]} {
6266 # seen parent already
6267 if {![info exists arcout($olds)]} {
6268 splitarc $olds
6269 }
6270 lappend arcids($a) $olds
6271 set arcend($a) $olds
6272 unset growing($a)
6273 }
6274 lappend allchildren($olds) $id
6275 lappend arcnos($olds) $a
6276 continue
6277 }
6278 }
6279 incr nbmp
6280 foreach a $arcnos($id) {
6281 lappend arcids($a) $id
6282 set arcend($a) $id
6283 unset growing($a)
6284 }
6285
6286 set ao {}
6287 foreach p $olds {
6288 lappend allchildren($p) $id
6289 set a [incr nextarc]
6290 set arcstart($a) $id
6291 set archeads($a) {}
6292 set arctags($a) {}
6293 set archeads($a) {}
6294 set arcids($a) {}
6295 lappend ao $a
6296 set growing($a) 1
6297 if {[info exists allparents($p)]} {
6298 # seen it already, may need to make a new branch
6299 if {![info exists arcout($p)]} {
6300 splitarc $p
6301 }
6302 lappend arcids($a) $p
6303 set arcend($a) $p
6304 unset growing($a)
6305 }
6306 lappend arcnos($p) $a
6307 }
6308 set arcout($id) $ao
6309 }
6310 if {$nid > 0} {
6311 global cached_dheads cached_dtags cached_atags
6312 catch {unset cached_dheads}
6313 catch {unset cached_dtags}
6314 catch {unset cached_atags}
6315 }
6316 if {![eof $fd]} {
6317 return [expr {$nid >= 1000? 2: 1}]
6318 }
6319 close $fd
6320 if {[incr allcommits -1] == 0} {
6321 notbusy allcommits
6322 }
6323 dispneartags 0
6324 return 0
6325}
6326
6327proc recalcarc {a} {
6328 global arctags archeads arcids idtags idheads
6329
6330 set at {}
6331 set ah {}
6332 foreach id [lrange $arcids($a) 0 end-1] {
6333 if {[info exists idtags($id)]} {
6334 lappend at $id
6335 }
6336 if {[info exists idheads($id)]} {
6337 lappend ah $id
6338 }
6339 }
6340 set arctags($a) $at
6341 set archeads($a) $ah
6342}
6343
6344proc splitarc {p} {
6345 global arcnos arcids nextarc nbmp arctags archeads idtags idheads
6346 global arcstart arcend arcout allparents growing
6347
6348 set a $arcnos($p)
6349 if {[llength $a] != 1} {
6350 puts "oops splitarc called but [llength $a] arcs already"
6351 return
6352 }
6353 set a [lindex $a 0]
6354 set i [lsearch -exact $arcids($a) $p]
6355 if {$i < 0} {
6356 puts "oops splitarc $p not in arc $a"
6357 return
6358 }
6359 set na [incr nextarc]
6360 if {[info exists arcend($a)]} {
6361 set arcend($na) $arcend($a)
6362 } else {
6363 set l [lindex $allparents([lindex $arcids($a) end]) 0]
6364 set j [lsearch -exact $arcnos($l) $a]
6365 set arcnos($l) [lreplace $arcnos($l) $j $j $na]
6366 }
6367 set tail [lrange $arcids($a) [expr {$i+1}] end]
6368 set arcids($a) [lrange $arcids($a) 0 $i]
6369 set arcend($a) $p
6370 set arcstart($na) $p
6371 set arcout($p) $na
6372 set arcids($na) $tail
6373 if {[info exists growing($a)]} {
6374 set growing($na) 1
6375 unset growing($a)
6376 }
6377 incr nbmp
6378
6379 foreach id $tail {
6380 if {[llength $arcnos($id)] == 1} {
6381 set arcnos($id) $na
6382 } else {
6383 set j [lsearch -exact $arcnos($id) $a]
6384 set arcnos($id) [lreplace $arcnos($id) $j $j $na]
6385 }
6386 }
6387
6388 # reconstruct tags and heads lists
6389 if {$arctags($a) ne {} || $archeads($a) ne {}} {
6390 recalcarc $a
6391 recalcarc $na
6392 } else {
6393 set arctags($na) {}
6394 set archeads($na) {}
6395 }
6396}
6397
6398# Update things for a new commit added that is a child of one
6399# existing commit. Used when cherry-picking.
6400proc addnewchild {id p} {
6401 global allids allparents allchildren idtags nextarc nbmp
6402 global arcnos arcids arctags arcout arcend arcstart archeads growing
6403 global seeds
6404
6405 lappend allids $id
6406 set allparents($id) [list $p]
6407 set allchildren($id) {}
6408 set arcnos($id) {}
6409 lappend seeds $id
6410 incr nbmp
6411 lappend allchildren($p) $id
6412 set a [incr nextarc]
6413 set arcstart($a) $id
6414 set archeads($a) {}
6415 set arctags($a) {}
6416 set arcids($a) [list $p]
6417 set arcend($a) $p
6418 if {![info exists arcout($p)]} {
6419 splitarc $p
6420 }
6421 lappend arcnos($p) $a
6422 set arcout($id) [list $a]
6423}
6424
6425# Returns 1 if a is an ancestor of b, -1 if b is an ancestor of a,
6426# or 0 if neither is true.
6427proc anc_or_desc {a b} {
6428 global arcout arcstart arcend arcnos cached_isanc
6429
6430 if {$arcnos($a) eq $arcnos($b)} {
6431 # Both are on the same arc(s); either both are the same BMP,
6432 # or if one is not a BMP, the other is also not a BMP or is
6433 # the BMP at end of the arc (and it only has 1 incoming arc).
6434 # Or both can be BMPs with no incoming arcs.
6435 if {$a eq $b || $arcnos($a) eq {}} {
6436 return 0
6437 }
6438 # assert {[llength $arcnos($a)] == 1}
6439 set arc [lindex $arcnos($a) 0]
6440 set i [lsearch -exact $arcids($arc) $a]
6441 set j [lsearch -exact $arcids($arc) $b]
6442 if {$i < 0 || $i > $j} {
6443 return 1
6444 } else {
6445 return -1
6446 }
6447 }
6448
6449 if {![info exists arcout($a)]} {
6450 set arc [lindex $arcnos($a) 0]
6451 if {[info exists arcend($arc)]} {
6452 set aend $arcend($arc)
6453 } else {
6454 set aend {}
6455 }
6456 set a $arcstart($arc)
6457 } else {
6458 set aend $a
6459 }
6460 if {![info exists arcout($b)]} {
6461 set arc [lindex $arcnos($b) 0]
6462 if {[info exists arcend($arc)]} {
6463 set bend $arcend($arc)
6464 } else {
6465 set bend {}
6466 }
6467 set b $arcstart($arc)
6468 } else {
6469 set bend $b
6470 }
6471 if {$a eq $bend} {
6472 return 1
6473 }
6474 if {$b eq $aend} {
6475 return -1
6476 }
6477 if {[info exists cached_isanc($a,$bend)]} {
6478 if {$cached_isanc($a,$bend)} {
6479 return 1
6480 }
6481 }
6482 if {[info exists cached_isanc($b,$aend)]} {
6483 if {$cached_isanc($b,$aend)} {
6484 return -1
6485 }
6486 if {[info exists cached_isanc($a,$bend)]} {
6487 return 0
6488 }
6489 }
6490
6491 set todo [list $a $b]
6492 set anc($a) a
6493 set anc($b) b
6494 for {set i 0} {$i < [llength $todo]} {incr i} {
6495 set x [lindex $todo $i]
6496 if {$anc($x) eq {}} {
6497 continue
6498 }
6499 foreach arc $arcnos($x) {
6500 set xd $arcstart($arc)
6501 if {$xd eq $bend} {
6502 set cached_isanc($a,$bend) 1
6503 set cached_isanc($b,$aend) 0
6504 return 1
6505 } elseif {$xd eq $aend} {
6506 set cached_isanc($b,$aend) 1
6507 set cached_isanc($a,$bend) 0
6508 return -1
6509 }
6510 if {![info exists anc($xd)]} {
6511 set anc($xd) $anc($x)
6512 lappend todo $xd
6513 } elseif {$anc($xd) ne $anc($x)} {
6514 set anc($xd) {}
6515 }
6516 }
6517 }
6518 set cached_isanc($a,$bend) 0
6519 set cached_isanc($b,$aend) 0
6520 return 0
6521}
6522
6523# This identifies whether $desc has an ancestor that is
6524# a growing tip of the graph and which is not an ancestor of $anc
6525# and returns 0 if so and 1 if not.
6526# If we subsequently discover a tag on such a growing tip, and that
6527# turns out to be a descendent of $anc (which it could, since we
6528# don't necessarily see children before parents), then $desc
6529# isn't a good choice to display as a descendent tag of
6530# $anc (since it is the descendent of another tag which is
6531# a descendent of $anc). Similarly, $anc isn't a good choice to
6532# display as a ancestor tag of $desc.
6533#
6534proc is_certain {desc anc} {
6535 global arcnos arcout arcstart arcend growing problems
6536
6537 set certain {}
6538 if {[llength $arcnos($anc)] == 1} {
6539 # tags on the same arc are certain
6540 if {$arcnos($desc) eq $arcnos($anc)} {
6541 return 1
6542 }
6543 if {![info exists arcout($anc)]} {
6544 # if $anc is partway along an arc, use the start of the arc instead
6545 set a [lindex $arcnos($anc) 0]
6546 set anc $arcstart($a)
6547 }
6548 }
6549 if {[llength $arcnos($desc)] > 1 || [info exists arcout($desc)]} {
6550 set x $desc
6551 } else {
6552 set a [lindex $arcnos($desc) 0]
6553 set x $arcend($a)
6554 }
6555 if {$x == $anc} {
6556 return 1
6557 }
6558 set anclist [list $x]
6559 set dl($x) 1
6560 set nnh 1
6561 set ngrowanc 0
6562 for {set i 0} {$i < [llength $anclist] && ($nnh > 0 || $ngrowanc > 0)} {incr i} {
6563 set x [lindex $anclist $i]
6564 if {$dl($x)} {
6565 incr nnh -1
6566 }
6567 set done($x) 1
6568 foreach a $arcout($x) {
6569 if {[info exists growing($a)]} {
6570 if {![info exists growanc($x)] && $dl($x)} {
6571 set growanc($x) 1
6572 incr ngrowanc
6573 }
6574 } else {
6575 set y $arcend($a)
6576 if {[info exists dl($y)]} {
6577 if {$dl($y)} {
6578 if {!$dl($x)} {
6579 set dl($y) 0
6580 if {![info exists done($y)]} {
6581 incr nnh -1
6582 }
6583 if {[info exists growanc($x)]} {
6584 incr ngrowanc -1
6585 }
6586 set xl [list $y]
6587 for {set k 0} {$k < [llength $xl]} {incr k} {
6588 set z [lindex $xl $k]
6589 foreach c $arcout($z) {
6590 if {[info exists arcend($c)]} {
6591 set v $arcend($c)
6592 if {[info exists dl($v)] && $dl($v)} {
6593 set dl($v) 0
6594 if {![info exists done($v)]} {
6595 incr nnh -1
6596 }
6597 if {[info exists growanc($v)]} {
6598 incr ngrowanc -1
6599 }
6600 lappend xl $v
6601 }
6602 }
6603 }
6604 }
6605 }
6606 }
6607 } elseif {$y eq $anc || !$dl($x)} {
6608 set dl($y) 0
6609 lappend anclist $y
6610 } else {
6611 set dl($y) 1
6612 lappend anclist $y
6613 incr nnh
6614 }
6615 }
6616 }
6617 }
6618 foreach x [array names growanc] {
6619 if {$dl($x)} {
6620 return 0
6621 }
6622 return 0
6623 }
6624 return 1
6625}
6626
6627proc validate_arctags {a} {
6628 global arctags idtags
6629
6630 set i -1
6631 set na $arctags($a)
6632 foreach id $arctags($a) {
6633 incr i
6634 if {![info exists idtags($id)]} {
6635 set na [lreplace $na $i $i]
6636 incr i -1
6637 }
6638 }
6639 set arctags($a) $na
6640}
6641
6642proc validate_archeads {a} {
6643 global archeads idheads
6644
6645 set i -1
6646 set na $archeads($a)
6647 foreach id $archeads($a) {
6648 incr i
6649 if {![info exists idheads($id)]} {
6650 set na [lreplace $na $i $i]
6651 incr i -1
6652 }
6653 }
6654 set archeads($a) $na
6655}
6656
6657# Return the list of IDs that have tags that are descendents of id,
6658# ignoring IDs that are descendents of IDs already reported.
6659proc desctags {id} {
6660 global arcnos arcstart arcids arctags idtags allparents
6661 global growing cached_dtags
6662
6663 if {![info exists allparents($id)]} {
6664 return {}
6665 }
6666 set t1 [clock clicks -milliseconds]
6667 set argid $id
6668 if {[llength $arcnos($id)] == 1 && [llength $allparents($id)] == 1} {
6669 # part-way along an arc; check that arc first
6670 set a [lindex $arcnos($id) 0]
6671 if {$arctags($a) ne {}} {
6672 validate_arctags $a
6673 set i [lsearch -exact $arcids($a) $id]
6674 set tid {}
6675 foreach t $arctags($a) {
6676 set j [lsearch -exact $arcids($a) $t]
6677 if {$j >= $i} break
6678 set tid $t
6679 }
6680 if {$tid ne {}} {
6681 return $tid
6682 }
6683 }
6684 set id $arcstart($a)
6685 if {[info exists idtags($id)]} {
6686 return $id
6687 }
6688 }
6689 if {[info exists cached_dtags($id)]} {
6690 return $cached_dtags($id)
6691 }
6692
6693 set origid $id
6694 set todo [list $id]
6695 set queued($id) 1
6696 set nc 1
6697 for {set i 0} {$i < [llength $todo] && $nc > 0} {incr i} {
6698 set id [lindex $todo $i]
6699 set done($id) 1
6700 set ta [info exists hastaggedancestor($id)]
6701 if {!$ta} {
6702 incr nc -1
6703 }
6704 # ignore tags on starting node
6705 if {!$ta && $i > 0} {
6706 if {[info exists idtags($id)]} {
6707 set tagloc($id) $id
6708 set ta 1
6709 } elseif {[info exists cached_dtags($id)]} {
6710 set tagloc($id) $cached_dtags($id)
6711 set ta 1
6712 }
6713 }
6714 foreach a $arcnos($id) {
6715 set d $arcstart($a)
6716 if {!$ta && $arctags($a) ne {}} {
6717 validate_arctags $a
6718 if {$arctags($a) ne {}} {
6719 lappend tagloc($id) [lindex $arctags($a) end]
6720 }
6721 }
6722 if {$ta || $arctags($a) ne {}} {
6723 set tomark [list $d]
6724 for {set j 0} {$j < [llength $tomark]} {incr j} {
6725 set dd [lindex $tomark $j]
6726 if {![info exists hastaggedancestor($dd)]} {
6727 if {[info exists done($dd)]} {
6728 foreach b $arcnos($dd) {
6729 lappend tomark $arcstart($b)
6730 }
6731 if {[info exists tagloc($dd)]} {
6732 unset tagloc($dd)
6733 }
6734 } elseif {[info exists queued($dd)]} {
6735 incr nc -1
6736 }
6737 set hastaggedancestor($dd) 1
6738 }
6739 }
6740 }
6741 if {![info exists queued($d)]} {
6742 lappend todo $d
6743 set queued($d) 1
6744 if {![info exists hastaggedancestor($d)]} {
6745 incr nc
6746 }
6747 }
6748 }
6749 }
6750 set tags {}
6751 foreach id [array names tagloc] {
6752 if {![info exists hastaggedancestor($id)]} {
6753 foreach t $tagloc($id) {
6754 if {[lsearch -exact $tags $t] < 0} {
6755 lappend tags $t
6756 }
6757 }
6758 }
6759 }
6760 set t2 [clock clicks -milliseconds]
6761 set loopix $i
6762
6763 # remove tags that are descendents of other tags
6764 for {set i 0} {$i < [llength $tags]} {incr i} {
6765 set a [lindex $tags $i]
6766 for {set j 0} {$j < $i} {incr j} {
6767 set b [lindex $tags $j]
6768 set r [anc_or_desc $a $b]
6769 if {$r == 1} {
6770 set tags [lreplace $tags $j $j]
6771 incr j -1
6772 incr i -1
6773 } elseif {$r == -1} {
6774 set tags [lreplace $tags $i $i]
6775 incr i -1
6776 break
6777 }
6778 }
6779 }
6780
6781 if {[array names growing] ne {}} {
6782 # graph isn't finished, need to check if any tag could get
6783 # eclipsed by another tag coming later. Simply ignore any
6784 # tags that could later get eclipsed.
6785 set ctags {}
6786 foreach t $tags {
6787 if {[is_certain $t $origid]} {
6788 lappend ctags $t
6789 }
6790 }
6791 if {$tags eq $ctags} {
6792 set cached_dtags($origid) $tags
6793 } else {
6794 set tags $ctags
6795 }
6796 } else {
6797 set cached_dtags($origid) $tags
6798 }
6799 set t3 [clock clicks -milliseconds]
6800 if {0 && $t3 - $t1 >= 100} {
6801 puts "iterating descendents ($loopix/[llength $todo] nodes) took\
6802 [expr {$t2-$t1}]+[expr {$t3-$t2}]ms, $nc candidates left"
6803 }
6804 return $tags
6805}
6806
6807proc anctags {id} {
6808 global arcnos arcids arcout arcend arctags idtags allparents
6809 global growing cached_atags
6810
6811 if {![info exists allparents($id)]} {
6812 return {}
6813 }
6814 set t1 [clock clicks -milliseconds]
6815 set argid $id
6816 if {[llength $arcnos($id)] == 1 && [llength $allparents($id)] == 1} {
6817 # part-way along an arc; check that arc first
6818 set a [lindex $arcnos($id) 0]
6819 if {$arctags($a) ne {}} {
6820 validate_arctags $a
6821 set i [lsearch -exact $arcids($a) $id]
6822 foreach t $arctags($a) {
6823 set j [lsearch -exact $arcids($a) $t]
6824 if {$j > $i} {
6825 return $t
6826 }
6827 }
6828 }
6829 if {![info exists arcend($a)]} {
6830 return {}
6831 }
6832 set id $arcend($a)
6833 if {[info exists idtags($id)]} {
6834 return $id
6835 }
6836 }
6837 if {[info exists cached_atags($id)]} {
6838 return $cached_atags($id)
6839 }
6840
6841 set origid $id
6842 set todo [list $id]
6843 set queued($id) 1
6844 set taglist {}
6845 set nc 1
6846 for {set i 0} {$i < [llength $todo] && $nc > 0} {incr i} {
6847 set id [lindex $todo $i]
6848 set done($id) 1
6849 set td [info exists hastaggeddescendent($id)]
6850 if {!$td} {
6851 incr nc -1
6852 }
6853 # ignore tags on starting node
6854 if {!$td && $i > 0} {
6855 if {[info exists idtags($id)]} {
6856 set tagloc($id) $id
6857 set td 1
6858 } elseif {[info exists cached_atags($id)]} {
6859 set tagloc($id) $cached_atags($id)
6860 set td 1
6861 }
6862 }
6863 foreach a $arcout($id) {
6864 if {!$td && $arctags($a) ne {}} {
6865 validate_arctags $a
6866 if {$arctags($a) ne {}} {
6867 lappend tagloc($id) [lindex $arctags($a) 0]
6868 }
6869 }
6870 if {![info exists arcend($a)]} continue
6871 set d $arcend($a)
6872 if {$td || $arctags($a) ne {}} {
6873 set tomark [list $d]
6874 for {set j 0} {$j < [llength $tomark]} {incr j} {
6875 set dd [lindex $tomark $j]
6876 if {![info exists hastaggeddescendent($dd)]} {
6877 if {[info exists done($dd)]} {
6878 foreach b $arcout($dd) {
6879 if {[info exists arcend($b)]} {
6880 lappend tomark $arcend($b)
6881 }
6882 }
6883 if {[info exists tagloc($dd)]} {
6884 unset tagloc($dd)
6885 }
6886 } elseif {[info exists queued($dd)]} {
6887 incr nc -1
6888 }
6889 set hastaggeddescendent($dd) 1
6890 }
6891 }
6892 }
6893 if {![info exists queued($d)]} {
6894 lappend todo $d
6895 set queued($d) 1
6896 if {![info exists hastaggeddescendent($d)]} {
6897 incr nc
6898 }
6899 }
6900 }
6901 }
6902 set t2 [clock clicks -milliseconds]
6903 set loopix $i
6904 set tags {}
6905 foreach id [array names tagloc] {
6906 if {![info exists hastaggeddescendent($id)]} {
6907 foreach t $tagloc($id) {
6908 if {[lsearch -exact $tags $t] < 0} {
6909 lappend tags $t
6910 }
6911 }
6912 }
6913 }
6914
6915 # remove tags that are ancestors of other tags
6916 for {set i 0} {$i < [llength $tags]} {incr i} {
6917 set a [lindex $tags $i]
6918 for {set j 0} {$j < $i} {incr j} {
6919 set b [lindex $tags $j]
6920 set r [anc_or_desc $a $b]
6921 if {$r == -1} {
6922 set tags [lreplace $tags $j $j]
6923 incr j -1
6924 incr i -1
6925 } elseif {$r == 1} {
6926 set tags [lreplace $tags $i $i]
6927 incr i -1
6928 break
6929 }
6930 }
6931 }
6932
6933 if {[array names growing] ne {}} {
6934 # graph isn't finished, need to check if any tag could get
6935 # eclipsed by another tag coming later. Simply ignore any
6936 # tags that could later get eclipsed.
6937 set ctags {}
6938 foreach t $tags {
6939 if {[is_certain $origid $t]} {
6940 lappend ctags $t
6941 }
6942 }
6943 if {$tags eq $ctags} {
6944 set cached_atags($origid) $tags
6945 } else {
6946 set tags $ctags
6947 }
6948 } else {
6949 set cached_atags($origid) $tags
6950 }
6951 set t3 [clock clicks -milliseconds]
6952 if {0 && $t3 - $t1 >= 100} {
6953 puts "iterating ancestors ($loopix/[llength $todo] nodes) took\
6954 [expr {$t2-$t1}]+[expr {$t3-$t2}]ms, $nc candidates left"
6955 }
6956 return $tags
6957}
6958
6959# Return the list of IDs that have heads that are descendents of id,
6960# including id itself if it has a head.
6961proc descheads {id} {
6962 global arcnos arcstart arcids archeads idheads cached_dheads
6963 global allparents
6964
6965 if {![info exists allparents($id)]} {
6966 return {}
6967 }
6968 set aret {}
6969 if {[llength $arcnos($id)] == 1 && [llength $allparents($id)] == 1} {
6970 # part-way along an arc; check it first
6971 set a [lindex $arcnos($id) 0]
6972 if {$archeads($a) ne {}} {
6973 validate_archeads $a
6974 set i [lsearch -exact $arcids($a) $id]
6975 foreach t $archeads($a) {
6976 set j [lsearch -exact $arcids($a) $t]
6977 if {$j > $i} break
6978 lappend aret $t
6979 }
6980 }
6981 set id $arcstart($a)
6982 }
6983 set origid $id
6984 set todo [list $id]
6985 set seen($id) 1
6986 set ret {}
6987 for {set i 0} {$i < [llength $todo]} {incr i} {
6988 set id [lindex $todo $i]
6989 if {[info exists cached_dheads($id)]} {
6990 set ret [concat $ret $cached_dheads($id)]
6991 } else {
6992 if {[info exists idheads($id)]} {
6993 lappend ret $id
6994 }
6995 foreach a $arcnos($id) {
6996 if {$archeads($a) ne {}} {
6997 validate_archeads $a
6998 if {$archeads($a) ne {}} {
6999 set ret [concat $ret $archeads($a)]
7000 }
7001 }
7002 set d $arcstart($a)
7003 if {![info exists seen($d)]} {
7004 lappend todo $d
7005 set seen($d) 1
7006 }
7007 }
7008 }
7009 }
7010 set ret [lsort -unique $ret]
7011 set cached_dheads($origid) $ret
7012 return [concat $ret $aret]
7013}
7014
7015proc addedtag {id} {
7016 global arcnos arcout cached_dtags cached_atags
7017
7018 if {![info exists arcnos($id)]} return
7019 if {![info exists arcout($id)]} {
7020 recalcarc [lindex $arcnos($id) 0]
7021 }
7022 catch {unset cached_dtags}
7023 catch {unset cached_atags}
7024}
7025
7026proc addedhead {hid head} {
7027 global arcnos arcout cached_dheads
7028
7029 if {![info exists arcnos($hid)]} return
7030 if {![info exists arcout($hid)]} {
7031 recalcarc [lindex $arcnos($hid) 0]
7032 }
7033 catch {unset cached_dheads}
7034}
7035
7036proc removedhead {hid head} {
7037 global cached_dheads
7038
7039 catch {unset cached_dheads}
7040}
7041
7042proc movedhead {hid head} {
7043 global arcnos arcout cached_dheads
7044
7045 if {![info exists arcnos($hid)]} return
7046 if {![info exists arcout($hid)]} {
7047 recalcarc [lindex $arcnos($hid) 0]
7048 }
7049 catch {unset cached_dheads}
7050}
7051
7052proc changedrefs {} {
7053 global cached_dheads cached_dtags cached_atags
7054 global arctags archeads arcnos arcout idheads idtags
7055
7056 foreach id [concat [array names idheads] [array names idtags]] {
7057 if {[info exists arcnos($id)] && ![info exists arcout($id)]} {
7058 set a [lindex $arcnos($id) 0]
7059 if {![info exists donearc($a)]} {
7060 recalcarc $a
7061 set donearc($a) 1
7062 }
7063 }
7064 }
7065 catch {unset cached_dtags}
7066 catch {unset cached_atags}
7067 catch {unset cached_dheads}
7068}
7069
7070proc rereadrefs {} {
7071 global idtags idheads idotherrefs mainhead
7072
7073 set refids [concat [array names idtags] \
7074 [array names idheads] [array names idotherrefs]]
7075 foreach id $refids {
7076 if {![info exists ref($id)]} {
7077 set ref($id) [listrefs $id]
7078 }
7079 }
7080 set oldmainhead $mainhead
7081 readrefs
7082 changedrefs
7083 set refids [lsort -unique [concat $refids [array names idtags] \
7084 [array names idheads] [array names idotherrefs]]]
7085 foreach id $refids {
7086 set v [listrefs $id]
7087 if {![info exists ref($id)] || $ref($id) != $v ||
7088 ($id eq $oldmainhead && $id ne $mainhead) ||
7089 ($id eq $mainhead && $id ne $oldmainhead)} {
7090 redrawtags $id
7091 }
7092 }
7093}
7094
7095proc listrefs {id} {
7096 global idtags idheads idotherrefs
7097
7098 set x {}
7099 if {[info exists idtags($id)]} {
7100 set x $idtags($id)
7101 }
7102 set y {}
7103 if {[info exists idheads($id)]} {
7104 set y $idheads($id)
7105 }
7106 set z {}
7107 if {[info exists idotherrefs($id)]} {
7108 set z $idotherrefs($id)
7109 }
7110 return [list $x $y $z]
7111}
7112
7113proc showtag {tag isnew} {
7114 global ctext tagcontents tagids linknum tagobjid
7115
7116 if {$isnew} {
7117 addtohistory [list showtag $tag 0]
7118 }
7119 $ctext conf -state normal
7120 clear_ctext
7121 set linknum 0
7122 if {![info exists tagcontents($tag)]} {
7123 catch {
7124 set tagcontents($tag) [exec git cat-file tag $tagobjid($tag)]
7125 }
7126 }
7127 if {[info exists tagcontents($tag)]} {
7128 set text $tagcontents($tag)
7129 } else {
7130 set text "Tag: $tag\nId: $tagids($tag)"
7131 }
7132 appendwithlinks $text {}
7133 $ctext conf -state disabled
7134 init_flist {}
7135}
7136
7137proc doquit {} {
7138 global stopped
7139 set stopped 100
7140 savestuff .
7141 destroy .
7142}
7143
7144proc doprefs {} {
7145 global maxwidth maxgraphpct diffopts
7146 global oldprefs prefstop showneartags showlocalchanges
7147 global bgcolor fgcolor ctext diffcolors selectbgcolor
7148 global uifont tabstop
7149
7150 set top .gitkprefs
7151 set prefstop $top
7152 if {[winfo exists $top]} {
7153 raise $top
7154 return
7155 }
7156 foreach v {maxwidth maxgraphpct diffopts showneartags showlocalchanges} {
7157 set oldprefs($v) [set $v]
7158 }
7159 toplevel $top
7160 wm title $top "Gitk preferences"
7161 label $top.ldisp -text "Commit list display options"
7162 $top.ldisp configure -font $uifont
7163 grid $top.ldisp - -sticky w -pady 10
7164 label $top.spacer -text " "
7165 label $top.maxwidthl -text "Maximum graph width (lines)" \
7166 -font optionfont
7167 spinbox $top.maxwidth -from 0 -to 100 -width 4 -textvariable maxwidth
7168 grid $top.spacer $top.maxwidthl $top.maxwidth -sticky w
7169 label $top.maxpctl -text "Maximum graph width (% of pane)" \
7170 -font optionfont
7171 spinbox $top.maxpct -from 1 -to 100 -width 4 -textvariable maxgraphpct
7172 grid x $top.maxpctl $top.maxpct -sticky w
7173 frame $top.showlocal
7174 label $top.showlocal.l -text "Show local changes" -font optionfont
7175 checkbutton $top.showlocal.b -variable showlocalchanges
7176 pack $top.showlocal.b $top.showlocal.l -side left
7177 grid x $top.showlocal -sticky w
7178
7179 label $top.ddisp -text "Diff display options"
7180 $top.ddisp configure -font $uifont
7181 grid $top.ddisp - -sticky w -pady 10
7182 label $top.diffoptl -text "Options for diff program" \
7183 -font optionfont
7184 entry $top.diffopt -width 20 -textvariable diffopts
7185 grid x $top.diffoptl $top.diffopt -sticky w
7186 frame $top.ntag
7187 label $top.ntag.l -text "Display nearby tags" -font optionfont
7188 checkbutton $top.ntag.b -variable showneartags
7189 pack $top.ntag.b $top.ntag.l -side left
7190 grid x $top.ntag -sticky w
7191 label $top.tabstopl -text "tabstop" -font optionfont
7192 spinbox $top.tabstop -from 1 -to 20 -width 4 -textvariable tabstop
7193 grid x $top.tabstopl $top.tabstop -sticky w
7194
7195 label $top.cdisp -text "Colors: press to choose"
7196 $top.cdisp configure -font $uifont
7197 grid $top.cdisp - -sticky w -pady 10
7198 label $top.bg -padx 40 -relief sunk -background $bgcolor
7199 button $top.bgbut -text "Background" -font optionfont \
7200 -command [list choosecolor bgcolor 0 $top.bg background setbg]
7201 grid x $top.bgbut $top.bg -sticky w
7202 label $top.fg -padx 40 -relief sunk -background $fgcolor
7203 button $top.fgbut -text "Foreground" -font optionfont \
7204 -command [list choosecolor fgcolor 0 $top.fg foreground setfg]
7205 grid x $top.fgbut $top.fg -sticky w
7206 label $top.diffold -padx 40 -relief sunk -background [lindex $diffcolors 0]
7207 button $top.diffoldbut -text "Diff: old lines" -font optionfont \
7208 -command [list choosecolor diffcolors 0 $top.diffold "diff old lines" \
7209 [list $ctext tag conf d0 -foreground]]
7210 grid x $top.diffoldbut $top.diffold -sticky w
7211 label $top.diffnew -padx 40 -relief sunk -background [lindex $diffcolors 1]
7212 button $top.diffnewbut -text "Diff: new lines" -font optionfont \
7213 -command [list choosecolor diffcolors 1 $top.diffnew "diff new lines" \
7214 [list $ctext tag conf d1 -foreground]]
7215 grid x $top.diffnewbut $top.diffnew -sticky w
7216 label $top.hunksep -padx 40 -relief sunk -background [lindex $diffcolors 2]
7217 button $top.hunksepbut -text "Diff: hunk header" -font optionfont \
7218 -command [list choosecolor diffcolors 2 $top.hunksep \
7219 "diff hunk header" \
7220 [list $ctext tag conf hunksep -foreground]]
7221 grid x $top.hunksepbut $top.hunksep -sticky w
7222 label $top.selbgsep -padx 40 -relief sunk -background $selectbgcolor
7223 button $top.selbgbut -text "Select bg" -font optionfont \
7224 -command [list choosecolor selectbgcolor 0 $top.selbgsep background setselbg]
7225 grid x $top.selbgbut $top.selbgsep -sticky w
7226
7227 frame $top.buts
7228 button $top.buts.ok -text "OK" -command prefsok -default active
7229 $top.buts.ok configure -font $uifont
7230 button $top.buts.can -text "Cancel" -command prefscan -default normal
7231 $top.buts.can configure -font $uifont
7232 grid $top.buts.ok $top.buts.can
7233 grid columnconfigure $top.buts 0 -weight 1 -uniform a
7234 grid columnconfigure $top.buts 1 -weight 1 -uniform a
7235 grid $top.buts - - -pady 10 -sticky ew
7236 bind $top <Visibility> "focus $top.buts.ok"
7237}
7238
7239proc choosecolor {v vi w x cmd} {
7240 global $v
7241
7242 set c [tk_chooseColor -initialcolor [lindex [set $v] $vi] \
7243 -title "Gitk: choose color for $x"]
7244 if {$c eq {}} return
7245 $w conf -background $c
7246 lset $v $vi $c
7247 eval $cmd $c
7248}
7249
7250proc setselbg {c} {
7251 global bglist cflist
7252 foreach w $bglist {
7253 $w configure -selectbackground $c
7254 }
7255 $cflist tag configure highlight \
7256 -background [$cflist cget -selectbackground]
7257 allcanvs itemconf secsel -fill $c
7258}
7259
7260proc setbg {c} {
7261 global bglist
7262
7263 foreach w $bglist {
7264 $w conf -background $c
7265 }
7266}
7267
7268proc setfg {c} {
7269 global fglist canv
7270
7271 foreach w $fglist {
7272 $w conf -foreground $c
7273 }
7274 allcanvs itemconf text -fill $c
7275 $canv itemconf circle -outline $c
7276}
7277
7278proc prefscan {} {
7279 global maxwidth maxgraphpct diffopts
7280 global oldprefs prefstop showneartags showlocalchanges
7281
7282 foreach v {maxwidth maxgraphpct diffopts showneartags showlocalchanges} {
7283 set $v $oldprefs($v)
7284 }
7285 catch {destroy $prefstop}
7286 unset prefstop
7287}
7288
7289proc prefsok {} {
7290 global maxwidth maxgraphpct
7291 global oldprefs prefstop showneartags showlocalchanges
7292 global charspc ctext tabstop
7293
7294 catch {destroy $prefstop}
7295 unset prefstop
7296 $ctext configure -tabs "[expr {$tabstop * $charspc}]"
7297 if {$showlocalchanges != $oldprefs(showlocalchanges)} {
7298 if {$showlocalchanges} {
7299 doshowlocalchanges
7300 } else {
7301 dohidelocalchanges
7302 }
7303 }
7304 if {$maxwidth != $oldprefs(maxwidth)
7305 || $maxgraphpct != $oldprefs(maxgraphpct)} {
7306 redisplay
7307 } elseif {$showneartags != $oldprefs(showneartags)} {
7308 reselectline
7309 }
7310}
7311
7312proc formatdate {d} {
7313 if {$d ne {}} {
7314 set d [clock format $d -format "%Y-%m-%d %H:%M:%S"]
7315 }
7316 return $d
7317}
7318
7319# This list of encoding names and aliases is distilled from
7320# http://www.iana.org/assignments/character-sets.
7321# Not all of them are supported by Tcl.
7322set encoding_aliases {
7323 { ANSI_X3.4-1968 iso-ir-6 ANSI_X3.4-1986 ISO_646.irv:1991 ASCII
7324 ISO646-US US-ASCII us IBM367 cp367 csASCII }
7325 { ISO-10646-UTF-1 csISO10646UTF1 }
7326 { ISO_646.basic:1983 ref csISO646basic1983 }
7327 { INVARIANT csINVARIANT }
7328 { ISO_646.irv:1983 iso-ir-2 irv csISO2IntlRefVersion }
7329 { BS_4730 iso-ir-4 ISO646-GB gb uk csISO4UnitedKingdom }
7330 { NATS-SEFI iso-ir-8-1 csNATSSEFI }
7331 { NATS-SEFI-ADD iso-ir-8-2 csNATSSEFIADD }
7332 { NATS-DANO iso-ir-9-1 csNATSDANO }
7333 { NATS-DANO-ADD iso-ir-9-2 csNATSDANOADD }
7334 { SEN_850200_B iso-ir-10 FI ISO646-FI ISO646-SE se csISO10Swedish }
7335 { SEN_850200_C iso-ir-11 ISO646-SE2 se2 csISO11SwedishForNames }
7336 { KS_C_5601-1987 iso-ir-149 KS_C_5601-1989 KSC_5601 korean csKSC56011987 }
7337 { ISO-2022-KR csISO2022KR }
7338 { EUC-KR csEUCKR }
7339 { ISO-2022-JP csISO2022JP }
7340 { ISO-2022-JP-2 csISO2022JP2 }
7341 { JIS_C6220-1969-jp JIS_C6220-1969 iso-ir-13 katakana x0201-7
7342 csISO13JISC6220jp }
7343 { JIS_C6220-1969-ro iso-ir-14 jp ISO646-JP csISO14JISC6220ro }
7344 { IT iso-ir-15 ISO646-IT csISO15Italian }
7345 { PT iso-ir-16 ISO646-PT csISO16Portuguese }
7346 { ES iso-ir-17 ISO646-ES csISO17Spanish }
7347 { greek7-old iso-ir-18 csISO18Greek7Old }
7348 { latin-greek iso-ir-19 csISO19LatinGreek }
7349 { DIN_66003 iso-ir-21 de ISO646-DE csISO21German }
7350 { NF_Z_62-010_(1973) iso-ir-25 ISO646-FR1 csISO25French }
7351 { Latin-greek-1 iso-ir-27 csISO27LatinGreek1 }
7352 { ISO_5427 iso-ir-37 csISO5427Cyrillic }
7353 { JIS_C6226-1978 iso-ir-42 csISO42JISC62261978 }
7354 { BS_viewdata iso-ir-47 csISO47BSViewdata }
7355 { INIS iso-ir-49 csISO49INIS }
7356 { INIS-8 iso-ir-50 csISO50INIS8 }
7357 { INIS-cyrillic iso-ir-51 csISO51INISCyrillic }
7358 { ISO_5427:1981 iso-ir-54 ISO5427Cyrillic1981 }
7359 { ISO_5428:1980 iso-ir-55 csISO5428Greek }
7360 { GB_1988-80 iso-ir-57 cn ISO646-CN csISO57GB1988 }
7361 { GB_2312-80 iso-ir-58 chinese csISO58GB231280 }
7362 { NS_4551-1 iso-ir-60 ISO646-NO no csISO60DanishNorwegian
7363 csISO60Norwegian1 }
7364 { NS_4551-2 ISO646-NO2 iso-ir-61 no2 csISO61Norwegian2 }
7365 { NF_Z_62-010 iso-ir-69 ISO646-FR fr csISO69French }
7366 { videotex-suppl iso-ir-70 csISO70VideotexSupp1 }
7367 { PT2 iso-ir-84 ISO646-PT2 csISO84Portuguese2 }
7368 { ES2 iso-ir-85 ISO646-ES2 csISO85Spanish2 }
7369 { MSZ_7795.3 iso-ir-86 ISO646-HU hu csISO86Hungarian }
7370 { JIS_C6226-1983 iso-ir-87 x0208 JIS_X0208-1983 csISO87JISX0208 }
7371 { greek7 iso-ir-88 csISO88Greek7 }
7372 { ASMO_449 ISO_9036 arabic7 iso-ir-89 csISO89ASMO449 }
7373 { iso-ir-90 csISO90 }
7374 { JIS_C6229-1984-a iso-ir-91 jp-ocr-a csISO91JISC62291984a }
7375 { JIS_C6229-1984-b iso-ir-92 ISO646-JP-OCR-B jp-ocr-b
7376 csISO92JISC62991984b }
7377 { JIS_C6229-1984-b-add iso-ir-93 jp-ocr-b-add csISO93JIS62291984badd }
7378 { JIS_C6229-1984-hand iso-ir-94 jp-ocr-hand csISO94JIS62291984hand }
7379 { JIS_C6229-1984-hand-add iso-ir-95 jp-ocr-hand-add
7380 csISO95JIS62291984handadd }
7381 { JIS_C6229-1984-kana iso-ir-96 csISO96JISC62291984kana }
7382 { ISO_2033-1983 iso-ir-98 e13b csISO2033 }
7383 { ANSI_X3.110-1983 iso-ir-99 CSA_T500-1983 NAPLPS csISO99NAPLPS }
7384 { ISO_8859-1:1987 iso-ir-100 ISO_8859-1 ISO-8859-1 latin1 l1 IBM819
7385 CP819 csISOLatin1 }
7386 { ISO_8859-2:1987 iso-ir-101 ISO_8859-2 ISO-8859-2 latin2 l2 csISOLatin2 }
7387 { T.61-7bit iso-ir-102 csISO102T617bit }
7388 { T.61-8bit T.61 iso-ir-103 csISO103T618bit }
7389 { ISO_8859-3:1988 iso-ir-109 ISO_8859-3 ISO-8859-3 latin3 l3 csISOLatin3 }
7390 { ISO_8859-4:1988 iso-ir-110 ISO_8859-4 ISO-8859-4 latin4 l4 csISOLatin4 }
7391 { ECMA-cyrillic iso-ir-111 KOI8-E csISO111ECMACyrillic }
7392 { CSA_Z243.4-1985-1 iso-ir-121 ISO646-CA csa7-1 ca csISO121Canadian1 }
7393 { CSA_Z243.4-1985-2 iso-ir-122 ISO646-CA2 csa7-2 csISO122Canadian2 }
7394 { CSA_Z243.4-1985-gr iso-ir-123 csISO123CSAZ24341985gr }
7395 { ISO_8859-6:1987 iso-ir-127 ISO_8859-6 ISO-8859-6 ECMA-114 ASMO-708
7396 arabic csISOLatinArabic }
7397 { ISO_8859-6-E csISO88596E ISO-8859-6-E }
7398 { ISO_8859-6-I csISO88596I ISO-8859-6-I }
7399 { ISO_8859-7:1987 iso-ir-126 ISO_8859-7 ISO-8859-7 ELOT_928 ECMA-118
7400 greek greek8 csISOLatinGreek }
7401 { T.101-G2 iso-ir-128 csISO128T101G2 }
7402 { ISO_8859-8:1988 iso-ir-138 ISO_8859-8 ISO-8859-8 hebrew
7403 csISOLatinHebrew }
7404 { ISO_8859-8-E csISO88598E ISO-8859-8-E }
7405 { ISO_8859-8-I csISO88598I ISO-8859-8-I }
7406 { CSN_369103 iso-ir-139 csISO139CSN369103 }
7407 { JUS_I.B1.002 iso-ir-141 ISO646-YU js yu csISO141JUSIB1002 }
7408 { ISO_6937-2-add iso-ir-142 csISOTextComm }
7409 { IEC_P27-1 iso-ir-143 csISO143IECP271 }
7410 { ISO_8859-5:1988 iso-ir-144 ISO_8859-5 ISO-8859-5 cyrillic
7411 csISOLatinCyrillic }
7412 { JUS_I.B1.003-serb iso-ir-146 serbian csISO146Serbian }
7413 { JUS_I.B1.003-mac macedonian iso-ir-147 csISO147Macedonian }
7414 { ISO_8859-9:1989 iso-ir-148 ISO_8859-9 ISO-8859-9 latin5 l5 csISOLatin5 }
7415 { greek-ccitt iso-ir-150 csISO150 csISO150GreekCCITT }
7416 { NC_NC00-10:81 cuba iso-ir-151 ISO646-CU csISO151Cuba }
7417 { ISO_6937-2-25 iso-ir-152 csISO6937Add }
7418 { GOST_19768-74 ST_SEV_358-88 iso-ir-153 csISO153GOST1976874 }
7419 { ISO_8859-supp iso-ir-154 latin1-2-5 csISO8859Supp }
7420 { ISO_10367-box iso-ir-155 csISO10367Box }
7421 { ISO-8859-10 iso-ir-157 l6 ISO_8859-10:1992 csISOLatin6 latin6 }
7422 { latin-lap lap iso-ir-158 csISO158Lap }
7423 { JIS_X0212-1990 x0212 iso-ir-159 csISO159JISX02121990 }
7424 { DS_2089 DS2089 ISO646-DK dk csISO646Danish }
7425 { us-dk csUSDK }
7426 { dk-us csDKUS }
7427 { JIS_X0201 X0201 csHalfWidthKatakana }
7428 { KSC5636 ISO646-KR csKSC5636 }
7429 { ISO-10646-UCS-2 csUnicode }
7430 { ISO-10646-UCS-4 csUCS4 }
7431 { DEC-MCS dec csDECMCS }
7432 { hp-roman8 roman8 r8 csHPRoman8 }
7433 { macintosh mac csMacintosh }
7434 { IBM037 cp037 ebcdic-cp-us ebcdic-cp-ca ebcdic-cp-wt ebcdic-cp-nl
7435 csIBM037 }
7436 { IBM038 EBCDIC-INT cp038 csIBM038 }
7437 { IBM273 CP273 csIBM273 }
7438 { IBM274 EBCDIC-BE CP274 csIBM274 }
7439 { IBM275 EBCDIC-BR cp275 csIBM275 }
7440 { IBM277 EBCDIC-CP-DK EBCDIC-CP-NO csIBM277 }
7441 { IBM278 CP278 ebcdic-cp-fi ebcdic-cp-se csIBM278 }
7442 { IBM280 CP280 ebcdic-cp-it csIBM280 }
7443 { IBM281 EBCDIC-JP-E cp281 csIBM281 }
7444 { IBM284 CP284 ebcdic-cp-es csIBM284 }
7445 { IBM285 CP285 ebcdic-cp-gb csIBM285 }
7446 { IBM290 cp290 EBCDIC-JP-kana csIBM290 }
7447 { IBM297 cp297 ebcdic-cp-fr csIBM297 }
7448 { IBM420 cp420 ebcdic-cp-ar1 csIBM420 }
7449 { IBM423 cp423 ebcdic-cp-gr csIBM423 }
7450 { IBM424 cp424 ebcdic-cp-he csIBM424 }
7451 { IBM437 cp437 437 csPC8CodePage437 }
7452 { IBM500 CP500 ebcdic-cp-be ebcdic-cp-ch csIBM500 }
7453 { IBM775 cp775 csPC775Baltic }
7454 { IBM850 cp850 850 csPC850Multilingual }
7455 { IBM851 cp851 851 csIBM851 }
7456 { IBM852 cp852 852 csPCp852 }
7457 { IBM855 cp855 855 csIBM855 }
7458 { IBM857 cp857 857 csIBM857 }
7459 { IBM860 cp860 860 csIBM860 }
7460 { IBM861 cp861 861 cp-is csIBM861 }
7461 { IBM862 cp862 862 csPC862LatinHebrew }
7462 { IBM863 cp863 863 csIBM863 }
7463 { IBM864 cp864 csIBM864 }
7464 { IBM865 cp865 865 csIBM865 }
7465 { IBM866 cp866 866 csIBM866 }
7466 { IBM868 CP868 cp-ar csIBM868 }
7467 { IBM869 cp869 869 cp-gr csIBM869 }
7468 { IBM870 CP870 ebcdic-cp-roece ebcdic-cp-yu csIBM870 }
7469 { IBM871 CP871 ebcdic-cp-is csIBM871 }
7470 { IBM880 cp880 EBCDIC-Cyrillic csIBM880 }
7471 { IBM891 cp891 csIBM891 }
7472 { IBM903 cp903 csIBM903 }
7473 { IBM904 cp904 904 csIBBM904 }
7474 { IBM905 CP905 ebcdic-cp-tr csIBM905 }
7475 { IBM918 CP918 ebcdic-cp-ar2 csIBM918 }
7476 { IBM1026 CP1026 csIBM1026 }
7477 { EBCDIC-AT-DE csIBMEBCDICATDE }
7478 { EBCDIC-AT-DE-A csEBCDICATDEA }
7479 { EBCDIC-CA-FR csEBCDICCAFR }
7480 { EBCDIC-DK-NO csEBCDICDKNO }
7481 { EBCDIC-DK-NO-A csEBCDICDKNOA }
7482 { EBCDIC-FI-SE csEBCDICFISE }
7483 { EBCDIC-FI-SE-A csEBCDICFISEA }
7484 { EBCDIC-FR csEBCDICFR }
7485 { EBCDIC-IT csEBCDICIT }
7486 { EBCDIC-PT csEBCDICPT }
7487 { EBCDIC-ES csEBCDICES }
7488 { EBCDIC-ES-A csEBCDICESA }
7489 { EBCDIC-ES-S csEBCDICESS }
7490 { EBCDIC-UK csEBCDICUK }
7491 { EBCDIC-US csEBCDICUS }
7492 { UNKNOWN-8BIT csUnknown8BiT }
7493 { MNEMONIC csMnemonic }
7494 { MNEM csMnem }
7495 { VISCII csVISCII }
7496 { VIQR csVIQR }
7497 { KOI8-R csKOI8R }
7498 { IBM00858 CCSID00858 CP00858 PC-Multilingual-850+euro }
7499 { IBM00924 CCSID00924 CP00924 ebcdic-Latin9--euro }
7500 { IBM01140 CCSID01140 CP01140 ebcdic-us-37+euro }
7501 { IBM01141 CCSID01141 CP01141 ebcdic-de-273+euro }
7502 { IBM01142 CCSID01142 CP01142 ebcdic-dk-277+euro ebcdic-no-277+euro }
7503 { IBM01143 CCSID01143 CP01143 ebcdic-fi-278+euro ebcdic-se-278+euro }
7504 { IBM01144 CCSID01144 CP01144 ebcdic-it-280+euro }
7505 { IBM01145 CCSID01145 CP01145 ebcdic-es-284+euro }
7506 { IBM01146 CCSID01146 CP01146 ebcdic-gb-285+euro }
7507 { IBM01147 CCSID01147 CP01147 ebcdic-fr-297+euro }
7508 { IBM01148 CCSID01148 CP01148 ebcdic-international-500+euro }
7509 { IBM01149 CCSID01149 CP01149 ebcdic-is-871+euro }
7510 { IBM1047 IBM-1047 }
7511 { PTCP154 csPTCP154 PT154 CP154 Cyrillic-Asian }
7512 { Amiga-1251 Ami1251 Amiga1251 Ami-1251 }
7513 { UNICODE-1-1 csUnicode11 }
7514 { CESU-8 csCESU-8 }
7515 { BOCU-1 csBOCU-1 }
7516 { UNICODE-1-1-UTF-7 csUnicode11UTF7 }
7517 { ISO-8859-14 iso-ir-199 ISO_8859-14:1998 ISO_8859-14 latin8 iso-celtic
7518 l8 }
7519 { ISO-8859-15 ISO_8859-15 Latin-9 }
7520 { ISO-8859-16 iso-ir-226 ISO_8859-16:2001 ISO_8859-16 latin10 l10 }
7521 { GBK CP936 MS936 windows-936 }
7522 { JIS_Encoding csJISEncoding }
7523 { Shift_JIS MS_Kanji csShiftJIS }
7524 { Extended_UNIX_Code_Packed_Format_for_Japanese csEUCPkdFmtJapanese
7525 EUC-JP }
7526 { Extended_UNIX_Code_Fixed_Width_for_Japanese csEUCFixWidJapanese }
7527 { ISO-10646-UCS-Basic csUnicodeASCII }
7528 { ISO-10646-Unicode-Latin1 csUnicodeLatin1 ISO-10646 }
7529 { ISO-Unicode-IBM-1261 csUnicodeIBM1261 }
7530 { ISO-Unicode-IBM-1268 csUnicodeIBM1268 }
7531 { ISO-Unicode-IBM-1276 csUnicodeIBM1276 }
7532 { ISO-Unicode-IBM-1264 csUnicodeIBM1264 }
7533 { ISO-Unicode-IBM-1265 csUnicodeIBM1265 }
7534 { ISO-8859-1-Windows-3.0-Latin-1 csWindows30Latin1 }
7535 { ISO-8859-1-Windows-3.1-Latin-1 csWindows31Latin1 }
7536 { ISO-8859-2-Windows-Latin-2 csWindows31Latin2 }
7537 { ISO-8859-9-Windows-Latin-5 csWindows31Latin5 }
7538 { Adobe-Standard-Encoding csAdobeStandardEncoding }
7539 { Ventura-US csVenturaUS }
7540 { Ventura-International csVenturaInternational }
7541 { PC8-Danish-Norwegian csPC8DanishNorwegian }
7542 { PC8-Turkish csPC8Turkish }
7543 { IBM-Symbols csIBMSymbols }
7544 { IBM-Thai csIBMThai }
7545 { HP-Legal csHPLegal }
7546 { HP-Pi-font csHPPiFont }
7547 { HP-Math8 csHPMath8 }
7548 { Adobe-Symbol-Encoding csHPPSMath }
7549 { HP-DeskTop csHPDesktop }
7550 { Ventura-Math csVenturaMath }
7551 { Microsoft-Publishing csMicrosoftPublishing }
7552 { Windows-31J csWindows31J }
7553 { GB2312 csGB2312 }
7554 { Big5 csBig5 }
7555}
7556
7557proc tcl_encoding {enc} {
7558 global encoding_aliases
7559 set names [encoding names]
7560 set lcnames [string tolower $names]
7561 set enc [string tolower $enc]
7562 set i [lsearch -exact $lcnames $enc]
7563 if {$i < 0} {
7564 # look for "isonnn" instead of "iso-nnn" or "iso_nnn"
7565 if {[regsub {^iso[-_]} $enc iso encx]} {
7566 set i [lsearch -exact $lcnames $encx]
7567 }
7568 }
7569 if {$i < 0} {
7570 foreach l $encoding_aliases {
7571 set ll [string tolower $l]
7572 if {[lsearch -exact $ll $enc] < 0} continue
7573 # look through the aliases for one that tcl knows about
7574 foreach e $ll {
7575 set i [lsearch -exact $lcnames $e]
7576 if {$i < 0} {
7577 if {[regsub {^iso[-_]} $e iso ex]} {
7578 set i [lsearch -exact $lcnames $ex]
7579 }
7580 }
7581 if {$i >= 0} break
7582 }
7583 break
7584 }
7585 }
7586 if {$i >= 0} {
7587 return [lindex $names $i]
7588 }
7589 return {}
7590}
7591
7592# defaults...
7593set datemode 0
7594set diffopts "-U 5 -p"
7595set wrcomcmd "git diff-tree --stdin -p --pretty"
7596
7597set gitencoding {}
7598catch {
7599 set gitencoding [exec git config --get i18n.commitencoding]
7600}
7601if {$gitencoding == ""} {
7602 set gitencoding "utf-8"
7603}
7604set tclencoding [tcl_encoding $gitencoding]
7605if {$tclencoding == {}} {
7606 puts stderr "Warning: encoding $gitencoding is not supported by Tcl/Tk"
7607}
7608
7609set mainfont {Helvetica 9}
7610set textfont {Courier 9}
7611set uifont {Helvetica 9 bold}
7612set tabstop 8
7613set findmergefiles 0
7614set maxgraphpct 50
7615set maxwidth 16
7616set revlistorder 0
7617set fastdate 0
7618set uparrowlen 7
7619set downarrowlen 7
7620set mingaplen 30
7621set cmitmode "patch"
7622set wrapcomment "none"
7623set showneartags 1
7624set maxrefs 20
7625set maxlinelen 200
7626set showlocalchanges 1
7627
7628set colors {green red blue magenta darkgrey brown orange}
7629set bgcolor white
7630set fgcolor black
7631set diffcolors {red "#00a000" blue}
7632set selectbgcolor gray85
7633
7634catch {source ~/.gitk}
7635
7636font create optionfont -family sans-serif -size -12
7637
7638# check that we can find a .git directory somewhere...
7639if {[catch {set gitdir [gitdir]}]} {
7640 show_error {} . "Cannot find a git repository here."
7641 exit 1
7642}
7643if {![file isdirectory $gitdir]} {
7644 show_error {} . "Cannot find the git directory \"$gitdir\"."
7645 exit 1
7646}
7647
7648set revtreeargs {}
7649set cmdline_files {}
7650set i 0
7651foreach arg $argv {
7652 switch -- $arg {
7653 "" { }
7654 "-d" { set datemode 1 }
7655 "--" {
7656 set cmdline_files [lrange $argv [expr {$i + 1}] end]
7657 break
7658 }
7659 default {
7660 lappend revtreeargs $arg
7661 }
7662 }
7663 incr i
7664}
7665
7666if {$i >= [llength $argv] && $revtreeargs ne {}} {
7667 # no -- on command line, but some arguments (other than -d)
7668 if {[catch {
7669 set f [eval exec git rev-parse --no-revs --no-flags $revtreeargs]
7670 set cmdline_files [split $f "\n"]
7671 set n [llength $cmdline_files]
7672 set revtreeargs [lrange $revtreeargs 0 end-$n]
7673 # Unfortunately git rev-parse doesn't produce an error when
7674 # something is both a revision and a filename. To be consistent
7675 # with git log and git rev-list, check revtreeargs for filenames.
7676 foreach arg $revtreeargs {
7677 if {[file exists $arg]} {
7678 show_error {} . "Ambiguous argument '$arg': both revision\
7679 and filename"
7680 exit 1
7681 }
7682 }
7683 } err]} {
7684 # unfortunately we get both stdout and stderr in $err,
7685 # so look for "fatal:".
7686 set i [string first "fatal:" $err]
7687 if {$i > 0} {
7688 set err [string range $err [expr {$i + 6}] end]
7689 }
7690 show_error {} . "Bad arguments to gitk:\n$err"
7691 exit 1
7692 }
7693}
7694
7695set nullid "0000000000000000000000000000000000000000"
7696set nullid2 "0000000000000000000000000000000000000001"
7697
7698
7699set runq {}
7700set history {}
7701set historyindex 0
7702set fh_serial 0
7703set nhl_names {}
7704set highlight_paths {}
7705set searchdirn -forwards
7706set boldrows {}
7707set boldnamerows {}
7708set diffelide {0 0}
7709set markingmatches 0
7710
7711set optim_delay 16
7712
7713set nextviewnum 1
7714set curview 0
7715set selectedview 0
7716set selectedhlview None
7717set viewfiles(0) {}
7718set viewperm(0) 0
7719set viewargs(0) {}
7720
7721set cmdlineok 0
7722set stopped 0
7723set stuffsaved 0
7724set patchnum 0
7725set lookingforhead 0
7726set localirow -1
7727set localfrow -1
7728set lserial 0
7729setcoords
7730makewindow
7731# wait for the window to become visible
7732tkwait visibility .
7733wm title . "[file tail $argv0]: [file tail [pwd]]"
7734readrefs
7735
7736if {$cmdline_files ne {} || $revtreeargs ne {}} {
7737 # create a view for the files/dirs specified on the command line
7738 set curview 1
7739 set selectedview 1
7740 set nextviewnum 2
7741 set viewname(1) "Command line"
7742 set viewfiles(1) $cmdline_files
7743 set viewargs(1) $revtreeargs
7744 set viewperm(1) 0
7745 addviewmenu 1
7746 .bar.view entryconf Edit* -state normal
7747 .bar.view entryconf Delete* -state normal
7748}
7749
7750if {[info exists permviews]} {
7751 foreach v $permviews {
7752 set n $nextviewnum
7753 incr nextviewnum
7754 set viewname($n) [lindex $v 0]
7755 set viewfiles($n) [lindex $v 1]
7756 set viewargs($n) [lindex $v 2]
7757 set viewperm($n) 1
7758 addviewmenu $n
7759 }
7760}
7761getcommits