1#!/bin/sh
2# Tcl ignores the next line -*- tcl -*- \
3exec wish "$0" -- "$@"
4
5# Copyright (C) 2005 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 ".git"
16 }
17}
18
19proc parse_args {rargs} {
20 global parsed_args
21
22 if {[catch {
23 set parse_args [concat --default HEAD $rargs]
24 set parsed_args [split [eval exec git-rev-parse $parse_args] "\n"]
25 }]} {
26 # if git-rev-parse failed for some reason...
27 if {$rargs == {}} {
28 set rargs HEAD
29 }
30 set parsed_args $rargs
31 }
32 return $parsed_args
33}
34
35proc start_rev_list {rlargs} {
36 global startmsecs nextupdate ncmupdate
37 global commfd leftover tclencoding datemode
38
39 set startmsecs [clock clicks -milliseconds]
40 set nextupdate [expr {$startmsecs + 100}]
41 set ncmupdate 1
42 initlayout
43 set order "--topo-order"
44 if {$datemode} {
45 set order "--date-order"
46 }
47 if {[catch {
48 set commfd [open [concat | git-rev-list --header $order \
49 --parents $rlargs] r]
50 } err]} {
51 puts stderr "Error executing git-rev-list: $err"
52 exit 1
53 }
54 set leftover {}
55 fconfigure $commfd -blocking 0 -translation lf
56 if {$tclencoding != {}} {
57 fconfigure $commfd -encoding $tclencoding
58 }
59 fileevent $commfd readable [list getcommitlines $commfd]
60 . config -cursor watch
61 settextcursor watch
62}
63
64proc getcommits {rargs} {
65 global phase canv mainfont
66
67 set phase getcommits
68 start_rev_list [parse_args $rargs]
69 $canv delete all
70 $canv create text 3 3 -anchor nw -text "Reading commits..." \
71 -font $mainfont -tags textitems
72}
73
74proc getcommitlines {commfd} {
75 global commitlisted nextupdate
76 global leftover
77 global displayorder commitidx commitrow commitdata
78
79 set stuff [read $commfd]
80 if {$stuff == {}} {
81 if {![eof $commfd]} return
82 # set it blocking so we wait for the process to terminate
83 fconfigure $commfd -blocking 1
84 if {![catch {close $commfd} err]} {
85 after idle finishcommits
86 return
87 }
88 if {[string range $err 0 4] == "usage"} {
89 set err \
90 "Gitk: error reading commits: bad arguments to git-rev-list.\
91 (Note: arguments to gitk are passed to git-rev-list\
92 to allow selection of commits to be displayed.)"
93 } else {
94 set err "Error reading commits: $err"
95 }
96 error_popup $err
97 exit 1
98 }
99 set start 0
100 set gotsome 0
101 while 1 {
102 set i [string first "\0" $stuff $start]
103 if {$i < 0} {
104 append leftover [string range $stuff $start end]
105 break
106 }
107 if {$start == 0} {
108 set cmit $leftover
109 append cmit [string range $stuff 0 [expr {$i - 1}]]
110 set leftover {}
111 } else {
112 set cmit [string range $stuff $start [expr {$i - 1}]]
113 }
114 set start [expr {$i + 1}]
115 set j [string first "\n" $cmit]
116 set ok 0
117 if {$j >= 0} {
118 set ids [string range $cmit 0 [expr {$j - 1}]]
119 set ok 1
120 foreach id $ids {
121 if {[string length $id] != 40} {
122 set ok 0
123 break
124 }
125 }
126 }
127 if {!$ok} {
128 set shortcmit $cmit
129 if {[string length $shortcmit] > 80} {
130 set shortcmit "[string range $shortcmit 0 80]..."
131 }
132 error_popup "Can't parse git-rev-list output: {$shortcmit}"
133 exit 1
134 }
135 set id [lindex $ids 0]
136 set olds [lrange $ids 1 end]
137 set commitlisted($id) 1
138 updatechildren $id $olds
139 set commitdata($id) [string range $cmit [expr {$j + 1}] end]
140 set commitrow($id) $commitidx
141 incr commitidx
142 lappend displayorder $id
143 set gotsome 1
144 }
145 if {$gotsome} {
146 layoutmore
147 }
148 if {[clock clicks -milliseconds] >= $nextupdate} {
149 doupdate 1
150 }
151}
152
153proc doupdate {reading} {
154 global commfd nextupdate numcommits ncmupdate
155
156 if {$reading} {
157 fileevent $commfd readable {}
158 }
159 update
160 set nextupdate [expr {[clock clicks -milliseconds] + 100}]
161 if {$numcommits < 100} {
162 set ncmupdate [expr {$numcommits + 1}]
163 } elseif {$numcommits < 10000} {
164 set ncmupdate [expr {$numcommits + 10}]
165 } else {
166 set ncmupdate [expr {$numcommits + 100}]
167 }
168 if {$reading} {
169 fileevent $commfd readable [list getcommitlines $commfd]
170 }
171}
172
173proc readcommit {id} {
174 if {[catch {set contents [exec git-cat-file commit $id]}]} return
175 updatechildren $id {}
176 parsecommit $id $contents 0
177}
178
179proc updatecommits {rargs} {
180 stopfindproc
181 foreach v {children nchildren parents nparents commitlisted
182 colormap selectedline matchinglines treediffs
183 mergefilelist currentid rowtextx commitrow
184 rowidlist rowoffsets idrowranges idrangedrawn iddrawn
185 linesegends crossings cornercrossings} {
186 global $v
187 catch {unset $v}
188 }
189 allcanvs delete all
190 readrefs
191 getcommits $rargs
192}
193
194proc updatechildren {id olds} {
195 global children nchildren parents nparents
196
197 if {![info exists nchildren($id)]} {
198 set children($id) {}
199 set nchildren($id) 0
200 }
201 set parents($id) $olds
202 set nparents($id) [llength $olds]
203 foreach p $olds {
204 if {![info exists nchildren($p)]} {
205 set children($p) [list $id]
206 set nchildren($p) 1
207 } elseif {[lsearch -exact $children($p) $id] < 0} {
208 lappend children($p) $id
209 incr nchildren($p)
210 }
211 }
212}
213
214proc parsecommit {id contents listed} {
215 global commitinfo cdate
216
217 set inhdr 1
218 set comment {}
219 set headline {}
220 set auname {}
221 set audate {}
222 set comname {}
223 set comdate {}
224 set hdrend [string first "\n\n" $contents]
225 if {$hdrend < 0} {
226 # should never happen...
227 set hdrend [string length $contents]
228 }
229 set header [string range $contents 0 [expr {$hdrend - 1}]]
230 set comment [string range $contents [expr {$hdrend + 2}] end]
231 foreach line [split $header "\n"] {
232 set tag [lindex $line 0]
233 if {$tag == "author"} {
234 set audate [lindex $line end-1]
235 set auname [lrange $line 1 end-2]
236 } elseif {$tag == "committer"} {
237 set comdate [lindex $line end-1]
238 set comname [lrange $line 1 end-2]
239 }
240 }
241 set headline {}
242 # take the first line of the comment as the headline
243 set i [string first "\n" $comment]
244 if {$i >= 0} {
245 set headline [string trim [string range $comment 0 $i]]
246 } else {
247 set headline $comment
248 }
249 if {!$listed} {
250 # git-rev-list indents the comment by 4 spaces;
251 # if we got this via git-cat-file, add the indentation
252 set newcomment {}
253 foreach line [split $comment "\n"] {
254 append newcomment " "
255 append newcomment $line
256 append newcomment "\n"
257 }
258 set comment $newcomment
259 }
260 if {$comdate != {}} {
261 set cdate($id) $comdate
262 }
263 set commitinfo($id) [list $headline $auname $audate \
264 $comname $comdate $comment]
265}
266
267proc getcommit {id} {
268 global commitdata commitinfo nparents
269
270 if {[info exists commitdata($id)]} {
271 parsecommit $id $commitdata($id) 1
272 } else {
273 readcommit $id
274 if {![info exists commitinfo($id)]} {
275 set commitinfo($id) {"No commit information available"}
276 set nparents($id) 0
277 }
278 }
279 return 1
280}
281
282proc readrefs {} {
283 global tagids idtags headids idheads tagcontents
284 global otherrefids idotherrefs
285
286 foreach v {tagids idtags headids idheads otherrefids idotherrefs} {
287 catch {unset $v}
288 }
289 set refd [open [list | git-ls-remote [gitdir]] r]
290 while {0 <= [set n [gets $refd line]]} {
291 if {![regexp {^([0-9a-f]{40}) refs/([^^]*)$} $line \
292 match id path]} {
293 continue
294 }
295 if {![regexp {^(tags|heads)/(.*)$} $path match type name]} {
296 set type others
297 set name $path
298 }
299 if {$type == "tags"} {
300 set tagids($name) $id
301 lappend idtags($id) $name
302 set obj {}
303 set type {}
304 set tag {}
305 catch {
306 set commit [exec git-rev-parse "$id^0"]
307 if {"$commit" != "$id"} {
308 set tagids($name) $commit
309 lappend idtags($commit) $name
310 }
311 }
312 catch {
313 set tagcontents($name) [exec git-cat-file tag "$id"]
314 }
315 } elseif { $type == "heads" } {
316 set headids($name) $id
317 lappend idheads($id) $name
318 } else {
319 set otherrefids($name) $id
320 lappend idotherrefs($id) $name
321 }
322 }
323 close $refd
324}
325
326proc error_popup msg {
327 set w .error
328 toplevel $w
329 wm transient $w .
330 message $w.m -text $msg -justify center -aspect 400
331 pack $w.m -side top -fill x -padx 20 -pady 20
332 button $w.ok -text OK -command "destroy $w"
333 pack $w.ok -side bottom -fill x
334 bind $w <Visibility> "grab $w; focus $w"
335 bind $w <Key-Return> "destroy $w"
336 tkwait window $w
337}
338
339proc makewindow {rargs} {
340 global canv canv2 canv3 linespc charspc ctext cflist textfont
341 global findtype findtypemenu findloc findstring fstring geometry
342 global entries sha1entry sha1string sha1but
343 global maincursor textcursor curtextcursor
344 global rowctxmenu mergemax
345
346 menu .bar
347 .bar add cascade -label "File" -menu .bar.file
348 menu .bar.file
349 .bar.file add command -label "Update" -command [list updatecommits $rargs]
350 .bar.file add command -label "Reread references" -command rereadrefs
351 .bar.file add command -label "Quit" -command doquit
352 menu .bar.edit
353 .bar add cascade -label "Edit" -menu .bar.edit
354 .bar.edit add command -label "Preferences" -command doprefs
355 menu .bar.help
356 .bar add cascade -label "Help" -menu .bar.help
357 .bar.help add command -label "About gitk" -command about
358 . configure -menu .bar
359
360 if {![info exists geometry(canv1)]} {
361 set geometry(canv1) [expr {45 * $charspc}]
362 set geometry(canv2) [expr {30 * $charspc}]
363 set geometry(canv3) [expr {15 * $charspc}]
364 set geometry(canvh) [expr {25 * $linespc + 4}]
365 set geometry(ctextw) 80
366 set geometry(ctexth) 30
367 set geometry(cflistw) 30
368 }
369 panedwindow .ctop -orient vertical
370 if {[info exists geometry(width)]} {
371 .ctop conf -width $geometry(width) -height $geometry(height)
372 set texth [expr {$geometry(height) - $geometry(canvh) - 56}]
373 set geometry(ctexth) [expr {($texth - 8) /
374 [font metrics $textfont -linespace]}]
375 }
376 frame .ctop.top
377 frame .ctop.top.bar
378 pack .ctop.top.bar -side bottom -fill x
379 set cscroll .ctop.top.csb
380 scrollbar $cscroll -command {allcanvs yview} -highlightthickness 0
381 pack $cscroll -side right -fill y
382 panedwindow .ctop.top.clist -orient horizontal -sashpad 0 -handlesize 4
383 pack .ctop.top.clist -side top -fill both -expand 1
384 .ctop add .ctop.top
385 set canv .ctop.top.clist.canv
386 canvas $canv -height $geometry(canvh) -width $geometry(canv1) \
387 -bg white -bd 0 \
388 -yscrollincr $linespc -yscrollcommand "scrollcanv $cscroll"
389 .ctop.top.clist add $canv
390 set canv2 .ctop.top.clist.canv2
391 canvas $canv2 -height $geometry(canvh) -width $geometry(canv2) \
392 -bg white -bd 0 -yscrollincr $linespc
393 .ctop.top.clist add $canv2
394 set canv3 .ctop.top.clist.canv3
395 canvas $canv3 -height $geometry(canvh) -width $geometry(canv3) \
396 -bg white -bd 0 -yscrollincr $linespc
397 .ctop.top.clist add $canv3
398 bind .ctop.top.clist <Configure> {resizeclistpanes %W %w}
399
400 set sha1entry .ctop.top.bar.sha1
401 set entries $sha1entry
402 set sha1but .ctop.top.bar.sha1label
403 button $sha1but -text "SHA1 ID: " -state disabled -relief flat \
404 -command gotocommit -width 8
405 $sha1but conf -disabledforeground [$sha1but cget -foreground]
406 pack .ctop.top.bar.sha1label -side left
407 entry $sha1entry -width 40 -font $textfont -textvariable sha1string
408 trace add variable sha1string write sha1change
409 pack $sha1entry -side left -pady 2
410
411 image create bitmap bm-left -data {
412 #define left_width 16
413 #define left_height 16
414 static unsigned char left_bits[] = {
415 0x00, 0x00, 0xc0, 0x01, 0xe0, 0x00, 0x70, 0x00, 0x38, 0x00, 0x1c, 0x00,
416 0x0e, 0x00, 0xff, 0x7f, 0xff, 0x7f, 0xff, 0x7f, 0x0e, 0x00, 0x1c, 0x00,
417 0x38, 0x00, 0x70, 0x00, 0xe0, 0x00, 0xc0, 0x01};
418 }
419 image create bitmap bm-right -data {
420 #define right_width 16
421 #define right_height 16
422 static unsigned char right_bits[] = {
423 0x00, 0x00, 0xc0, 0x01, 0x80, 0x03, 0x00, 0x07, 0x00, 0x0e, 0x00, 0x1c,
424 0x00, 0x38, 0xff, 0x7f, 0xff, 0x7f, 0xff, 0x7f, 0x00, 0x38, 0x00, 0x1c,
425 0x00, 0x0e, 0x00, 0x07, 0x80, 0x03, 0xc0, 0x01};
426 }
427 button .ctop.top.bar.leftbut -image bm-left -command goback \
428 -state disabled -width 26
429 pack .ctop.top.bar.leftbut -side left -fill y
430 button .ctop.top.bar.rightbut -image bm-right -command goforw \
431 -state disabled -width 26
432 pack .ctop.top.bar.rightbut -side left -fill y
433
434 button .ctop.top.bar.findbut -text "Find" -command dofind
435 pack .ctop.top.bar.findbut -side left
436 set findstring {}
437 set fstring .ctop.top.bar.findstring
438 lappend entries $fstring
439 entry $fstring -width 30 -font $textfont -textvariable findstring
440 pack $fstring -side left -expand 1 -fill x
441 set findtype Exact
442 set findtypemenu [tk_optionMenu .ctop.top.bar.findtype \
443 findtype Exact IgnCase Regexp]
444 set findloc "All fields"
445 tk_optionMenu .ctop.top.bar.findloc findloc "All fields" Headline \
446 Comments Author Committer Files Pickaxe
447 pack .ctop.top.bar.findloc -side right
448 pack .ctop.top.bar.findtype -side right
449 # for making sure type==Exact whenever loc==Pickaxe
450 trace add variable findloc write findlocchange
451
452 panedwindow .ctop.cdet -orient horizontal
453 .ctop add .ctop.cdet
454 frame .ctop.cdet.left
455 set ctext .ctop.cdet.left.ctext
456 text $ctext -bg white -state disabled -font $textfont \
457 -width $geometry(ctextw) -height $geometry(ctexth) \
458 -yscrollcommand ".ctop.cdet.left.sb set" -wrap none
459 scrollbar .ctop.cdet.left.sb -command "$ctext yview"
460 pack .ctop.cdet.left.sb -side right -fill y
461 pack $ctext -side left -fill both -expand 1
462 .ctop.cdet add .ctop.cdet.left
463
464 $ctext tag conf filesep -font [concat $textfont bold] -back "#aaaaaa"
465 $ctext tag conf hunksep -fore blue
466 $ctext tag conf d0 -fore red
467 $ctext tag conf d1 -fore "#00a000"
468 $ctext tag conf m0 -fore red
469 $ctext tag conf m1 -fore blue
470 $ctext tag conf m2 -fore green
471 $ctext tag conf m3 -fore purple
472 $ctext tag conf m4 -fore brown
473 $ctext tag conf m5 -fore "#009090"
474 $ctext tag conf m6 -fore magenta
475 $ctext tag conf m7 -fore "#808000"
476 $ctext tag conf m8 -fore "#009000"
477 $ctext tag conf m9 -fore "#ff0080"
478 $ctext tag conf m10 -fore cyan
479 $ctext tag conf m11 -fore "#b07070"
480 $ctext tag conf m12 -fore "#70b0f0"
481 $ctext tag conf m13 -fore "#70f0b0"
482 $ctext tag conf m14 -fore "#f0b070"
483 $ctext tag conf m15 -fore "#ff70b0"
484 $ctext tag conf mmax -fore darkgrey
485 set mergemax 16
486 $ctext tag conf mresult -font [concat $textfont bold]
487 $ctext tag conf msep -font [concat $textfont bold]
488 $ctext tag conf found -back yellow
489
490 frame .ctop.cdet.right
491 set cflist .ctop.cdet.right.cfiles
492 listbox $cflist -bg white -selectmode extended -width $geometry(cflistw) \
493 -yscrollcommand ".ctop.cdet.right.sb set"
494 scrollbar .ctop.cdet.right.sb -command "$cflist yview"
495 pack .ctop.cdet.right.sb -side right -fill y
496 pack $cflist -side left -fill both -expand 1
497 .ctop.cdet add .ctop.cdet.right
498 bind .ctop.cdet <Configure> {resizecdetpanes %W %w}
499
500 pack .ctop -side top -fill both -expand 1
501
502 bindall <1> {selcanvline %W %x %y}
503 #bindall <B1-Motion> {selcanvline %W %x %y}
504 bindall <ButtonRelease-4> "allcanvs yview scroll -5 units"
505 bindall <ButtonRelease-5> "allcanvs yview scroll 5 units"
506 bindall <2> "allcanvs scan mark 0 %y"
507 bindall <B2-Motion> "allcanvs scan dragto 0 %y"
508 bind . <Key-Up> "selnextline -1"
509 bind . <Key-Down> "selnextline 1"
510 bind . <Key-Right> "goforw"
511 bind . <Key-Left> "goback"
512 bind . <Key-Prior> "allcanvs yview scroll -1 pages"
513 bind . <Key-Next> "allcanvs yview scroll 1 pages"
514 bindkey <Key-Delete> "$ctext yview scroll -1 pages"
515 bindkey <Key-BackSpace> "$ctext yview scroll -1 pages"
516 bindkey <Key-space> "$ctext yview scroll 1 pages"
517 bindkey p "selnextline -1"
518 bindkey n "selnextline 1"
519 bindkey z "goback"
520 bindkey x "goforw"
521 bindkey i "selnextline -1"
522 bindkey k "selnextline 1"
523 bindkey j "goback"
524 bindkey l "goforw"
525 bindkey b "$ctext yview scroll -1 pages"
526 bindkey d "$ctext yview scroll 18 units"
527 bindkey u "$ctext yview scroll -18 units"
528 bindkey / {findnext 1}
529 bindkey <Key-Return> {findnext 0}
530 bindkey ? findprev
531 bindkey f nextfile
532 bind . <Control-q> doquit
533 bind . <Control-f> dofind
534 bind . <Control-g> {findnext 0}
535 bind . <Control-r> findprev
536 bind . <Control-equal> {incrfont 1}
537 bind . <Control-KP_Add> {incrfont 1}
538 bind . <Control-minus> {incrfont -1}
539 bind . <Control-KP_Subtract> {incrfont -1}
540 bind $cflist <<ListboxSelect>> listboxsel
541 bind . <Destroy> {savestuff %W}
542 bind . <Button-1> "click %W"
543 bind $fstring <Key-Return> dofind
544 bind $sha1entry <Key-Return> gotocommit
545 bind $sha1entry <<PasteSelection>> clearsha1
546
547 set maincursor [. cget -cursor]
548 set textcursor [$ctext cget -cursor]
549 set curtextcursor $textcursor
550
551 set rowctxmenu .rowctxmenu
552 menu $rowctxmenu -tearoff 0
553 $rowctxmenu add command -label "Diff this -> selected" \
554 -command {diffvssel 0}
555 $rowctxmenu add command -label "Diff selected -> this" \
556 -command {diffvssel 1}
557 $rowctxmenu add command -label "Make patch" -command mkpatch
558 $rowctxmenu add command -label "Create tag" -command mktag
559 $rowctxmenu add command -label "Write commit to file" -command writecommit
560}
561
562proc scrollcanv {cscroll f0 f1} {
563 $cscroll set $f0 $f1
564 drawfrac $f0 $f1
565}
566
567# when we make a key binding for the toplevel, make sure
568# it doesn't get triggered when that key is pressed in the
569# find string entry widget.
570proc bindkey {ev script} {
571 global entries
572 bind . $ev $script
573 set escript [bind Entry $ev]
574 if {$escript == {}} {
575 set escript [bind Entry <Key>]
576 }
577 foreach e $entries {
578 bind $e $ev "$escript; break"
579 }
580}
581
582# set the focus back to the toplevel for any click outside
583# the entry widgets
584proc click {w} {
585 global entries
586 foreach e $entries {
587 if {$w == $e} return
588 }
589 focus .
590}
591
592proc savestuff {w} {
593 global canv canv2 canv3 ctext cflist mainfont textfont
594 global stuffsaved findmergefiles maxgraphpct
595 global maxwidth
596
597 if {$stuffsaved} return
598 if {![winfo viewable .]} return
599 catch {
600 set f [open "~/.gitk-new" w]
601 puts $f [list set mainfont $mainfont]
602 puts $f [list set textfont $textfont]
603 puts $f [list set findmergefiles $findmergefiles]
604 puts $f [list set maxgraphpct $maxgraphpct]
605 puts $f [list set maxwidth $maxwidth]
606 puts $f "set geometry(width) [winfo width .ctop]"
607 puts $f "set geometry(height) [winfo height .ctop]"
608 puts $f "set geometry(canv1) [expr {[winfo width $canv]-2}]"
609 puts $f "set geometry(canv2) [expr {[winfo width $canv2]-2}]"
610 puts $f "set geometry(canv3) [expr {[winfo width $canv3]-2}]"
611 puts $f "set geometry(canvh) [expr {[winfo height $canv]-2}]"
612 set wid [expr {([winfo width $ctext] - 8) \
613 / [font measure $textfont "0"]}]
614 puts $f "set geometry(ctextw) $wid"
615 set wid [expr {([winfo width $cflist] - 11) \
616 / [font measure [$cflist cget -font] "0"]}]
617 puts $f "set geometry(cflistw) $wid"
618 close $f
619 file rename -force "~/.gitk-new" "~/.gitk"
620 }
621 set stuffsaved 1
622}
623
624proc resizeclistpanes {win w} {
625 global oldwidth
626 if {[info exists oldwidth($win)]} {
627 set s0 [$win sash coord 0]
628 set s1 [$win sash coord 1]
629 if {$w < 60} {
630 set sash0 [expr {int($w/2 - 2)}]
631 set sash1 [expr {int($w*5/6 - 2)}]
632 } else {
633 set factor [expr {1.0 * $w / $oldwidth($win)}]
634 set sash0 [expr {int($factor * [lindex $s0 0])}]
635 set sash1 [expr {int($factor * [lindex $s1 0])}]
636 if {$sash0 < 30} {
637 set sash0 30
638 }
639 if {$sash1 < $sash0 + 20} {
640 set sash1 [expr {$sash0 + 20}]
641 }
642 if {$sash1 > $w - 10} {
643 set sash1 [expr {$w - 10}]
644 if {$sash0 > $sash1 - 20} {
645 set sash0 [expr {$sash1 - 20}]
646 }
647 }
648 }
649 $win sash place 0 $sash0 [lindex $s0 1]
650 $win sash place 1 $sash1 [lindex $s1 1]
651 }
652 set oldwidth($win) $w
653}
654
655proc resizecdetpanes {win w} {
656 global oldwidth
657 if {[info exists oldwidth($win)]} {
658 set s0 [$win sash coord 0]
659 if {$w < 60} {
660 set sash0 [expr {int($w*3/4 - 2)}]
661 } else {
662 set factor [expr {1.0 * $w / $oldwidth($win)}]
663 set sash0 [expr {int($factor * [lindex $s0 0])}]
664 if {$sash0 < 45} {
665 set sash0 45
666 }
667 if {$sash0 > $w - 15} {
668 set sash0 [expr {$w - 15}]
669 }
670 }
671 $win sash place 0 $sash0 [lindex $s0 1]
672 }
673 set oldwidth($win) $w
674}
675
676proc allcanvs args {
677 global canv canv2 canv3
678 eval $canv $args
679 eval $canv2 $args
680 eval $canv3 $args
681}
682
683proc bindall {event action} {
684 global canv canv2 canv3
685 bind $canv $event $action
686 bind $canv2 $event $action
687 bind $canv3 $event $action
688}
689
690proc about {} {
691 set w .about
692 if {[winfo exists $w]} {
693 raise $w
694 return
695 }
696 toplevel $w
697 wm title $w "About gitk"
698 message $w.m -text {
699Gitk - a commit viewer for git
700
701Copyright © 2005-2006 Paul Mackerras
702
703Use and redistribute under the terms of the GNU General Public License} \
704 -justify center -aspect 400
705 pack $w.m -side top -fill x -padx 20 -pady 20
706 button $w.ok -text Close -command "destroy $w"
707 pack $w.ok -side bottom
708}
709
710proc shortids {ids} {
711 set res {}
712 foreach id $ids {
713 if {[llength $id] > 1} {
714 lappend res [shortids $id]
715 } elseif {[regexp {^[0-9a-f]{40}$} $id]} {
716 lappend res [string range $id 0 7]
717 } else {
718 lappend res $id
719 }
720 }
721 return $res
722}
723
724proc incrange {l x o} {
725 set n [llength $l]
726 while {$x < $n} {
727 set e [lindex $l $x]
728 if {$e ne {}} {
729 lset l $x [expr {$e + $o}]
730 }
731 incr x
732 }
733 return $l
734}
735
736proc ntimes {n o} {
737 set ret {}
738 for {} {$n > 0} {incr n -1} {
739 lappend ret $o
740 }
741 return $ret
742}
743
744proc usedinrange {id l1 l2} {
745 global children commitrow
746
747 if {[info exists commitrow($id)]} {
748 set r $commitrow($id)
749 if {$l1 <= $r && $r <= $l2} {
750 return [expr {$r - $l1 + 1}]
751 }
752 }
753 foreach c $children($id) {
754 if {[info exists commitrow($c)]} {
755 set r $commitrow($c)
756 if {$l1 <= $r && $r <= $l2} {
757 return [expr {$r - $l1 + 1}]
758 }
759 }
760 }
761 return 0
762}
763
764proc sanity {row {full 0}} {
765 global rowidlist rowoffsets
766
767 set col -1
768 set ids [lindex $rowidlist $row]
769 foreach id $ids {
770 incr col
771 if {$id eq {}} continue
772 if {$col < [llength $ids] - 1 &&
773 [lsearch -exact -start [expr {$col+1}] $ids $id] >= 0} {
774 puts "oops: [shortids $id] repeated in row $row col $col: {[shortids [lindex $rowidlist $row]]}"
775 }
776 set o [lindex $rowoffsets $row $col]
777 set y $row
778 set x $col
779 while {$o ne {}} {
780 incr y -1
781 incr x $o
782 if {[lindex $rowidlist $y $x] != $id} {
783 puts "oops: rowoffsets wrong at row [expr {$y+1}] col [expr {$x-$o}]"
784 puts " id=[shortids $id] check started at row $row"
785 for {set i $row} {$i >= $y} {incr i -1} {
786 puts " row $i ids={[shortids [lindex $rowidlist $i]]} offs={[lindex $rowoffsets $i]}"
787 }
788 break
789 }
790 if {!$full} break
791 set o [lindex $rowoffsets $y $x]
792 }
793 }
794}
795
796proc makeuparrow {oid x y z} {
797 global rowidlist rowoffsets uparrowlen idrowranges
798
799 for {set i 1} {$i < $uparrowlen && $y > 1} {incr i} {
800 incr y -1
801 incr x $z
802 set off0 [lindex $rowoffsets $y]
803 for {set x0 $x} {1} {incr x0} {
804 if {$x0 >= [llength $off0]} {
805 set x0 [llength [lindex $rowoffsets [expr {$y-1}]]]
806 break
807 }
808 set z [lindex $off0 $x0]
809 if {$z ne {}} {
810 incr x0 $z
811 break
812 }
813 }
814 set z [expr {$x0 - $x}]
815 lset rowidlist $y [linsert [lindex $rowidlist $y] $x $oid]
816 lset rowoffsets $y [linsert [lindex $rowoffsets $y] $x $z]
817 }
818 set tmp [lreplace [lindex $rowoffsets $y] $x $x {}]
819 lset rowoffsets $y [incrange $tmp [expr {$x+1}] -1]
820 lappend idrowranges($oid) $y
821}
822
823proc initlayout {} {
824 global rowidlist rowoffsets displayorder
825 global rowlaidout rowoptim
826 global idinlist rowchk
827 global commitidx numcommits
828 global nextcolor
829
830 set commitidx 0
831 set numcommits 0
832 set displayorder {}
833 set nextcolor 0
834 set rowidlist {{}}
835 set rowoffsets {{}}
836 catch {unset idinlist}
837 catch {unset rowchk}
838 set rowlaidout 0
839 set rowoptim 0
840}
841
842proc visiblerows {} {
843 global canv numcommits linespc
844
845 set ymax [lindex [$canv cget -scrollregion] 3]
846 if {$ymax eq {} || $ymax == 0} return
847 set f [$canv yview]
848 set y0 [expr {int([lindex $f 0] * $ymax)}]
849 set r0 [expr {int(($y0 - 3) / $linespc) - 1}]
850 if {$r0 < 0} {
851 set r0 0
852 }
853 set y1 [expr {int([lindex $f 1] * $ymax)}]
854 set r1 [expr {int(($y1 - 3) / $linespc) + 1}]
855 if {$r1 >= $numcommits} {
856 set r1 [expr {$numcommits - 1}]
857 }
858 return [list $r0 $r1]
859}
860
861proc layoutmore {} {
862 global rowlaidout rowoptim commitidx numcommits optim_delay
863 global uparrowlen
864
865 set row $rowlaidout
866 set rowlaidout [layoutrows $row $commitidx 0]
867 set orow [expr {$rowlaidout - $uparrowlen - 1}]
868 if {$orow > $rowoptim} {
869 checkcrossings $rowoptim $orow
870 optimize_rows $rowoptim 0 $orow
871 set rowoptim $orow
872 }
873 set canshow [expr {$rowoptim - $optim_delay}]
874 if {$canshow > $numcommits} {
875 showstuff $canshow
876 }
877}
878
879proc showstuff {canshow} {
880 global numcommits
881 global canvy0 linespc
882 global linesegends idrowranges idrangedrawn
883
884 if {$numcommits == 0} {
885 global phase
886 set phase "incrdraw"
887 allcanvs delete all
888 }
889 set row $numcommits
890 set numcommits $canshow
891 allcanvs conf -scrollregion \
892 [list 0 0 0 [expr {$canvy0 + ($numcommits - 0.5) * $linespc + 2}]]
893 set rows [visiblerows]
894 set r0 [lindex $rows 0]
895 set r1 [lindex $rows 1]
896 for {set r $row} {$r < $canshow} {incr r} {
897 if {[info exists linesegends($r)]} {
898 foreach id $linesegends($r) {
899 set i -1
900 foreach {s e} $idrowranges($id) {
901 incr i
902 if {$e ne {} && $e < $numcommits && $s <= $r1 && $e >= $r0
903 && ![info exists idrangedrawn($id,$i)]} {
904 drawlineseg $id $i
905 set idrangedrawn($id,$i) 1
906 }
907 }
908 }
909 }
910 }
911 if {$canshow > $r1} {
912 set canshow $r1
913 }
914 while {$row < $canshow} {
915 drawcmitrow $row
916 incr row
917 }
918}
919
920proc layoutrows {row endrow last} {
921 global rowidlist rowoffsets displayorder
922 global uparrowlen downarrowlen maxwidth mingaplen
923 global nchildren parents nparents
924 global idrowranges linesegends
925 global commitidx
926 global idinlist rowchk
927
928 set idlist [lindex $rowidlist $row]
929 set offs [lindex $rowoffsets $row]
930 while {$row < $endrow} {
931 set id [lindex $displayorder $row]
932 set oldolds {}
933 set newolds {}
934 foreach p $parents($id) {
935 if {![info exists idinlist($p)]} {
936 lappend newolds $p
937 } elseif {!$idinlist($p)} {
938 lappend oldolds $p
939 }
940 }
941 set nev [expr {[llength $idlist] + [llength $newolds]
942 + [llength $oldolds] - $maxwidth + 1}]
943 if {$nev > 0} {
944 if {!$last && $row + $uparrowlen + $mingaplen >= $commitidx} break
945 for {set x [llength $idlist]} {[incr x -1] >= 0} {} {
946 set i [lindex $idlist $x]
947 if {![info exists rowchk($i)] || $row >= $rowchk($i)} {
948 set r [usedinrange $i [expr {$row - $downarrowlen}] \
949 [expr {$row + $uparrowlen + $mingaplen}]]
950 if {$r == 0} {
951 set idlist [lreplace $idlist $x $x]
952 set offs [lreplace $offs $x $x]
953 set offs [incrange $offs $x 1]
954 set idinlist($i) 0
955 set rm1 [expr {$row - 1}]
956 lappend linesegends($rm1) $i
957 lappend idrowranges($i) $rm1
958 if {[incr nev -1] <= 0} break
959 continue
960 }
961 set rowchk($id) [expr {$row + $r}]
962 }
963 }
964 lset rowidlist $row $idlist
965 lset rowoffsets $row $offs
966 }
967 set col [lsearch -exact $idlist $id]
968 if {$col < 0} {
969 set col [llength $idlist]
970 lappend idlist $id
971 lset rowidlist $row $idlist
972 set z {}
973 if {$nchildren($id) > 0} {
974 set z [expr {[llength [lindex $rowidlist [expr {$row-1}]]] - $col}]
975 unset idinlist($id)
976 }
977 lappend offs $z
978 lset rowoffsets $row $offs
979 if {$z ne {}} {
980 makeuparrow $id $col $row $z
981 }
982 } else {
983 unset idinlist($id)
984 }
985 if {[info exists idrowranges($id)]} {
986 lappend idrowranges($id) $row
987 }
988 incr row
989 set offs [ntimes [llength $idlist] 0]
990 set l [llength $newolds]
991 set idlist [eval lreplace \$idlist $col $col $newolds]
992 set o 0
993 if {$l != 1} {
994 set offs [lrange $offs 0 [expr {$col - 1}]]
995 foreach x $newolds {
996 lappend offs {}
997 incr o -1
998 }
999 incr o
1000 set tmp [expr {[llength $idlist] - [llength $offs]}]
1001 if {$tmp > 0} {
1002 set offs [concat $offs [ntimes $tmp $o]]
1003 }
1004 } else {
1005 lset offs $col {}
1006 }
1007 foreach i $newolds {
1008 set idinlist($i) 1
1009 set idrowranges($i) $row
1010 }
1011 incr col $l
1012 foreach oid $oldolds {
1013 set idinlist($oid) 1
1014 set idlist [linsert $idlist $col $oid]
1015 set offs [linsert $offs $col $o]
1016 makeuparrow $oid $col $row $o
1017 incr col
1018 }
1019 lappend rowidlist $idlist
1020 lappend rowoffsets $offs
1021 }
1022 return $row
1023}
1024
1025proc addextraid {id row} {
1026 global displayorder commitrow commitinfo nparents
1027 global commitidx
1028
1029 incr commitidx
1030 lappend displayorder $id
1031 set commitrow($id) $row
1032 readcommit $id
1033 if {![info exists commitinfo($id)]} {
1034 set commitinfo($id) {"No commit information available"}
1035 set nparents($id) 0
1036 }
1037}
1038
1039proc layouttail {} {
1040 global rowidlist rowoffsets idinlist commitidx
1041 global idrowranges
1042
1043 set row $commitidx
1044 set idlist [lindex $rowidlist $row]
1045 while {$idlist ne {}} {
1046 set col [expr {[llength $idlist] - 1}]
1047 set id [lindex $idlist $col]
1048 addextraid $id $row
1049 unset idinlist($id)
1050 lappend idrowranges($id) $row
1051 incr row
1052 set offs [ntimes $col 0]
1053 set idlist [lreplace $idlist $col $col]
1054 lappend rowidlist $idlist
1055 lappend rowoffsets $offs
1056 }
1057
1058 foreach id [array names idinlist] {
1059 addextraid $id $row
1060 lset rowidlist $row [list $id]
1061 lset rowoffsets $row 0
1062 makeuparrow $id 0 $row 0
1063 lappend idrowranges($id) $row
1064 incr row
1065 lappend rowidlist {}
1066 lappend rowoffsets {}
1067 }
1068}
1069
1070proc insert_pad {row col npad} {
1071 global rowidlist rowoffsets
1072
1073 set pad [ntimes $npad {}]
1074 lset rowidlist $row [eval linsert [list [lindex $rowidlist $row]] $col $pad]
1075 set tmp [eval linsert [list [lindex $rowoffsets $row]] $col $pad]
1076 lset rowoffsets $row [incrange $tmp [expr {$col + $npad}] [expr {-$npad}]]
1077}
1078
1079proc optimize_rows {row col endrow} {
1080 global rowidlist rowoffsets idrowranges linesegends displayorder
1081
1082 for {} {$row < $endrow} {incr row} {
1083 set idlist [lindex $rowidlist $row]
1084 set offs [lindex $rowoffsets $row]
1085 set haspad 0
1086 set downarrowcols {}
1087 if {[info exists linesegends($row)]} {
1088 set downarrowcols $linesegends($row)
1089 if {$col > 0} {
1090 while {$downarrowcols ne {}} {
1091 set i [lsearch -exact $idlist [lindex $downarrowcols 0]]
1092 if {$i < 0 || $i >= $col} break
1093 set downarrowcols [lrange $downarrowcols 1 end]
1094 }
1095 }
1096 }
1097 for {} {$col < [llength $offs]} {incr col} {
1098 if {[lindex $idlist $col] eq {}} {
1099 set haspad 1
1100 continue
1101 }
1102 set z [lindex $offs $col]
1103 if {$z eq {}} continue
1104 set isarrow 0
1105 set x0 [expr {$col + $z}]
1106 set y0 [expr {$row - 1}]
1107 set z0 [lindex $rowoffsets $y0 $x0]
1108 if {$z0 eq {}} {
1109 set id [lindex $idlist $col]
1110 if {[info exists idrowranges($id)] &&
1111 $y0 > [lindex $idrowranges($id) 0]} {
1112 set isarrow 1
1113 }
1114 } elseif {$downarrowcols ne {} &&
1115 [lindex $idlist $col] eq [lindex $downarrowcols 0]} {
1116 set downarrowcols [lrange $downarrowcols 1 end]
1117 set isarrow 1
1118 }
1119 if {$z < -1 || ($z < 0 && $isarrow)} {
1120 set npad [expr {-1 - $z + $isarrow}]
1121 set offs [incrange $offs $col $npad]
1122 insert_pad $y0 $x0 $npad
1123 if {$y0 > 0} {
1124 optimize_rows $y0 $x0 $row
1125 }
1126 set z [lindex $offs $col]
1127 set x0 [expr {$col + $z}]
1128 set z0 [lindex $rowoffsets $y0 $x0]
1129 } elseif {$z > 1 || ($z > 0 && $isarrow)} {
1130 set npad [expr {$z - 1 + $isarrow}]
1131 set y1 [expr {$row + 1}]
1132 set offs2 [lindex $rowoffsets $y1]
1133 set x1 -1
1134 foreach z $offs2 {
1135 incr x1
1136 if {$z eq {} || $x1 + $z < $col} continue
1137 if {$x1 + $z > $col} {
1138 incr npad
1139 }
1140 lset rowoffsets $y1 [incrange $offs2 $x1 $npad]
1141 break
1142 }
1143 set pad [ntimes $npad {}]
1144 set idlist [eval linsert \$idlist $col $pad]
1145 set tmp [eval linsert \$offs $col $pad]
1146 incr col $npad
1147 set offs [incrange $tmp $col [expr {-$npad}]]
1148 set z [lindex $offs $col]
1149 set haspad 1
1150 }
1151 if {$z0 eq {} && !$isarrow} {
1152 # this line links to its first child on row $row-2
1153 set rm2 [expr {$row - 2}]
1154 set id [lindex $displayorder $rm2]
1155 set xc [lsearch -exact [lindex $rowidlist $rm2] $id]
1156 if {$xc >= 0} {
1157 set z0 [expr {$xc - $x0}]
1158 }
1159 }
1160 if {$z0 ne {} && $z < 0 && $z0 > 0} {
1161 insert_pad $y0 $x0 1
1162 set offs [incrange $offs $col 1]
1163 optimize_rows $y0 [expr {$x0 + 1}] $row
1164 }
1165 }
1166 if {!$haspad} {
1167 set o {}
1168 for {set col [llength $idlist]} {[incr col -1] >= 0} {} {
1169 set o [lindex $offs $col]
1170 if {$o eq {}} {
1171 # check if this is the link to the first child
1172 set id [lindex $idlist $col]
1173 if {[info exists idrowranges($id)] &&
1174 $row == [lindex $idrowranges($id) 0]} {
1175 # it is, work out offset to child
1176 set y0 [expr {$row - 1}]
1177 set id [lindex $displayorder $y0]
1178 set x0 [lsearch -exact [lindex $rowidlist $y0] $id]
1179 if {$x0 >= 0} {
1180 set o [expr {$x0 - $col}]
1181 }
1182 }
1183 }
1184 if {$o eq {} || $o <= 0} break
1185 }
1186 if {$o ne {} && [incr col] < [llength $idlist]} {
1187 set y1 [expr {$row + 1}]
1188 set offs2 [lindex $rowoffsets $y1]
1189 set x1 -1
1190 foreach z $offs2 {
1191 incr x1
1192 if {$z eq {} || $x1 + $z < $col} continue
1193 lset rowoffsets $y1 [incrange $offs2 $x1 1]
1194 break
1195 }
1196 set idlist [linsert $idlist $col {}]
1197 set tmp [linsert $offs $col {}]
1198 incr col
1199 set offs [incrange $tmp $col -1]
1200 }
1201 }
1202 lset rowidlist $row $idlist
1203 lset rowoffsets $row $offs
1204 set col 0
1205 }
1206}
1207
1208proc xc {row col} {
1209 global canvx0 linespc
1210 return [expr {$canvx0 + $col * $linespc}]
1211}
1212
1213proc yc {row} {
1214 global canvy0 linespc
1215 return [expr {$canvy0 + $row * $linespc}]
1216}
1217
1218proc linewidth {id} {
1219 global thickerline lthickness
1220
1221 set wid $lthickness
1222 if {[info exists thickerline] && $id eq $thickerline} {
1223 set wid [expr {2 * $lthickness}]
1224 }
1225 return $wid
1226}
1227
1228proc drawlineseg {id i} {
1229 global rowoffsets rowidlist idrowranges
1230 global displayorder
1231 global canv colormap
1232
1233 set startrow [lindex $idrowranges($id) [expr {2 * $i}]]
1234 set row [lindex $idrowranges($id) [expr {2 * $i + 1}]]
1235 if {$startrow == $row} return
1236 assigncolor $id
1237 set coords {}
1238 set col [lsearch -exact [lindex $rowidlist $row] $id]
1239 if {$col < 0} {
1240 puts "oops: drawline: id $id not on row $row"
1241 return
1242 }
1243 set lasto {}
1244 set ns 0
1245 while {1} {
1246 set o [lindex $rowoffsets $row $col]
1247 if {$o eq {}} break
1248 if {$o ne $lasto} {
1249 # changing direction
1250 set x [xc $row $col]
1251 set y [yc $row]
1252 lappend coords $x $y
1253 set lasto $o
1254 }
1255 incr col $o
1256 incr row -1
1257 }
1258 set x [xc $row $col]
1259 set y [yc $row]
1260 lappend coords $x $y
1261 if {$i == 0} {
1262 # draw the link to the first child as part of this line
1263 incr row -1
1264 set child [lindex $displayorder $row]
1265 set ccol [lsearch -exact [lindex $rowidlist $row] $child]
1266 if {$ccol >= 0} {
1267 set x [xc $row $ccol]
1268 set y [yc $row]
1269 if {$ccol < $col - 1} {
1270 lappend coords [xc $row [expr {$col - 1}]] [yc $row]
1271 } elseif {$ccol > $col + 1} {
1272 lappend coords [xc $row [expr {$col + 1}]] [yc $row]
1273 }
1274 lappend coords $x $y
1275 }
1276 }
1277 if {[llength $coords] < 4} return
1278 set last [expr {[llength $idrowranges($id)] / 2 - 1}]
1279 set arrow [expr {2 * ($i > 0) + ($i < $last)}]
1280 set arrow [lindex {none first last both} $arrow]
1281 set t [$canv create line $coords -width [linewidth $id] \
1282 -fill $colormap($id) -tags lines.$id -arrow $arrow]
1283 $canv lower $t
1284 bindline $t $id
1285}
1286
1287proc drawparentlinks {id row col olds} {
1288 global rowidlist canv colormap idrowranges
1289
1290 set row2 [expr {$row + 1}]
1291 set x [xc $row $col]
1292 set y [yc $row]
1293 set y2 [yc $row2]
1294 set ids [lindex $rowidlist $row2]
1295 # rmx = right-most X coord used
1296 set rmx 0
1297 foreach p $olds {
1298 if {[info exists idrowranges($p)] &&
1299 $row2 == [lindex $idrowranges($p) 0] &&
1300 $row2 < [lindex $idrowranges($p) 1]} {
1301 # drawlineseg will do this one for us
1302 continue
1303 }
1304 set i [lsearch -exact $ids $p]
1305 if {$i < 0} {
1306 puts "oops, parent $p of $id not in list"
1307 continue
1308 }
1309 assigncolor $p
1310 # should handle duplicated parents here...
1311 set coords [list $x $y]
1312 if {$i < $col - 1} {
1313 lappend coords [xc $row [expr {$i + 1}]] $y
1314 } elseif {$i > $col + 1} {
1315 lappend coords [xc $row [expr {$i - 1}]] $y
1316 }
1317 set x2 [xc $row2 $i]
1318 if {$x2 > $rmx} {
1319 set rmx $x2
1320 }
1321 lappend coords $x2 $y2
1322 set t [$canv create line $coords -width [linewidth $p] \
1323 -fill $colormap($p) -tags lines.$p]
1324 $canv lower $t
1325 bindline $t $p
1326 }
1327 return $rmx
1328}
1329
1330proc drawlines {id} {
1331 global colormap canv
1332 global idrowranges idrangedrawn
1333 global children iddrawn commitrow rowidlist
1334
1335 $canv delete lines.$id
1336 set nr [expr {[llength $idrowranges($id)] / 2}]
1337 for {set i 0} {$i < $nr} {incr i} {
1338 if {[info exists idrangedrawn($id,$i)]} {
1339 drawlineseg $id $i
1340 }
1341 }
1342 if {[info exists children($id)]} {
1343 foreach child $children($id) {
1344 if {[info exists iddrawn($child)]} {
1345 set row $commitrow($child)
1346 set col [lsearch -exact [lindex $rowidlist $row] $child]
1347 if {$col >= 0} {
1348 drawparentlinks $child $row $col [list $id]
1349 }
1350 }
1351 }
1352 }
1353}
1354
1355proc drawcmittext {id row col rmx} {
1356 global linespc canv canv2 canv3 canvy0
1357 global commitlisted commitinfo rowidlist
1358 global rowtextx idpos idtags idheads idotherrefs
1359 global linehtag linentag linedtag
1360 global mainfont namefont
1361
1362 set ofill [expr {[info exists commitlisted($id)]? "blue": "white"}]
1363 set x [xc $row $col]
1364 set y [yc $row]
1365 set orad [expr {$linespc / 3}]
1366 set t [$canv create oval [expr {$x - $orad}] [expr {$y - $orad}] \
1367 [expr {$x + $orad - 1}] [expr {$y + $orad - 1}] \
1368 -fill $ofill -outline black -width 1]
1369 $canv raise $t
1370 $canv bind $t <1> {selcanvline {} %x %y}
1371 set xt [xc $row [llength [lindex $rowidlist $row]]]
1372 if {$xt < $rmx} {
1373 set xt $rmx
1374 }
1375 set rowtextx($row) $xt
1376 set idpos($id) [list $x $xt $y]
1377 if {[info exists idtags($id)] || [info exists idheads($id)]
1378 || [info exists idotherrefs($id)]} {
1379 set xt [drawtags $id $x $xt $y]
1380 }
1381 set headline [lindex $commitinfo($id) 0]
1382 set name [lindex $commitinfo($id) 1]
1383 set date [lindex $commitinfo($id) 2]
1384 set date [formatdate $date]
1385 set linehtag($row) [$canv create text $xt $y -anchor w \
1386 -text $headline -font $mainfont ]
1387 $canv bind $linehtag($row) <Button-3> "rowmenu %X %Y $id"
1388 set linentag($row) [$canv2 create text 3 $y -anchor w \
1389 -text $name -font $namefont]
1390 set linedtag($row) [$canv3 create text 3 $y -anchor w \
1391 -text $date -font $mainfont]
1392}
1393
1394proc drawcmitrow {row} {
1395 global displayorder rowidlist
1396 global idrowranges idrangedrawn iddrawn
1397 global commitinfo commitlisted parents numcommits
1398
1399 if {$row >= $numcommits} return
1400 foreach id [lindex $rowidlist $row] {
1401 if {![info exists idrowranges($id)]} continue
1402 set i -1
1403 foreach {s e} $idrowranges($id) {
1404 incr i
1405 if {$row < $s} continue
1406 if {$e eq {}} break
1407 if {$row <= $e} {
1408 if {$e < $numcommits && ![info exists idrangedrawn($id,$i)]} {
1409 drawlineseg $id $i
1410 set idrangedrawn($id,$i) 1
1411 }
1412 break
1413 }
1414 }
1415 }
1416
1417 set id [lindex $displayorder $row]
1418 if {[info exists iddrawn($id)]} return
1419 set col [lsearch -exact [lindex $rowidlist $row] $id]
1420 if {$col < 0} {
1421 puts "oops, row $row id $id not in list"
1422 return
1423 }
1424 if {![info exists commitinfo($id)]} {
1425 getcommit $id
1426 }
1427 assigncolor $id
1428 if {[info exists commitlisted($id)] && [info exists parents($id)]
1429 && $parents($id) ne {}} {
1430 set rmx [drawparentlinks $id $row $col $parents($id)]
1431 } else {
1432 set rmx 0
1433 }
1434 drawcmittext $id $row $col $rmx
1435 set iddrawn($id) 1
1436}
1437
1438proc drawfrac {f0 f1} {
1439 global numcommits canv
1440 global linespc
1441
1442 set ymax [lindex [$canv cget -scrollregion] 3]
1443 if {$ymax eq {} || $ymax == 0} return
1444 set y0 [expr {int($f0 * $ymax)}]
1445 set row [expr {int(($y0 - 3) / $linespc) - 1}]
1446 if {$row < 0} {
1447 set row 0
1448 }
1449 set y1 [expr {int($f1 * $ymax)}]
1450 set endrow [expr {int(($y1 - 3) / $linespc) + 1}]
1451 if {$endrow >= $numcommits} {
1452 set endrow [expr {$numcommits - 1}]
1453 }
1454 for {} {$row <= $endrow} {incr row} {
1455 drawcmitrow $row
1456 }
1457}
1458
1459proc drawvisible {} {
1460 global canv
1461 eval drawfrac [$canv yview]
1462}
1463
1464proc clear_display {} {
1465 global iddrawn idrangedrawn
1466
1467 allcanvs delete all
1468 catch {unset iddrawn}
1469 catch {unset idrangedrawn}
1470}
1471
1472proc assigncolor {id} {
1473 global colormap colors nextcolor
1474 global parents nparents children nchildren
1475 global cornercrossings crossings
1476
1477 if {[info exists colormap($id)]} return
1478 set ncolors [llength $colors]
1479 if {$nchildren($id) == 1} {
1480 set child [lindex $children($id) 0]
1481 if {[info exists colormap($child)]
1482 && $nparents($child) == 1} {
1483 set colormap($id) $colormap($child)
1484 return
1485 }
1486 }
1487 set badcolors {}
1488 if {[info exists cornercrossings($id)]} {
1489 foreach x $cornercrossings($id) {
1490 if {[info exists colormap($x)]
1491 && [lsearch -exact $badcolors $colormap($x)] < 0} {
1492 lappend badcolors $colormap($x)
1493 }
1494 }
1495 if {[llength $badcolors] >= $ncolors} {
1496 set badcolors {}
1497 }
1498 }
1499 set origbad $badcolors
1500 if {[llength $badcolors] < $ncolors - 1} {
1501 if {[info exists crossings($id)]} {
1502 foreach x $crossings($id) {
1503 if {[info exists colormap($x)]
1504 && [lsearch -exact $badcolors $colormap($x)] < 0} {
1505 lappend badcolors $colormap($x)
1506 }
1507 }
1508 if {[llength $badcolors] >= $ncolors} {
1509 set badcolors $origbad
1510 }
1511 }
1512 set origbad $badcolors
1513 }
1514 if {[llength $badcolors] < $ncolors - 1} {
1515 foreach child $children($id) {
1516 if {[info exists colormap($child)]
1517 && [lsearch -exact $badcolors $colormap($child)] < 0} {
1518 lappend badcolors $colormap($child)
1519 }
1520 if {[info exists parents($child)]} {
1521 foreach p $parents($child) {
1522 if {[info exists colormap($p)]
1523 && [lsearch -exact $badcolors $colormap($p)] < 0} {
1524 lappend badcolors $colormap($p)
1525 }
1526 }
1527 }
1528 }
1529 if {[llength $badcolors] >= $ncolors} {
1530 set badcolors $origbad
1531 }
1532 }
1533 for {set i 0} {$i <= $ncolors} {incr i} {
1534 set c [lindex $colors $nextcolor]
1535 if {[incr nextcolor] >= $ncolors} {
1536 set nextcolor 0
1537 }
1538 if {[lsearch -exact $badcolors $c]} break
1539 }
1540 set colormap($id) $c
1541}
1542
1543proc bindline {t id} {
1544 global canv
1545
1546 $canv bind $t <Enter> "lineenter %x %y $id"
1547 $canv bind $t <Motion> "linemotion %x %y $id"
1548 $canv bind $t <Leave> "lineleave $id"
1549 $canv bind $t <Button-1> "lineclick %x %y $id 1"
1550}
1551
1552proc drawtags {id x xt y1} {
1553 global idtags idheads idotherrefs
1554 global linespc lthickness
1555 global canv mainfont commitrow rowtextx
1556
1557 set marks {}
1558 set ntags 0
1559 set nheads 0
1560 if {[info exists idtags($id)]} {
1561 set marks $idtags($id)
1562 set ntags [llength $marks]
1563 }
1564 if {[info exists idheads($id)]} {
1565 set marks [concat $marks $idheads($id)]
1566 set nheads [llength $idheads($id)]
1567 }
1568 if {[info exists idotherrefs($id)]} {
1569 set marks [concat $marks $idotherrefs($id)]
1570 }
1571 if {$marks eq {}} {
1572 return $xt
1573 }
1574
1575 set delta [expr {int(0.5 * ($linespc - $lthickness))}]
1576 set yt [expr {$y1 - 0.5 * $linespc}]
1577 set yb [expr {$yt + $linespc - 1}]
1578 set xvals {}
1579 set wvals {}
1580 foreach tag $marks {
1581 set wid [font measure $mainfont $tag]
1582 lappend xvals $xt
1583 lappend wvals $wid
1584 set xt [expr {$xt + $delta + $wid + $lthickness + $linespc}]
1585 }
1586 set t [$canv create line $x $y1 [lindex $xvals end] $y1 \
1587 -width $lthickness -fill black -tags tag.$id]
1588 $canv lower $t
1589 foreach tag $marks x $xvals wid $wvals {
1590 set xl [expr {$x + $delta}]
1591 set xr [expr {$x + $delta + $wid + $lthickness}]
1592 if {[incr ntags -1] >= 0} {
1593 # draw a tag
1594 set t [$canv create polygon $x [expr {$yt + $delta}] $xl $yt \
1595 $xr $yt $xr $yb $xl $yb $x [expr {$yb - $delta}] \
1596 -width 1 -outline black -fill yellow -tags tag.$id]
1597 $canv bind $t <1> [list showtag $tag 1]
1598 set rowtextx($commitrow($id)) [expr {$xr + $linespc}]
1599 } else {
1600 # draw a head or other ref
1601 if {[incr nheads -1] >= 0} {
1602 set col green
1603 } else {
1604 set col "#ddddff"
1605 }
1606 set xl [expr {$xl - $delta/2}]
1607 $canv create polygon $x $yt $xr $yt $xr $yb $x $yb \
1608 -width 1 -outline black -fill $col -tags tag.$id
1609 }
1610 set t [$canv create text $xl $y1 -anchor w -text $tag \
1611 -font $mainfont -tags tag.$id]
1612 if {$ntags >= 0} {
1613 $canv bind $t <1> [list showtag $tag 1]
1614 }
1615 }
1616 return $xt
1617}
1618
1619proc checkcrossings {row endrow} {
1620 global displayorder parents rowidlist
1621
1622 for {} {$row < $endrow} {incr row} {
1623 set id [lindex $displayorder $row]
1624 set i [lsearch -exact [lindex $rowidlist $row] $id]
1625 if {$i < 0} continue
1626 set idlist [lindex $rowidlist [expr {$row+1}]]
1627 foreach p $parents($id) {
1628 set j [lsearch -exact $idlist $p]
1629 if {$j > 0} {
1630 if {$j < $i - 1} {
1631 notecrossings $row $p $j $i [expr {$j+1}]
1632 } elseif {$j > $i + 1} {
1633 notecrossings $row $p $i $j [expr {$j-1}]
1634 }
1635 }
1636 }
1637 }
1638}
1639
1640proc notecrossings {row id lo hi corner} {
1641 global rowidlist crossings cornercrossings
1642
1643 for {set i $lo} {[incr i] < $hi} {} {
1644 set p [lindex [lindex $rowidlist $row] $i]
1645 if {$p == {}} continue
1646 if {$i == $corner} {
1647 if {![info exists cornercrossings($id)]
1648 || [lsearch -exact $cornercrossings($id) $p] < 0} {
1649 lappend cornercrossings($id) $p
1650 }
1651 if {![info exists cornercrossings($p)]
1652 || [lsearch -exact $cornercrossings($p) $id] < 0} {
1653 lappend cornercrossings($p) $id
1654 }
1655 } else {
1656 if {![info exists crossings($id)]
1657 || [lsearch -exact $crossings($id) $p] < 0} {
1658 lappend crossings($id) $p
1659 }
1660 if {![info exists crossings($p)]
1661 || [lsearch -exact $crossings($p) $id] < 0} {
1662 lappend crossings($p) $id
1663 }
1664 }
1665 }
1666}
1667
1668proc xcoord {i level ln} {
1669 global canvx0 xspc1 xspc2
1670
1671 set x [expr {$canvx0 + $i * $xspc1($ln)}]
1672 if {$i > 0 && $i == $level} {
1673 set x [expr {$x + 0.5 * ($xspc2 - $xspc1($ln))}]
1674 } elseif {$i > $level} {
1675 set x [expr {$x + $xspc2 - $xspc1($ln)}]
1676 }
1677 return $x
1678}
1679
1680proc finishcommits {} {
1681 global commitidx phase
1682 global canv mainfont ctext maincursor textcursor
1683 global findinprogress
1684
1685 if {$commitidx > 0} {
1686 drawrest
1687 } else {
1688 $canv delete all
1689 $canv create text 3 3 -anchor nw -text "No commits selected" \
1690 -font $mainfont -tags textitems
1691 }
1692 if {![info exists findinprogress]} {
1693 . config -cursor $maincursor
1694 settextcursor $textcursor
1695 }
1696 set phase {}
1697}
1698
1699# Don't change the text pane cursor if it is currently the hand cursor,
1700# showing that we are over a sha1 ID link.
1701proc settextcursor {c} {
1702 global ctext curtextcursor
1703
1704 if {[$ctext cget -cursor] == $curtextcursor} {
1705 $ctext config -cursor $c
1706 }
1707 set curtextcursor $c
1708}
1709
1710proc drawrest {} {
1711 global numcommits
1712 global startmsecs
1713 global canvy0 numcommits linespc
1714 global rowlaidout commitidx
1715
1716 set row $rowlaidout
1717 layoutrows $rowlaidout $commitidx 1
1718 layouttail
1719 optimize_rows $row 0 $commitidx
1720 showstuff $commitidx
1721
1722 set drawmsecs [expr {[clock clicks -milliseconds] - $startmsecs}]
1723 #puts "overall $drawmsecs ms for $numcommits commits"
1724}
1725
1726proc findmatches {f} {
1727 global findtype foundstring foundstrlen
1728 if {$findtype == "Regexp"} {
1729 set matches [regexp -indices -all -inline $foundstring $f]
1730 } else {
1731 if {$findtype == "IgnCase"} {
1732 set str [string tolower $f]
1733 } else {
1734 set str $f
1735 }
1736 set matches {}
1737 set i 0
1738 while {[set j [string first $foundstring $str $i]] >= 0} {
1739 lappend matches [list $j [expr {$j+$foundstrlen-1}]]
1740 set i [expr {$j + $foundstrlen}]
1741 }
1742 }
1743 return $matches
1744}
1745
1746proc dofind {} {
1747 global findtype findloc findstring markedmatches commitinfo
1748 global numcommits displayorder linehtag linentag linedtag
1749 global mainfont namefont canv canv2 canv3 selectedline
1750 global matchinglines foundstring foundstrlen matchstring
1751 global commitdata
1752
1753 stopfindproc
1754 unmarkmatches
1755 focus .
1756 set matchinglines {}
1757 if {$findloc == "Pickaxe"} {
1758 findpatches
1759 return
1760 }
1761 if {$findtype == "IgnCase"} {
1762 set foundstring [string tolower $findstring]
1763 } else {
1764 set foundstring $findstring
1765 }
1766 set foundstrlen [string length $findstring]
1767 if {$foundstrlen == 0} return
1768 regsub -all {[*?\[\\]} $foundstring {\\&} matchstring
1769 set matchstring "*$matchstring*"
1770 if {$findloc == "Files"} {
1771 findfiles
1772 return
1773 }
1774 if {![info exists selectedline]} {
1775 set oldsel -1
1776 } else {
1777 set oldsel $selectedline
1778 }
1779 set didsel 0
1780 set fldtypes {Headline Author Date Committer CDate Comment}
1781 set l -1
1782 foreach id $displayorder {
1783 set d $commitdata($id)
1784 incr l
1785 if {$findtype == "Regexp"} {
1786 set doesmatch [regexp $foundstring $d]
1787 } elseif {$findtype == "IgnCase"} {
1788 set doesmatch [string match -nocase $matchstring $d]
1789 } else {
1790 set doesmatch [string match $matchstring $d]
1791 }
1792 if {!$doesmatch} continue
1793 if {![info exists commitinfo($id)]} {
1794 getcommit $id
1795 }
1796 set info $commitinfo($id)
1797 set doesmatch 0
1798 foreach f $info ty $fldtypes {
1799 if {$findloc != "All fields" && $findloc != $ty} {
1800 continue
1801 }
1802 set matches [findmatches $f]
1803 if {$matches == {}} continue
1804 set doesmatch 1
1805 if {$ty == "Headline"} {
1806 drawcmitrow $l
1807 markmatches $canv $l $f $linehtag($l) $matches $mainfont
1808 } elseif {$ty == "Author"} {
1809 drawcmitrow $l
1810 markmatches $canv2 $l $f $linentag($l) $matches $namefont
1811 } elseif {$ty == "Date"} {
1812 drawcmitrow $l
1813 markmatches $canv3 $l $f $linedtag($l) $matches $mainfont
1814 }
1815 }
1816 if {$doesmatch} {
1817 lappend matchinglines $l
1818 if {!$didsel && $l > $oldsel} {
1819 findselectline $l
1820 set didsel 1
1821 }
1822 }
1823 }
1824 if {$matchinglines == {}} {
1825 bell
1826 } elseif {!$didsel} {
1827 findselectline [lindex $matchinglines 0]
1828 }
1829}
1830
1831proc findselectline {l} {
1832 global findloc commentend ctext
1833 selectline $l 1
1834 if {$findloc == "All fields" || $findloc == "Comments"} {
1835 # highlight the matches in the comments
1836 set f [$ctext get 1.0 $commentend]
1837 set matches [findmatches $f]
1838 foreach match $matches {
1839 set start [lindex $match 0]
1840 set end [expr {[lindex $match 1] + 1}]
1841 $ctext tag add found "1.0 + $start c" "1.0 + $end c"
1842 }
1843 }
1844}
1845
1846proc findnext {restart} {
1847 global matchinglines selectedline
1848 if {![info exists matchinglines]} {
1849 if {$restart} {
1850 dofind
1851 }
1852 return
1853 }
1854 if {![info exists selectedline]} return
1855 foreach l $matchinglines {
1856 if {$l > $selectedline} {
1857 findselectline $l
1858 return
1859 }
1860 }
1861 bell
1862}
1863
1864proc findprev {} {
1865 global matchinglines selectedline
1866 if {![info exists matchinglines]} {
1867 dofind
1868 return
1869 }
1870 if {![info exists selectedline]} return
1871 set prev {}
1872 foreach l $matchinglines {
1873 if {$l >= $selectedline} break
1874 set prev $l
1875 }
1876 if {$prev != {}} {
1877 findselectline $prev
1878 } else {
1879 bell
1880 }
1881}
1882
1883proc findlocchange {name ix op} {
1884 global findloc findtype findtypemenu
1885 if {$findloc == "Pickaxe"} {
1886 set findtype Exact
1887 set state disabled
1888 } else {
1889 set state normal
1890 }
1891 $findtypemenu entryconf 1 -state $state
1892 $findtypemenu entryconf 2 -state $state
1893}
1894
1895proc stopfindproc {{done 0}} {
1896 global findprocpid findprocfile findids
1897 global ctext findoldcursor phase maincursor textcursor
1898 global findinprogress
1899
1900 catch {unset findids}
1901 if {[info exists findprocpid]} {
1902 if {!$done} {
1903 catch {exec kill $findprocpid}
1904 }
1905 catch {close $findprocfile}
1906 unset findprocpid
1907 }
1908 if {[info exists findinprogress]} {
1909 unset findinprogress
1910 if {$phase != "incrdraw"} {
1911 . config -cursor $maincursor
1912 settextcursor $textcursor
1913 }
1914 }
1915}
1916
1917proc findpatches {} {
1918 global findstring selectedline numcommits
1919 global findprocpid findprocfile
1920 global finddidsel ctext displayorder findinprogress
1921 global findinsertpos
1922
1923 if {$numcommits == 0} return
1924
1925 # make a list of all the ids to search, starting at the one
1926 # after the selected line (if any)
1927 if {[info exists selectedline]} {
1928 set l $selectedline
1929 } else {
1930 set l -1
1931 }
1932 set inputids {}
1933 for {set i 0} {$i < $numcommits} {incr i} {
1934 if {[incr l] >= $numcommits} {
1935 set l 0
1936 }
1937 append inputids [lindex $displayorder $l] "\n"
1938 }
1939
1940 if {[catch {
1941 set f [open [list | git-diff-tree --stdin -s -r -S$findstring \
1942 << $inputids] r]
1943 } err]} {
1944 error_popup "Error starting search process: $err"
1945 return
1946 }
1947
1948 set findinsertpos end
1949 set findprocfile $f
1950 set findprocpid [pid $f]
1951 fconfigure $f -blocking 0
1952 fileevent $f readable readfindproc
1953 set finddidsel 0
1954 . config -cursor watch
1955 settextcursor watch
1956 set findinprogress 1
1957}
1958
1959proc readfindproc {} {
1960 global findprocfile finddidsel
1961 global commitrow matchinglines findinsertpos
1962
1963 set n [gets $findprocfile line]
1964 if {$n < 0} {
1965 if {[eof $findprocfile]} {
1966 stopfindproc 1
1967 if {!$finddidsel} {
1968 bell
1969 }
1970 }
1971 return
1972 }
1973 if {![regexp {^[0-9a-f]{40}} $line id]} {
1974 error_popup "Can't parse git-diff-tree output: $line"
1975 stopfindproc
1976 return
1977 }
1978 if {![info exists commitrow($id)]} {
1979 puts stderr "spurious id: $id"
1980 return
1981 }
1982 set l $commitrow($id)
1983 insertmatch $l $id
1984}
1985
1986proc insertmatch {l id} {
1987 global matchinglines findinsertpos finddidsel
1988
1989 if {$findinsertpos == "end"} {
1990 if {$matchinglines != {} && $l < [lindex $matchinglines 0]} {
1991 set matchinglines [linsert $matchinglines 0 $l]
1992 set findinsertpos 1
1993 } else {
1994 lappend matchinglines $l
1995 }
1996 } else {
1997 set matchinglines [linsert $matchinglines $findinsertpos $l]
1998 incr findinsertpos
1999 }
2000 markheadline $l $id
2001 if {!$finddidsel} {
2002 findselectline $l
2003 set finddidsel 1
2004 }
2005}
2006
2007proc findfiles {} {
2008 global selectedline numcommits displayorder ctext
2009 global ffileline finddidsel parents nparents
2010 global findinprogress findstartline findinsertpos
2011 global treediffs fdiffid fdiffsneeded fdiffpos
2012 global findmergefiles
2013
2014 if {$numcommits == 0} return
2015
2016 if {[info exists selectedline]} {
2017 set l [expr {$selectedline + 1}]
2018 } else {
2019 set l 0
2020 }
2021 set ffileline $l
2022 set findstartline $l
2023 set diffsneeded {}
2024 set fdiffsneeded {}
2025 while 1 {
2026 set id [lindex $displayorder $l]
2027 if {$findmergefiles || $nparents($id) == 1} {
2028 if {![info exists treediffs($id)]} {
2029 append diffsneeded "$id\n"
2030 lappend fdiffsneeded $id
2031 }
2032 }
2033 if {[incr l] >= $numcommits} {
2034 set l 0
2035 }
2036 if {$l == $findstartline} break
2037 }
2038
2039 # start off a git-diff-tree process if needed
2040 if {$diffsneeded ne {}} {
2041 if {[catch {
2042 set df [open [list | git-diff-tree -r --stdin << $diffsneeded] r]
2043 } err ]} {
2044 error_popup "Error starting search process: $err"
2045 return
2046 }
2047 catch {unset fdiffid}
2048 set fdiffpos 0
2049 fconfigure $df -blocking 0
2050 fileevent $df readable [list readfilediffs $df]
2051 }
2052
2053 set finddidsel 0
2054 set findinsertpos end
2055 set id [lindex $displayorder $l]
2056 . config -cursor watch
2057 settextcursor watch
2058 set findinprogress 1
2059 findcont $id
2060 update
2061}
2062
2063proc readfilediffs {df} {
2064 global findid fdiffid fdiffs
2065
2066 set n [gets $df line]
2067 if {$n < 0} {
2068 if {[eof $df]} {
2069 donefilediff
2070 if {[catch {close $df} err]} {
2071 stopfindproc
2072 bell
2073 error_popup "Error in git-diff-tree: $err"
2074 } elseif {[info exists findid]} {
2075 set id $findid
2076 stopfindproc
2077 bell
2078 error_popup "Couldn't find diffs for $id"
2079 }
2080 }
2081 return
2082 }
2083 if {[regexp {^([0-9a-f]{40})$} $line match id]} {
2084 # start of a new string of diffs
2085 donefilediff
2086 set fdiffid $id
2087 set fdiffs {}
2088 } elseif {[string match ":*" $line]} {
2089 lappend fdiffs [lindex $line 5]
2090 }
2091}
2092
2093proc donefilediff {} {
2094 global fdiffid fdiffs treediffs findid
2095 global fdiffsneeded fdiffpos
2096
2097 if {[info exists fdiffid]} {
2098 while {[lindex $fdiffsneeded $fdiffpos] ne $fdiffid
2099 && $fdiffpos < [llength $fdiffsneeded]} {
2100 # git-diff-tree doesn't output anything for a commit
2101 # which doesn't change anything
2102 set nullid [lindex $fdiffsneeded $fdiffpos]
2103 set treediffs($nullid) {}
2104 if {[info exists findid] && $nullid eq $findid} {
2105 unset findid
2106 findcont $nullid
2107 }
2108 incr fdiffpos
2109 }
2110 incr fdiffpos
2111
2112 if {![info exists treediffs($fdiffid)]} {
2113 set treediffs($fdiffid) $fdiffs
2114 }
2115 if {[info exists findid] && $fdiffid eq $findid} {
2116 unset findid
2117 findcont $fdiffid
2118 }
2119 }
2120}
2121
2122proc findcont {id} {
2123 global findid treediffs parents nparents
2124 global ffileline findstartline finddidsel
2125 global displayorder numcommits matchinglines findinprogress
2126 global findmergefiles
2127
2128 set l $ffileline
2129 while 1 {
2130 if {$findmergefiles || $nparents($id) == 1} {
2131 if {![info exists treediffs($id)]} {
2132 set findid $id
2133 set ffileline $l
2134 return
2135 }
2136 set doesmatch 0
2137 foreach f $treediffs($id) {
2138 set x [findmatches $f]
2139 if {$x != {}} {
2140 set doesmatch 1
2141 break
2142 }
2143 }
2144 if {$doesmatch} {
2145 insertmatch $l $id
2146 }
2147 }
2148 if {[incr l] >= $numcommits} {
2149 set l 0
2150 }
2151 if {$l == $findstartline} break
2152 set id [lindex $displayorder $l]
2153 }
2154 stopfindproc
2155 if {!$finddidsel} {
2156 bell
2157 }
2158}
2159
2160# mark a commit as matching by putting a yellow background
2161# behind the headline
2162proc markheadline {l id} {
2163 global canv mainfont linehtag
2164
2165 drawcmitrow $l
2166 set bbox [$canv bbox $linehtag($l)]
2167 set t [$canv create rect $bbox -outline {} -tags matches -fill yellow]
2168 $canv lower $t
2169}
2170
2171# mark the bits of a headline, author or date that match a find string
2172proc markmatches {canv l str tag matches font} {
2173 set bbox [$canv bbox $tag]
2174 set x0 [lindex $bbox 0]
2175 set y0 [lindex $bbox 1]
2176 set y1 [lindex $bbox 3]
2177 foreach match $matches {
2178 set start [lindex $match 0]
2179 set end [lindex $match 1]
2180 if {$start > $end} continue
2181 set xoff [font measure $font [string range $str 0 [expr {$start-1}]]]
2182 set xlen [font measure $font [string range $str 0 [expr {$end}]]]
2183 set t [$canv create rect [expr {$x0+$xoff}] $y0 \
2184 [expr {$x0+$xlen+2}] $y1 \
2185 -outline {} -tags matches -fill yellow]
2186 $canv lower $t
2187 }
2188}
2189
2190proc unmarkmatches {} {
2191 global matchinglines findids
2192 allcanvs delete matches
2193 catch {unset matchinglines}
2194 catch {unset findids}
2195}
2196
2197proc selcanvline {w x y} {
2198 global canv canvy0 ctext linespc
2199 global rowtextx
2200 set ymax [lindex [$canv cget -scrollregion] 3]
2201 if {$ymax == {}} return
2202 set yfrac [lindex [$canv yview] 0]
2203 set y [expr {$y + $yfrac * $ymax}]
2204 set l [expr {int(($y - $canvy0) / $linespc + 0.5)}]
2205 if {$l < 0} {
2206 set l 0
2207 }
2208 if {$w eq $canv} {
2209 if {![info exists rowtextx($l)] || $x < $rowtextx($l)} return
2210 }
2211 unmarkmatches
2212 selectline $l 1
2213}
2214
2215proc commit_descriptor {p} {
2216 global commitinfo
2217 set l "..."
2218 if {[info exists commitinfo($p)]} {
2219 set l [lindex $commitinfo($p) 0]
2220 }
2221 return "$p ($l)"
2222}
2223
2224# append some text to the ctext widget, and make any SHA1 ID
2225# that we know about be a clickable link.
2226proc appendwithlinks {text} {
2227 global ctext commitrow linknum
2228
2229 set start [$ctext index "end - 1c"]
2230 $ctext insert end $text
2231 $ctext insert end "\n"
2232 set links [regexp -indices -all -inline {[0-9a-f]{40}} $text]
2233 foreach l $links {
2234 set s [lindex $l 0]
2235 set e [lindex $l 1]
2236 set linkid [string range $text $s $e]
2237 if {![info exists commitrow($linkid)]} continue
2238 incr e
2239 $ctext tag add link "$start + $s c" "$start + $e c"
2240 $ctext tag add link$linknum "$start + $s c" "$start + $e c"
2241 $ctext tag bind link$linknum <1> [list selectline $commitrow($linkid) 1]
2242 incr linknum
2243 }
2244 $ctext tag conf link -foreground blue -underline 1
2245 $ctext tag bind link <Enter> { %W configure -cursor hand2 }
2246 $ctext tag bind link <Leave> { %W configure -cursor $curtextcursor }
2247}
2248
2249proc selectline {l isnew} {
2250 global canv canv2 canv3 ctext commitinfo selectedline
2251 global displayorder linehtag linentag linedtag
2252 global canvy0 linespc parents nparents children
2253 global cflist currentid sha1entry
2254 global commentend idtags linknum
2255 global mergemax numcommits
2256
2257 $canv delete hover
2258 normalline
2259 if {$l < 0 || $l >= $numcommits} return
2260 set y [expr {$canvy0 + $l * $linespc}]
2261 set ymax [lindex [$canv cget -scrollregion] 3]
2262 set ytop [expr {$y - $linespc - 1}]
2263 set ybot [expr {$y + $linespc + 1}]
2264 set wnow [$canv yview]
2265 set wtop [expr {[lindex $wnow 0] * $ymax}]
2266 set wbot [expr {[lindex $wnow 1] * $ymax}]
2267 set wh [expr {$wbot - $wtop}]
2268 set newtop $wtop
2269 if {$ytop < $wtop} {
2270 if {$ybot < $wtop} {
2271 set newtop [expr {$y - $wh / 2.0}]
2272 } else {
2273 set newtop $ytop
2274 if {$newtop > $wtop - $linespc} {
2275 set newtop [expr {$wtop - $linespc}]
2276 }
2277 }
2278 } elseif {$ybot > $wbot} {
2279 if {$ytop > $wbot} {
2280 set newtop [expr {$y - $wh / 2.0}]
2281 } else {
2282 set newtop [expr {$ybot - $wh}]
2283 if {$newtop < $wtop + $linespc} {
2284 set newtop [expr {$wtop + $linespc}]
2285 }
2286 }
2287 }
2288 if {$newtop != $wtop} {
2289 if {$newtop < 0} {
2290 set newtop 0
2291 }
2292 allcanvs yview moveto [expr {$newtop * 1.0 / $ymax}]
2293 drawvisible
2294 }
2295
2296 if {![info exists linehtag($l)]} return
2297 $canv delete secsel
2298 set t [eval $canv create rect [$canv bbox $linehtag($l)] -outline {{}} \
2299 -tags secsel -fill [$canv cget -selectbackground]]
2300 $canv lower $t
2301 $canv2 delete secsel
2302 set t [eval $canv2 create rect [$canv2 bbox $linentag($l)] -outline {{}} \
2303 -tags secsel -fill [$canv2 cget -selectbackground]]
2304 $canv2 lower $t
2305 $canv3 delete secsel
2306 set t [eval $canv3 create rect [$canv3 bbox $linedtag($l)] -outline {{}} \
2307 -tags secsel -fill [$canv3 cget -selectbackground]]
2308 $canv3 lower $t
2309
2310 if {$isnew} {
2311 addtohistory [list selectline $l 0]
2312 }
2313
2314 set selectedline $l
2315
2316 set id [lindex $displayorder $l]
2317 set currentid $id
2318 $sha1entry delete 0 end
2319 $sha1entry insert 0 $id
2320 $sha1entry selection from 0
2321 $sha1entry selection to end
2322
2323 $ctext conf -state normal
2324 $ctext delete 0.0 end
2325 set linknum 0
2326 $ctext mark set fmark.0 0.0
2327 $ctext mark gravity fmark.0 left
2328 set info $commitinfo($id)
2329 set date [formatdate [lindex $info 2]]
2330 $ctext insert end "Author: [lindex $info 1] $date\n"
2331 set date [formatdate [lindex $info 4]]
2332 $ctext insert end "Committer: [lindex $info 3] $date\n"
2333 if {[info exists idtags($id)]} {
2334 $ctext insert end "Tags:"
2335 foreach tag $idtags($id) {
2336 $ctext insert end " $tag"
2337 }
2338 $ctext insert end "\n"
2339 }
2340
2341 set comment {}
2342 if {$nparents($id) > 1} {
2343 set np 0
2344 foreach p $parents($id) {
2345 if {$np >= $mergemax} {
2346 set tag mmax
2347 } else {
2348 set tag m$np
2349 }
2350 $ctext insert end "Parent: " $tag
2351 appendwithlinks [commit_descriptor $p]
2352 incr np
2353 }
2354 } else {
2355 if {[info exists parents($id)]} {
2356 foreach p $parents($id) {
2357 append comment "Parent: [commit_descriptor $p]\n"
2358 }
2359 }
2360 }
2361
2362 if {[info exists children($id)]} {
2363 foreach c $children($id) {
2364 append comment "Child: [commit_descriptor $c]\n"
2365 }
2366 }
2367 append comment "\n"
2368 append comment [lindex $info 5]
2369
2370 # make anything that looks like a SHA1 ID be a clickable link
2371 appendwithlinks $comment
2372
2373 $ctext tag delete Comments
2374 $ctext tag remove found 1.0 end
2375 $ctext conf -state disabled
2376 set commentend [$ctext index "end - 1c"]
2377
2378 $cflist delete 0 end
2379 $cflist insert end "Comments"
2380 if {$nparents($id) == 1} {
2381 startdiff $id
2382 } elseif {$nparents($id) > 1} {
2383 mergediff $id
2384 }
2385}
2386
2387proc selnextline {dir} {
2388 global selectedline
2389 if {![info exists selectedline]} return
2390 set l [expr {$selectedline + $dir}]
2391 unmarkmatches
2392 selectline $l 1
2393}
2394
2395proc unselectline {} {
2396 global selectedline
2397
2398 catch {unset selectedline}
2399 allcanvs delete secsel
2400}
2401
2402proc addtohistory {cmd} {
2403 global history historyindex
2404
2405 if {$historyindex > 0
2406 && [lindex $history [expr {$historyindex - 1}]] == $cmd} {
2407 return
2408 }
2409
2410 if {$historyindex < [llength $history]} {
2411 set history [lreplace $history $historyindex end $cmd]
2412 } else {
2413 lappend history $cmd
2414 }
2415 incr historyindex
2416 if {$historyindex > 1} {
2417 .ctop.top.bar.leftbut conf -state normal
2418 } else {
2419 .ctop.top.bar.leftbut conf -state disabled
2420 }
2421 .ctop.top.bar.rightbut conf -state disabled
2422}
2423
2424proc goback {} {
2425 global history historyindex
2426
2427 if {$historyindex > 1} {
2428 incr historyindex -1
2429 set cmd [lindex $history [expr {$historyindex - 1}]]
2430 eval $cmd
2431 .ctop.top.bar.rightbut conf -state normal
2432 }
2433 if {$historyindex <= 1} {
2434 .ctop.top.bar.leftbut conf -state disabled
2435 }
2436}
2437
2438proc goforw {} {
2439 global history historyindex
2440
2441 if {$historyindex < [llength $history]} {
2442 set cmd [lindex $history $historyindex]
2443 incr historyindex
2444 eval $cmd
2445 .ctop.top.bar.leftbut conf -state normal
2446 }
2447 if {$historyindex >= [llength $history]} {
2448 .ctop.top.bar.rightbut conf -state disabled
2449 }
2450}
2451
2452proc mergediff {id} {
2453 global parents diffmergeid diffopts mdifffd
2454 global difffilestart diffids
2455
2456 set diffmergeid $id
2457 set diffids $id
2458 catch {unset difffilestart}
2459 # this doesn't seem to actually affect anything...
2460 set env(GIT_DIFF_OPTS) $diffopts
2461 set cmd [concat | git-diff-tree --no-commit-id --cc $id]
2462 if {[catch {set mdf [open $cmd r]} err]} {
2463 error_popup "Error getting merge diffs: $err"
2464 return
2465 }
2466 fconfigure $mdf -blocking 0
2467 set mdifffd($id) $mdf
2468 fileevent $mdf readable [list getmergediffline $mdf $id]
2469 set nextupdate [expr {[clock clicks -milliseconds] + 100}]
2470}
2471
2472proc getmergediffline {mdf id} {
2473 global diffmergeid ctext cflist nextupdate nparents mergemax
2474 global difffilestart mdifffd
2475
2476 set n [gets $mdf line]
2477 if {$n < 0} {
2478 if {[eof $mdf]} {
2479 close $mdf
2480 }
2481 return
2482 }
2483 if {![info exists diffmergeid] || $id != $diffmergeid
2484 || $mdf != $mdifffd($id)} {
2485 return
2486 }
2487 $ctext conf -state normal
2488 if {[regexp {^diff --cc (.*)} $line match fname]} {
2489 # start of a new file
2490 $ctext insert end "\n"
2491 set here [$ctext index "end - 1c"]
2492 set i [$cflist index end]
2493 $ctext mark set fmark.$i $here
2494 $ctext mark gravity fmark.$i left
2495 set difffilestart([expr {$i-1}]) $here
2496 $cflist insert end $fname
2497 set l [expr {(78 - [string length $fname]) / 2}]
2498 set pad [string range "----------------------------------------" 1 $l]
2499 $ctext insert end "$pad $fname $pad\n" filesep
2500 } elseif {[regexp {^@@} $line]} {
2501 $ctext insert end "$line\n" hunksep
2502 } elseif {[regexp {^[0-9a-f]{40}$} $line] || [regexp {^index} $line]} {
2503 # do nothing
2504 } else {
2505 # parse the prefix - one ' ', '-' or '+' for each parent
2506 set np $nparents($id)
2507 set spaces {}
2508 set minuses {}
2509 set pluses {}
2510 set isbad 0
2511 for {set j 0} {$j < $np} {incr j} {
2512 set c [string range $line $j $j]
2513 if {$c == " "} {
2514 lappend spaces $j
2515 } elseif {$c == "-"} {
2516 lappend minuses $j
2517 } elseif {$c == "+"} {
2518 lappend pluses $j
2519 } else {
2520 set isbad 1
2521 break
2522 }
2523 }
2524 set tags {}
2525 set num {}
2526 if {!$isbad && $minuses ne {} && $pluses eq {}} {
2527 # line doesn't appear in result, parents in $minuses have the line
2528 set num [lindex $minuses 0]
2529 } elseif {!$isbad && $pluses ne {} && $minuses eq {}} {
2530 # line appears in result, parents in $pluses don't have the line
2531 lappend tags mresult
2532 set num [lindex $spaces 0]
2533 }
2534 if {$num ne {}} {
2535 if {$num >= $mergemax} {
2536 set num "max"
2537 }
2538 lappend tags m$num
2539 }
2540 $ctext insert end "$line\n" $tags
2541 }
2542 $ctext conf -state disabled
2543 if {[clock clicks -milliseconds] >= $nextupdate} {
2544 incr nextupdate 100
2545 fileevent $mdf readable {}
2546 update
2547 fileevent $mdf readable [list getmergediffline $mdf $id]
2548 }
2549}
2550
2551proc startdiff {ids} {
2552 global treediffs diffids treepending diffmergeid
2553
2554 set diffids $ids
2555 catch {unset diffmergeid}
2556 if {![info exists treediffs($ids)]} {
2557 if {![info exists treepending]} {
2558 gettreediffs $ids
2559 }
2560 } else {
2561 addtocflist $ids
2562 }
2563}
2564
2565proc addtocflist {ids} {
2566 global treediffs cflist
2567 foreach f $treediffs($ids) {
2568 $cflist insert end $f
2569 }
2570 getblobdiffs $ids
2571}
2572
2573proc gettreediffs {ids} {
2574 global treediff parents treepending
2575 set treepending $ids
2576 set treediff {}
2577 if {[catch \
2578 {set gdtf [open [concat | git-diff-tree --no-commit-id -r $ids] r]} \
2579 ]} return
2580 fconfigure $gdtf -blocking 0
2581 fileevent $gdtf readable [list gettreediffline $gdtf $ids]
2582}
2583
2584proc gettreediffline {gdtf ids} {
2585 global treediff treediffs treepending diffids diffmergeid
2586
2587 set n [gets $gdtf line]
2588 if {$n < 0} {
2589 if {![eof $gdtf]} return
2590 close $gdtf
2591 set treediffs($ids) $treediff
2592 unset treepending
2593 if {$ids != $diffids} {
2594 if {![info exists diffmergeid]} {
2595 gettreediffs $diffids
2596 }
2597 } else {
2598 addtocflist $ids
2599 }
2600 return
2601 }
2602 set file [lindex $line 5]
2603 lappend treediff $file
2604}
2605
2606proc getblobdiffs {ids} {
2607 global diffopts blobdifffd diffids env curdifftag curtagstart
2608 global difffilestart nextupdate diffinhdr treediffs
2609
2610 set env(GIT_DIFF_OPTS) $diffopts
2611 set cmd [concat | git-diff-tree --no-commit-id -r -p -C $ids]
2612 if {[catch {set bdf [open $cmd r]} err]} {
2613 puts "error getting diffs: $err"
2614 return
2615 }
2616 set diffinhdr 0
2617 fconfigure $bdf -blocking 0
2618 set blobdifffd($ids) $bdf
2619 set curdifftag Comments
2620 set curtagstart 0.0
2621 catch {unset difffilestart}
2622 fileevent $bdf readable [list getblobdiffline $bdf $diffids]
2623 set nextupdate [expr {[clock clicks -milliseconds] + 100}]
2624}
2625
2626proc getblobdiffline {bdf ids} {
2627 global diffids blobdifffd ctext curdifftag curtagstart
2628 global diffnexthead diffnextnote difffilestart
2629 global nextupdate diffinhdr treediffs
2630
2631 set n [gets $bdf line]
2632 if {$n < 0} {
2633 if {[eof $bdf]} {
2634 close $bdf
2635 if {$ids == $diffids && $bdf == $blobdifffd($ids)} {
2636 $ctext tag add $curdifftag $curtagstart end
2637 }
2638 }
2639 return
2640 }
2641 if {$ids != $diffids || $bdf != $blobdifffd($ids)} {
2642 return
2643 }
2644 $ctext conf -state normal
2645 if {[regexp {^diff --git a/(.*) b/(.*)} $line match fname newname]} {
2646 # start of a new file
2647 $ctext insert end "\n"
2648 $ctext tag add $curdifftag $curtagstart end
2649 set curtagstart [$ctext index "end - 1c"]
2650 set header $newname
2651 set here [$ctext index "end - 1c"]
2652 set i [lsearch -exact $treediffs($diffids) $fname]
2653 if {$i >= 0} {
2654 set difffilestart($i) $here
2655 incr i
2656 $ctext mark set fmark.$i $here
2657 $ctext mark gravity fmark.$i left
2658 }
2659 if {$newname != $fname} {
2660 set i [lsearch -exact $treediffs($diffids) $newname]
2661 if {$i >= 0} {
2662 set difffilestart($i) $here
2663 incr i
2664 $ctext mark set fmark.$i $here
2665 $ctext mark gravity fmark.$i left
2666 }
2667 }
2668 set curdifftag "f:$fname"
2669 $ctext tag delete $curdifftag
2670 set l [expr {(78 - [string length $header]) / 2}]
2671 set pad [string range "----------------------------------------" 1 $l]
2672 $ctext insert end "$pad $header $pad\n" filesep
2673 set diffinhdr 1
2674 } elseif {$diffinhdr && [string compare -length 3 $line "---"] == 0} {
2675 # do nothing
2676 } elseif {$diffinhdr && [string compare -length 3 $line "+++"] == 0} {
2677 set diffinhdr 0
2678 } elseif {[regexp {^@@ -([0-9]+),([0-9]+) \+([0-9]+),([0-9]+) @@(.*)} \
2679 $line match f1l f1c f2l f2c rest]} {
2680 $ctext insert end "$line\n" hunksep
2681 set diffinhdr 0
2682 } else {
2683 set x [string range $line 0 0]
2684 if {$x == "-" || $x == "+"} {
2685 set tag [expr {$x == "+"}]
2686 $ctext insert end "$line\n" d$tag
2687 } elseif {$x == " "} {
2688 $ctext insert end "$line\n"
2689 } elseif {$diffinhdr || $x == "\\"} {
2690 # e.g. "\ No newline at end of file"
2691 $ctext insert end "$line\n" filesep
2692 } else {
2693 # Something else we don't recognize
2694 if {$curdifftag != "Comments"} {
2695 $ctext insert end "\n"
2696 $ctext tag add $curdifftag $curtagstart end
2697 set curtagstart [$ctext index "end - 1c"]
2698 set curdifftag Comments
2699 }
2700 $ctext insert end "$line\n" filesep
2701 }
2702 }
2703 $ctext conf -state disabled
2704 if {[clock clicks -milliseconds] >= $nextupdate} {
2705 incr nextupdate 100
2706 fileevent $bdf readable {}
2707 update
2708 fileevent $bdf readable "getblobdiffline $bdf {$ids}"
2709 }
2710}
2711
2712proc nextfile {} {
2713 global difffilestart ctext
2714 set here [$ctext index @0,0]
2715 for {set i 0} {[info exists difffilestart($i)]} {incr i} {
2716 if {[$ctext compare $difffilestart($i) > $here]} {
2717 if {![info exists pos]
2718 || [$ctext compare $difffilestart($i) < $pos]} {
2719 set pos $difffilestart($i)
2720 }
2721 }
2722 }
2723 if {[info exists pos]} {
2724 $ctext yview $pos
2725 }
2726}
2727
2728proc listboxsel {} {
2729 global ctext cflist currentid
2730 if {![info exists currentid]} return
2731 set sel [lsort [$cflist curselection]]
2732 if {$sel eq {}} return
2733 set first [lindex $sel 0]
2734 catch {$ctext yview fmark.$first}
2735}
2736
2737proc setcoords {} {
2738 global linespc charspc canvx0 canvy0 mainfont
2739 global xspc1 xspc2 lthickness
2740
2741 set linespc [font metrics $mainfont -linespace]
2742 set charspc [font measure $mainfont "m"]
2743 set canvy0 [expr {int(3 + 0.5 * $linespc)}]
2744 set canvx0 [expr {int(3 + 0.5 * $linespc)}]
2745 set lthickness [expr {int($linespc / 9) + 1}]
2746 set xspc1(0) $linespc
2747 set xspc2 $linespc
2748}
2749
2750proc redisplay {} {
2751 global canv canvy0 linespc numcommits
2752 global selectedline
2753
2754 set ymax [lindex [$canv cget -scrollregion] 3]
2755 if {$ymax eq {} || $ymax == 0} return
2756 set span [$canv yview]
2757 clear_display
2758 allcanvs conf -scrollregion \
2759 [list 0 0 0 [expr {$canvy0 + ($numcommits - 0.5) * $linespc + 2}]]
2760 allcanvs yview moveto [lindex $span 0]
2761 drawvisible
2762 if {[info exists selectedline]} {
2763 selectline $selectedline 0
2764 }
2765}
2766
2767proc incrfont {inc} {
2768 global mainfont namefont textfont ctext canv phase
2769 global stopped entries
2770 unmarkmatches
2771 set mainfont [lreplace $mainfont 1 1 [expr {[lindex $mainfont 1] + $inc}]]
2772 set namefont [lreplace $namefont 1 1 [expr {[lindex $namefont 1] + $inc}]]
2773 set textfont [lreplace $textfont 1 1 [expr {[lindex $textfont 1] + $inc}]]
2774 setcoords
2775 $ctext conf -font $textfont
2776 $ctext tag conf filesep -font [concat $textfont bold]
2777 foreach e $entries {
2778 $e conf -font $mainfont
2779 }
2780 if {$phase == "getcommits"} {
2781 $canv itemconf textitems -font $mainfont
2782 }
2783 redisplay
2784}
2785
2786proc clearsha1 {} {
2787 global sha1entry sha1string
2788 if {[string length $sha1string] == 40} {
2789 $sha1entry delete 0 end
2790 }
2791}
2792
2793proc sha1change {n1 n2 op} {
2794 global sha1string currentid sha1but
2795 if {$sha1string == {}
2796 || ([info exists currentid] && $sha1string == $currentid)} {
2797 set state disabled
2798 } else {
2799 set state normal
2800 }
2801 if {[$sha1but cget -state] == $state} return
2802 if {$state == "normal"} {
2803 $sha1but conf -state normal -relief raised -text "Goto: "
2804 } else {
2805 $sha1but conf -state disabled -relief flat -text "SHA1 ID: "
2806 }
2807}
2808
2809proc gotocommit {} {
2810 global sha1string currentid commitrow tagids
2811 global displayorder numcommits
2812
2813 if {$sha1string == {}
2814 || ([info exists currentid] && $sha1string == $currentid)} return
2815 if {[info exists tagids($sha1string)]} {
2816 set id $tagids($sha1string)
2817 } else {
2818 set id [string tolower $sha1string]
2819 if {[regexp {^[0-9a-f]{4,39}$} $id]} {
2820 set matches {}
2821 foreach i $displayorder {
2822 if {[string match $id* $i]} {
2823 lappend matches $i
2824 }
2825 }
2826 if {$matches ne {}} {
2827 if {[llength $matches] > 1} {
2828 error_popup "Short SHA1 id $id is ambiguous"
2829 return
2830 }
2831 set id [lindex $matches 0]
2832 }
2833 }
2834 }
2835 if {[info exists commitrow($id)]} {
2836 selectline $commitrow($id) 1
2837 return
2838 }
2839 if {[regexp {^[0-9a-fA-F]{4,}$} $sha1string]} {
2840 set type "SHA1 id"
2841 } else {
2842 set type "Tag"
2843 }
2844 error_popup "$type $sha1string is not known"
2845}
2846
2847proc lineenter {x y id} {
2848 global hoverx hovery hoverid hovertimer
2849 global commitinfo canv
2850
2851 if {![info exists commitinfo($id)] && ![getcommit $id]} return
2852 set hoverx $x
2853 set hovery $y
2854 set hoverid $id
2855 if {[info exists hovertimer]} {
2856 after cancel $hovertimer
2857 }
2858 set hovertimer [after 500 linehover]
2859 $canv delete hover
2860}
2861
2862proc linemotion {x y id} {
2863 global hoverx hovery hoverid hovertimer
2864
2865 if {[info exists hoverid] && $id == $hoverid} {
2866 set hoverx $x
2867 set hovery $y
2868 if {[info exists hovertimer]} {
2869 after cancel $hovertimer
2870 }
2871 set hovertimer [after 500 linehover]
2872 }
2873}
2874
2875proc lineleave {id} {
2876 global hoverid hovertimer canv
2877
2878 if {[info exists hoverid] && $id == $hoverid} {
2879 $canv delete hover
2880 if {[info exists hovertimer]} {
2881 after cancel $hovertimer
2882 unset hovertimer
2883 }
2884 unset hoverid
2885 }
2886}
2887
2888proc linehover {} {
2889 global hoverx hovery hoverid hovertimer
2890 global canv linespc lthickness
2891 global commitinfo mainfont
2892
2893 set text [lindex $commitinfo($hoverid) 0]
2894 set ymax [lindex [$canv cget -scrollregion] 3]
2895 if {$ymax == {}} return
2896 set yfrac [lindex [$canv yview] 0]
2897 set x [expr {$hoverx + 2 * $linespc}]
2898 set y [expr {$hovery + $yfrac * $ymax - $linespc / 2}]
2899 set x0 [expr {$x - 2 * $lthickness}]
2900 set y0 [expr {$y - 2 * $lthickness}]
2901 set x1 [expr {$x + [font measure $mainfont $text] + 2 * $lthickness}]
2902 set y1 [expr {$y + $linespc + 2 * $lthickness}]
2903 set t [$canv create rectangle $x0 $y0 $x1 $y1 \
2904 -fill \#ffff80 -outline black -width 1 -tags hover]
2905 $canv raise $t
2906 set t [$canv create text $x $y -anchor nw -text $text -tags hover -font $mainfont]
2907 $canv raise $t
2908}
2909
2910proc clickisonarrow {id y} {
2911 global lthickness idrowranges
2912
2913 set thresh [expr {2 * $lthickness + 6}]
2914 set n [expr {[llength $idrowranges($id)] - 1}]
2915 for {set i 1} {$i < $n} {incr i} {
2916 set row [lindex $idrowranges($id) $i]
2917 if {abs([yc $row] - $y) < $thresh} {
2918 return $i
2919 }
2920 }
2921 return {}
2922}
2923
2924proc arrowjump {id n y} {
2925 global idrowranges canv
2926
2927 # 1 <-> 2, 3 <-> 4, etc...
2928 set n [expr {(($n - 1) ^ 1) + 1}]
2929 set row [lindex $idrowranges($id) $n]
2930 set yt [yc $row]
2931 set ymax [lindex [$canv cget -scrollregion] 3]
2932 if {$ymax eq {} || $ymax <= 0} return
2933 set view [$canv yview]
2934 set yspan [expr {[lindex $view 1] - [lindex $view 0]}]
2935 set yfrac [expr {$yt / $ymax - $yspan / 2}]
2936 if {$yfrac < 0} {
2937 set yfrac 0
2938 }
2939 allcanvs yview moveto $yfrac
2940}
2941
2942proc lineclick {x y id isnew} {
2943 global ctext commitinfo children cflist canv thickerline
2944
2945 if {![info exists commitinfo($id)] && ![getcommit $id]} return
2946 unmarkmatches
2947 unselectline
2948 normalline
2949 $canv delete hover
2950 # draw this line thicker than normal
2951 set thickerline $id
2952 drawlines $id
2953 if {$isnew} {
2954 set ymax [lindex [$canv cget -scrollregion] 3]
2955 if {$ymax eq {}} return
2956 set yfrac [lindex [$canv yview] 0]
2957 set y [expr {$y + $yfrac * $ymax}]
2958 }
2959 set dirn [clickisonarrow $id $y]
2960 if {$dirn ne {}} {
2961 arrowjump $id $dirn $y
2962 return
2963 }
2964
2965 if {$isnew} {
2966 addtohistory [list lineclick $x $y $id 0]
2967 }
2968 # fill the details pane with info about this line
2969 $ctext conf -state normal
2970 $ctext delete 0.0 end
2971 $ctext tag conf link -foreground blue -underline 1
2972 $ctext tag bind link <Enter> { %W configure -cursor hand2 }
2973 $ctext tag bind link <Leave> { %W configure -cursor $curtextcursor }
2974 $ctext insert end "Parent:\t"
2975 $ctext insert end $id [list link link0]
2976 $ctext tag bind link0 <1> [list selbyid $id]
2977 set info $commitinfo($id)
2978 $ctext insert end "\n\t[lindex $info 0]\n"
2979 $ctext insert end "\tAuthor:\t[lindex $info 1]\n"
2980 set date [formatdate [lindex $info 2]]
2981 $ctext insert end "\tDate:\t$date\n"
2982 if {[info exists children($id)]} {
2983 $ctext insert end "\nChildren:"
2984 set i 0
2985 foreach child $children($id) {
2986 incr i
2987 if {![info exists commitinfo($child)] && ![getcommit $child]} continue
2988 set info $commitinfo($child)
2989 $ctext insert end "\n\t"
2990 $ctext insert end $child [list link link$i]
2991 $ctext tag bind link$i <1> [list selbyid $child]
2992 $ctext insert end "\n\t[lindex $info 0]"
2993 $ctext insert end "\n\tAuthor:\t[lindex $info 1]"
2994 set date [formatdate [lindex $info 2]]
2995 $ctext insert end "\n\tDate:\t$date\n"
2996 }
2997 }
2998 $ctext conf -state disabled
2999
3000 $cflist delete 0 end
3001}
3002
3003proc normalline {} {
3004 global thickerline
3005 if {[info exists thickerline]} {
3006 set id $thickerline
3007 unset thickerline
3008 drawlines $id
3009 }
3010}
3011
3012proc selbyid {id} {
3013 global commitrow
3014 if {[info exists commitrow($id)]} {
3015 selectline $commitrow($id) 1
3016 }
3017}
3018
3019proc mstime {} {
3020 global startmstime
3021 if {![info exists startmstime]} {
3022 set startmstime [clock clicks -milliseconds]
3023 }
3024 return [format "%.3f" [expr {([clock click -milliseconds] - $startmstime) / 1000.0}]]
3025}
3026
3027proc rowmenu {x y id} {
3028 global rowctxmenu commitrow selectedline rowmenuid
3029
3030 if {![info exists selectedline] || $commitrow($id) eq $selectedline} {
3031 set state disabled
3032 } else {
3033 set state normal
3034 }
3035 $rowctxmenu entryconfigure 0 -state $state
3036 $rowctxmenu entryconfigure 1 -state $state
3037 $rowctxmenu entryconfigure 2 -state $state
3038 set rowmenuid $id
3039 tk_popup $rowctxmenu $x $y
3040}
3041
3042proc diffvssel {dirn} {
3043 global rowmenuid selectedline displayorder
3044
3045 if {![info exists selectedline]} return
3046 if {$dirn} {
3047 set oldid [lindex $displayorder $selectedline]
3048 set newid $rowmenuid
3049 } else {
3050 set oldid $rowmenuid
3051 set newid [lindex $displayorder $selectedline]
3052 }
3053 addtohistory [list doseldiff $oldid $newid]
3054 doseldiff $oldid $newid
3055}
3056
3057proc doseldiff {oldid newid} {
3058 global ctext cflist
3059 global commitinfo
3060
3061 $ctext conf -state normal
3062 $ctext delete 0.0 end
3063 $ctext mark set fmark.0 0.0
3064 $ctext mark gravity fmark.0 left
3065 $cflist delete 0 end
3066 $cflist insert end "Top"
3067 $ctext insert end "From "
3068 $ctext tag conf link -foreground blue -underline 1
3069 $ctext tag bind link <Enter> { %W configure -cursor hand2 }
3070 $ctext tag bind link <Leave> { %W configure -cursor $curtextcursor }
3071 $ctext tag bind link0 <1> [list selbyid $oldid]
3072 $ctext insert end $oldid [list link link0]
3073 $ctext insert end "\n "
3074 $ctext insert end [lindex $commitinfo($oldid) 0]
3075 $ctext insert end "\n\nTo "
3076 $ctext tag bind link1 <1> [list selbyid $newid]
3077 $ctext insert end $newid [list link link1]
3078 $ctext insert end "\n "
3079 $ctext insert end [lindex $commitinfo($newid) 0]
3080 $ctext insert end "\n"
3081 $ctext conf -state disabled
3082 $ctext tag delete Comments
3083 $ctext tag remove found 1.0 end
3084 startdiff [list $oldid $newid]
3085}
3086
3087proc mkpatch {} {
3088 global rowmenuid currentid commitinfo patchtop patchnum
3089
3090 if {![info exists currentid]} return
3091 set oldid $currentid
3092 set oldhead [lindex $commitinfo($oldid) 0]
3093 set newid $rowmenuid
3094 set newhead [lindex $commitinfo($newid) 0]
3095 set top .patch
3096 set patchtop $top
3097 catch {destroy $top}
3098 toplevel $top
3099 label $top.title -text "Generate patch"
3100 grid $top.title - -pady 10
3101 label $top.from -text "From:"
3102 entry $top.fromsha1 -width 40 -relief flat
3103 $top.fromsha1 insert 0 $oldid
3104 $top.fromsha1 conf -state readonly
3105 grid $top.from $top.fromsha1 -sticky w
3106 entry $top.fromhead -width 60 -relief flat
3107 $top.fromhead insert 0 $oldhead
3108 $top.fromhead conf -state readonly
3109 grid x $top.fromhead -sticky w
3110 label $top.to -text "To:"
3111 entry $top.tosha1 -width 40 -relief flat
3112 $top.tosha1 insert 0 $newid
3113 $top.tosha1 conf -state readonly
3114 grid $top.to $top.tosha1 -sticky w
3115 entry $top.tohead -width 60 -relief flat
3116 $top.tohead insert 0 $newhead
3117 $top.tohead conf -state readonly
3118 grid x $top.tohead -sticky w
3119 button $top.rev -text "Reverse" -command mkpatchrev -padx 5
3120 grid $top.rev x -pady 10
3121 label $top.flab -text "Output file:"
3122 entry $top.fname -width 60
3123 $top.fname insert 0 [file normalize "patch$patchnum.patch"]
3124 incr patchnum
3125 grid $top.flab $top.fname -sticky w
3126 frame $top.buts
3127 button $top.buts.gen -text "Generate" -command mkpatchgo
3128 button $top.buts.can -text "Cancel" -command mkpatchcan
3129 grid $top.buts.gen $top.buts.can
3130 grid columnconfigure $top.buts 0 -weight 1 -uniform a
3131 grid columnconfigure $top.buts 1 -weight 1 -uniform a
3132 grid $top.buts - -pady 10 -sticky ew
3133 focus $top.fname
3134}
3135
3136proc mkpatchrev {} {
3137 global patchtop
3138
3139 set oldid [$patchtop.fromsha1 get]
3140 set oldhead [$patchtop.fromhead get]
3141 set newid [$patchtop.tosha1 get]
3142 set newhead [$patchtop.tohead get]
3143 foreach e [list fromsha1 fromhead tosha1 tohead] \
3144 v [list $newid $newhead $oldid $oldhead] {
3145 $patchtop.$e conf -state normal
3146 $patchtop.$e delete 0 end
3147 $patchtop.$e insert 0 $v
3148 $patchtop.$e conf -state readonly
3149 }
3150}
3151
3152proc mkpatchgo {} {
3153 global patchtop
3154
3155 set oldid [$patchtop.fromsha1 get]
3156 set newid [$patchtop.tosha1 get]
3157 set fname [$patchtop.fname get]
3158 if {[catch {exec git-diff-tree -p $oldid $newid >$fname &} err]} {
3159 error_popup "Error creating patch: $err"
3160 }
3161 catch {destroy $patchtop}
3162 unset patchtop
3163}
3164
3165proc mkpatchcan {} {
3166 global patchtop
3167
3168 catch {destroy $patchtop}
3169 unset patchtop
3170}
3171
3172proc mktag {} {
3173 global rowmenuid mktagtop commitinfo
3174
3175 set top .maketag
3176 set mktagtop $top
3177 catch {destroy $top}
3178 toplevel $top
3179 label $top.title -text "Create tag"
3180 grid $top.title - -pady 10
3181 label $top.id -text "ID:"
3182 entry $top.sha1 -width 40 -relief flat
3183 $top.sha1 insert 0 $rowmenuid
3184 $top.sha1 conf -state readonly
3185 grid $top.id $top.sha1 -sticky w
3186 entry $top.head -width 60 -relief flat
3187 $top.head insert 0 [lindex $commitinfo($rowmenuid) 0]
3188 $top.head conf -state readonly
3189 grid x $top.head -sticky w
3190 label $top.tlab -text "Tag name:"
3191 entry $top.tag -width 60
3192 grid $top.tlab $top.tag -sticky w
3193 frame $top.buts
3194 button $top.buts.gen -text "Create" -command mktaggo
3195 button $top.buts.can -text "Cancel" -command mktagcan
3196 grid $top.buts.gen $top.buts.can
3197 grid columnconfigure $top.buts 0 -weight 1 -uniform a
3198 grid columnconfigure $top.buts 1 -weight 1 -uniform a
3199 grid $top.buts - -pady 10 -sticky ew
3200 focus $top.tag
3201}
3202
3203proc domktag {} {
3204 global mktagtop env tagids idtags
3205
3206 set id [$mktagtop.sha1 get]
3207 set tag [$mktagtop.tag get]
3208 if {$tag == {}} {
3209 error_popup "No tag name specified"
3210 return
3211 }
3212 if {[info exists tagids($tag)]} {
3213 error_popup "Tag \"$tag\" already exists"
3214 return
3215 }
3216 if {[catch {
3217 set dir [gitdir]
3218 set fname [file join $dir "refs/tags" $tag]
3219 set f [open $fname w]
3220 puts $f $id
3221 close $f
3222 } err]} {
3223 error_popup "Error creating tag: $err"
3224 return
3225 }
3226
3227 set tagids($tag) $id
3228 lappend idtags($id) $tag
3229 redrawtags $id
3230}
3231
3232proc redrawtags {id} {
3233 global canv linehtag commitrow idpos selectedline
3234
3235 if {![info exists commitrow($id)]} return
3236 drawcmitrow $commitrow($id)
3237 $canv delete tag.$id
3238 set xt [eval drawtags $id $idpos($id)]
3239 $canv coords $linehtag($commitrow($id)) $xt [lindex $idpos($id) 2]
3240 if {[info exists selectedline] && $selectedline == $commitrow($id)} {
3241 selectline $selectedline 0
3242 }
3243}
3244
3245proc mktagcan {} {
3246 global mktagtop
3247
3248 catch {destroy $mktagtop}
3249 unset mktagtop
3250}
3251
3252proc mktaggo {} {
3253 domktag
3254 mktagcan
3255}
3256
3257proc writecommit {} {
3258 global rowmenuid wrcomtop commitinfo wrcomcmd
3259
3260 set top .writecommit
3261 set wrcomtop $top
3262 catch {destroy $top}
3263 toplevel $top
3264 label $top.title -text "Write commit to file"
3265 grid $top.title - -pady 10
3266 label $top.id -text "ID:"
3267 entry $top.sha1 -width 40 -relief flat
3268 $top.sha1 insert 0 $rowmenuid
3269 $top.sha1 conf -state readonly
3270 grid $top.id $top.sha1 -sticky w
3271 entry $top.head -width 60 -relief flat
3272 $top.head insert 0 [lindex $commitinfo($rowmenuid) 0]
3273 $top.head conf -state readonly
3274 grid x $top.head -sticky w
3275 label $top.clab -text "Command:"
3276 entry $top.cmd -width 60 -textvariable wrcomcmd
3277 grid $top.clab $top.cmd -sticky w -pady 10
3278 label $top.flab -text "Output file:"
3279 entry $top.fname -width 60
3280 $top.fname insert 0 [file normalize "commit-[string range $rowmenuid 0 6]"]
3281 grid $top.flab $top.fname -sticky w
3282 frame $top.buts
3283 button $top.buts.gen -text "Write" -command wrcomgo
3284 button $top.buts.can -text "Cancel" -command wrcomcan
3285 grid $top.buts.gen $top.buts.can
3286 grid columnconfigure $top.buts 0 -weight 1 -uniform a
3287 grid columnconfigure $top.buts 1 -weight 1 -uniform a
3288 grid $top.buts - -pady 10 -sticky ew
3289 focus $top.fname
3290}
3291
3292proc wrcomgo {} {
3293 global wrcomtop
3294
3295 set id [$wrcomtop.sha1 get]
3296 set cmd "echo $id | [$wrcomtop.cmd get]"
3297 set fname [$wrcomtop.fname get]
3298 if {[catch {exec sh -c $cmd >$fname &} err]} {
3299 error_popup "Error writing commit: $err"
3300 }
3301 catch {destroy $wrcomtop}
3302 unset wrcomtop
3303}
3304
3305proc wrcomcan {} {
3306 global wrcomtop
3307
3308 catch {destroy $wrcomtop}
3309 unset wrcomtop
3310}
3311
3312proc listrefs {id} {
3313 global idtags idheads idotherrefs
3314
3315 set x {}
3316 if {[info exists idtags($id)]} {
3317 set x $idtags($id)
3318 }
3319 set y {}
3320 if {[info exists idheads($id)]} {
3321 set y $idheads($id)
3322 }
3323 set z {}
3324 if {[info exists idotherrefs($id)]} {
3325 set z $idotherrefs($id)
3326 }
3327 return [list $x $y $z]
3328}
3329
3330proc rereadrefs {} {
3331 global idtags idheads idotherrefs
3332 global tagids headids otherrefids
3333
3334 set refids [concat [array names idtags] \
3335 [array names idheads] [array names idotherrefs]]
3336 foreach id $refids {
3337 if {![info exists ref($id)]} {
3338 set ref($id) [listrefs $id]
3339 }
3340 }
3341 readrefs
3342 set refids [lsort -unique [concat $refids [array names idtags] \
3343 [array names idheads] [array names idotherrefs]]]
3344 foreach id $refids {
3345 set v [listrefs $id]
3346 if {![info exists ref($id)] || $ref($id) != $v} {
3347 redrawtags $id
3348 }
3349 }
3350}
3351
3352proc showtag {tag isnew} {
3353 global ctext cflist tagcontents tagids linknum
3354
3355 if {$isnew} {
3356 addtohistory [list showtag $tag 0]
3357 }
3358 $ctext conf -state normal
3359 $ctext delete 0.0 end
3360 set linknum 0
3361 if {[info exists tagcontents($tag)]} {
3362 set text $tagcontents($tag)
3363 } else {
3364 set text "Tag: $tag\nId: $tagids($tag)"
3365 }
3366 appendwithlinks $text
3367 $ctext conf -state disabled
3368 $cflist delete 0 end
3369}
3370
3371proc doquit {} {
3372 global stopped
3373 set stopped 100
3374 destroy .
3375}
3376
3377proc doprefs {} {
3378 global maxwidth maxgraphpct diffopts findmergefiles
3379 global oldprefs prefstop
3380
3381 set top .gitkprefs
3382 set prefstop $top
3383 if {[winfo exists $top]} {
3384 raise $top
3385 return
3386 }
3387 foreach v {maxwidth maxgraphpct diffopts findmergefiles} {
3388 set oldprefs($v) [set $v]
3389 }
3390 toplevel $top
3391 wm title $top "Gitk preferences"
3392 label $top.ldisp -text "Commit list display options"
3393 grid $top.ldisp - -sticky w -pady 10
3394 label $top.spacer -text " "
3395 label $top.maxwidthl -text "Maximum graph width (lines)" \
3396 -font optionfont
3397 spinbox $top.maxwidth -from 0 -to 100 -width 4 -textvariable maxwidth
3398 grid $top.spacer $top.maxwidthl $top.maxwidth -sticky w
3399 label $top.maxpctl -text "Maximum graph width (% of pane)" \
3400 -font optionfont
3401 spinbox $top.maxpct -from 1 -to 100 -width 4 -textvariable maxgraphpct
3402 grid x $top.maxpctl $top.maxpct -sticky w
3403 checkbutton $top.findm -variable findmergefiles
3404 label $top.findml -text "Include merges for \"Find\" in \"Files\"" \
3405 -font optionfont
3406 grid $top.findm $top.findml - -sticky w
3407 label $top.ddisp -text "Diff display options"
3408 grid $top.ddisp - -sticky w -pady 10
3409 label $top.diffoptl -text "Options for diff program" \
3410 -font optionfont
3411 entry $top.diffopt -width 20 -textvariable diffopts
3412 grid x $top.diffoptl $top.diffopt -sticky w
3413 frame $top.buts
3414 button $top.buts.ok -text "OK" -command prefsok
3415 button $top.buts.can -text "Cancel" -command prefscan
3416 grid $top.buts.ok $top.buts.can
3417 grid columnconfigure $top.buts 0 -weight 1 -uniform a
3418 grid columnconfigure $top.buts 1 -weight 1 -uniform a
3419 grid $top.buts - - -pady 10 -sticky ew
3420}
3421
3422proc prefscan {} {
3423 global maxwidth maxgraphpct diffopts findmergefiles
3424 global oldprefs prefstop
3425
3426 foreach v {maxwidth maxgraphpct diffopts findmergefiles} {
3427 set $v $oldprefs($v)
3428 }
3429 catch {destroy $prefstop}
3430 unset prefstop
3431}
3432
3433proc prefsok {} {
3434 global maxwidth maxgraphpct
3435 global oldprefs prefstop
3436
3437 catch {destroy $prefstop}
3438 unset prefstop
3439 if {$maxwidth != $oldprefs(maxwidth)
3440 || $maxgraphpct != $oldprefs(maxgraphpct)} {
3441 redisplay
3442 }
3443}
3444
3445proc formatdate {d} {
3446 return [clock format $d -format "%Y-%m-%d %H:%M:%S"]
3447}
3448
3449# This list of encoding names and aliases is distilled from
3450# http://www.iana.org/assignments/character-sets.
3451# Not all of them are supported by Tcl.
3452set encoding_aliases {
3453 { ANSI_X3.4-1968 iso-ir-6 ANSI_X3.4-1986 ISO_646.irv:1991 ASCII
3454 ISO646-US US-ASCII us IBM367 cp367 csASCII }
3455 { ISO-10646-UTF-1 csISO10646UTF1 }
3456 { ISO_646.basic:1983 ref csISO646basic1983 }
3457 { INVARIANT csINVARIANT }
3458 { ISO_646.irv:1983 iso-ir-2 irv csISO2IntlRefVersion }
3459 { BS_4730 iso-ir-4 ISO646-GB gb uk csISO4UnitedKingdom }
3460 { NATS-SEFI iso-ir-8-1 csNATSSEFI }
3461 { NATS-SEFI-ADD iso-ir-8-2 csNATSSEFIADD }
3462 { NATS-DANO iso-ir-9-1 csNATSDANO }
3463 { NATS-DANO-ADD iso-ir-9-2 csNATSDANOADD }
3464 { SEN_850200_B iso-ir-10 FI ISO646-FI ISO646-SE se csISO10Swedish }
3465 { SEN_850200_C iso-ir-11 ISO646-SE2 se2 csISO11SwedishForNames }
3466 { KS_C_5601-1987 iso-ir-149 KS_C_5601-1989 KSC_5601 korean csKSC56011987 }
3467 { ISO-2022-KR csISO2022KR }
3468 { EUC-KR csEUCKR }
3469 { ISO-2022-JP csISO2022JP }
3470 { ISO-2022-JP-2 csISO2022JP2 }
3471 { JIS_C6220-1969-jp JIS_C6220-1969 iso-ir-13 katakana x0201-7
3472 csISO13JISC6220jp }
3473 { JIS_C6220-1969-ro iso-ir-14 jp ISO646-JP csISO14JISC6220ro }
3474 { IT iso-ir-15 ISO646-IT csISO15Italian }
3475 { PT iso-ir-16 ISO646-PT csISO16Portuguese }
3476 { ES iso-ir-17 ISO646-ES csISO17Spanish }
3477 { greek7-old iso-ir-18 csISO18Greek7Old }
3478 { latin-greek iso-ir-19 csISO19LatinGreek }
3479 { DIN_66003 iso-ir-21 de ISO646-DE csISO21German }
3480 { NF_Z_62-010_(1973) iso-ir-25 ISO646-FR1 csISO25French }
3481 { Latin-greek-1 iso-ir-27 csISO27LatinGreek1 }
3482 { ISO_5427 iso-ir-37 csISO5427Cyrillic }
3483 { JIS_C6226-1978 iso-ir-42 csISO42JISC62261978 }
3484 { BS_viewdata iso-ir-47 csISO47BSViewdata }
3485 { INIS iso-ir-49 csISO49INIS }
3486 { INIS-8 iso-ir-50 csISO50INIS8 }
3487 { INIS-cyrillic iso-ir-51 csISO51INISCyrillic }
3488 { ISO_5427:1981 iso-ir-54 ISO5427Cyrillic1981 }
3489 { ISO_5428:1980 iso-ir-55 csISO5428Greek }
3490 { GB_1988-80 iso-ir-57 cn ISO646-CN csISO57GB1988 }
3491 { GB_2312-80 iso-ir-58 chinese csISO58GB231280 }
3492 { NS_4551-1 iso-ir-60 ISO646-NO no csISO60DanishNorwegian
3493 csISO60Norwegian1 }
3494 { NS_4551-2 ISO646-NO2 iso-ir-61 no2 csISO61Norwegian2 }
3495 { NF_Z_62-010 iso-ir-69 ISO646-FR fr csISO69French }
3496 { videotex-suppl iso-ir-70 csISO70VideotexSupp1 }
3497 { PT2 iso-ir-84 ISO646-PT2 csISO84Portuguese2 }
3498 { ES2 iso-ir-85 ISO646-ES2 csISO85Spanish2 }
3499 { MSZ_7795.3 iso-ir-86 ISO646-HU hu csISO86Hungarian }
3500 { JIS_C6226-1983 iso-ir-87 x0208 JIS_X0208-1983 csISO87JISX0208 }
3501 { greek7 iso-ir-88 csISO88Greek7 }
3502 { ASMO_449 ISO_9036 arabic7 iso-ir-89 csISO89ASMO449 }
3503 { iso-ir-90 csISO90 }
3504 { JIS_C6229-1984-a iso-ir-91 jp-ocr-a csISO91JISC62291984a }
3505 { JIS_C6229-1984-b iso-ir-92 ISO646-JP-OCR-B jp-ocr-b
3506 csISO92JISC62991984b }
3507 { JIS_C6229-1984-b-add iso-ir-93 jp-ocr-b-add csISO93JIS62291984badd }
3508 { JIS_C6229-1984-hand iso-ir-94 jp-ocr-hand csISO94JIS62291984hand }
3509 { JIS_C6229-1984-hand-add iso-ir-95 jp-ocr-hand-add
3510 csISO95JIS62291984handadd }
3511 { JIS_C6229-1984-kana iso-ir-96 csISO96JISC62291984kana }
3512 { ISO_2033-1983 iso-ir-98 e13b csISO2033 }
3513 { ANSI_X3.110-1983 iso-ir-99 CSA_T500-1983 NAPLPS csISO99NAPLPS }
3514 { ISO_8859-1:1987 iso-ir-100 ISO_8859-1 ISO-8859-1 latin1 l1 IBM819
3515 CP819 csISOLatin1 }
3516 { ISO_8859-2:1987 iso-ir-101 ISO_8859-2 ISO-8859-2 latin2 l2 csISOLatin2 }
3517 { T.61-7bit iso-ir-102 csISO102T617bit }
3518 { T.61-8bit T.61 iso-ir-103 csISO103T618bit }
3519 { ISO_8859-3:1988 iso-ir-109 ISO_8859-3 ISO-8859-3 latin3 l3 csISOLatin3 }
3520 { ISO_8859-4:1988 iso-ir-110 ISO_8859-4 ISO-8859-4 latin4 l4 csISOLatin4 }
3521 { ECMA-cyrillic iso-ir-111 KOI8-E csISO111ECMACyrillic }
3522 { CSA_Z243.4-1985-1 iso-ir-121 ISO646-CA csa7-1 ca csISO121Canadian1 }
3523 { CSA_Z243.4-1985-2 iso-ir-122 ISO646-CA2 csa7-2 csISO122Canadian2 }
3524 { CSA_Z243.4-1985-gr iso-ir-123 csISO123CSAZ24341985gr }
3525 { ISO_8859-6:1987 iso-ir-127 ISO_8859-6 ISO-8859-6 ECMA-114 ASMO-708
3526 arabic csISOLatinArabic }
3527 { ISO_8859-6-E csISO88596E ISO-8859-6-E }
3528 { ISO_8859-6-I csISO88596I ISO-8859-6-I }
3529 { ISO_8859-7:1987 iso-ir-126 ISO_8859-7 ISO-8859-7 ELOT_928 ECMA-118
3530 greek greek8 csISOLatinGreek }
3531 { T.101-G2 iso-ir-128 csISO128T101G2 }
3532 { ISO_8859-8:1988 iso-ir-138 ISO_8859-8 ISO-8859-8 hebrew
3533 csISOLatinHebrew }
3534 { ISO_8859-8-E csISO88598E ISO-8859-8-E }
3535 { ISO_8859-8-I csISO88598I ISO-8859-8-I }
3536 { CSN_369103 iso-ir-139 csISO139CSN369103 }
3537 { JUS_I.B1.002 iso-ir-141 ISO646-YU js yu csISO141JUSIB1002 }
3538 { ISO_6937-2-add iso-ir-142 csISOTextComm }
3539 { IEC_P27-1 iso-ir-143 csISO143IECP271 }
3540 { ISO_8859-5:1988 iso-ir-144 ISO_8859-5 ISO-8859-5 cyrillic
3541 csISOLatinCyrillic }
3542 { JUS_I.B1.003-serb iso-ir-146 serbian csISO146Serbian }
3543 { JUS_I.B1.003-mac macedonian iso-ir-147 csISO147Macedonian }
3544 { ISO_8859-9:1989 iso-ir-148 ISO_8859-9 ISO-8859-9 latin5 l5 csISOLatin5 }
3545 { greek-ccitt iso-ir-150 csISO150 csISO150GreekCCITT }
3546 { NC_NC00-10:81 cuba iso-ir-151 ISO646-CU csISO151Cuba }
3547 { ISO_6937-2-25 iso-ir-152 csISO6937Add }
3548 { GOST_19768-74 ST_SEV_358-88 iso-ir-153 csISO153GOST1976874 }
3549 { ISO_8859-supp iso-ir-154 latin1-2-5 csISO8859Supp }
3550 { ISO_10367-box iso-ir-155 csISO10367Box }
3551 { ISO-8859-10 iso-ir-157 l6 ISO_8859-10:1992 csISOLatin6 latin6 }
3552 { latin-lap lap iso-ir-158 csISO158Lap }
3553 { JIS_X0212-1990 x0212 iso-ir-159 csISO159JISX02121990 }
3554 { DS_2089 DS2089 ISO646-DK dk csISO646Danish }
3555 { us-dk csUSDK }
3556 { dk-us csDKUS }
3557 { JIS_X0201 X0201 csHalfWidthKatakana }
3558 { KSC5636 ISO646-KR csKSC5636 }
3559 { ISO-10646-UCS-2 csUnicode }
3560 { ISO-10646-UCS-4 csUCS4 }
3561 { DEC-MCS dec csDECMCS }
3562 { hp-roman8 roman8 r8 csHPRoman8 }
3563 { macintosh mac csMacintosh }
3564 { IBM037 cp037 ebcdic-cp-us ebcdic-cp-ca ebcdic-cp-wt ebcdic-cp-nl
3565 csIBM037 }
3566 { IBM038 EBCDIC-INT cp038 csIBM038 }
3567 { IBM273 CP273 csIBM273 }
3568 { IBM274 EBCDIC-BE CP274 csIBM274 }
3569 { IBM275 EBCDIC-BR cp275 csIBM275 }
3570 { IBM277 EBCDIC-CP-DK EBCDIC-CP-NO csIBM277 }
3571 { IBM278 CP278 ebcdic-cp-fi ebcdic-cp-se csIBM278 }
3572 { IBM280 CP280 ebcdic-cp-it csIBM280 }
3573 { IBM281 EBCDIC-JP-E cp281 csIBM281 }
3574 { IBM284 CP284 ebcdic-cp-es csIBM284 }
3575 { IBM285 CP285 ebcdic-cp-gb csIBM285 }
3576 { IBM290 cp290 EBCDIC-JP-kana csIBM290 }
3577 { IBM297 cp297 ebcdic-cp-fr csIBM297 }
3578 { IBM420 cp420 ebcdic-cp-ar1 csIBM420 }
3579 { IBM423 cp423 ebcdic-cp-gr csIBM423 }
3580 { IBM424 cp424 ebcdic-cp-he csIBM424 }
3581 { IBM437 cp437 437 csPC8CodePage437 }
3582 { IBM500 CP500 ebcdic-cp-be ebcdic-cp-ch csIBM500 }
3583 { IBM775 cp775 csPC775Baltic }
3584 { IBM850 cp850 850 csPC850Multilingual }
3585 { IBM851 cp851 851 csIBM851 }
3586 { IBM852 cp852 852 csPCp852 }
3587 { IBM855 cp855 855 csIBM855 }
3588 { IBM857 cp857 857 csIBM857 }
3589 { IBM860 cp860 860 csIBM860 }
3590 { IBM861 cp861 861 cp-is csIBM861 }
3591 { IBM862 cp862 862 csPC862LatinHebrew }
3592 { IBM863 cp863 863 csIBM863 }
3593 { IBM864 cp864 csIBM864 }
3594 { IBM865 cp865 865 csIBM865 }
3595 { IBM866 cp866 866 csIBM866 }
3596 { IBM868 CP868 cp-ar csIBM868 }
3597 { IBM869 cp869 869 cp-gr csIBM869 }
3598 { IBM870 CP870 ebcdic-cp-roece ebcdic-cp-yu csIBM870 }
3599 { IBM871 CP871 ebcdic-cp-is csIBM871 }
3600 { IBM880 cp880 EBCDIC-Cyrillic csIBM880 }
3601 { IBM891 cp891 csIBM891 }
3602 { IBM903 cp903 csIBM903 }
3603 { IBM904 cp904 904 csIBBM904 }
3604 { IBM905 CP905 ebcdic-cp-tr csIBM905 }
3605 { IBM918 CP918 ebcdic-cp-ar2 csIBM918 }
3606 { IBM1026 CP1026 csIBM1026 }
3607 { EBCDIC-AT-DE csIBMEBCDICATDE }
3608 { EBCDIC-AT-DE-A csEBCDICATDEA }
3609 { EBCDIC-CA-FR csEBCDICCAFR }
3610 { EBCDIC-DK-NO csEBCDICDKNO }
3611 { EBCDIC-DK-NO-A csEBCDICDKNOA }
3612 { EBCDIC-FI-SE csEBCDICFISE }
3613 { EBCDIC-FI-SE-A csEBCDICFISEA }
3614 { EBCDIC-FR csEBCDICFR }
3615 { EBCDIC-IT csEBCDICIT }
3616 { EBCDIC-PT csEBCDICPT }
3617 { EBCDIC-ES csEBCDICES }
3618 { EBCDIC-ES-A csEBCDICESA }
3619 { EBCDIC-ES-S csEBCDICESS }
3620 { EBCDIC-UK csEBCDICUK }
3621 { EBCDIC-US csEBCDICUS }
3622 { UNKNOWN-8BIT csUnknown8BiT }
3623 { MNEMONIC csMnemonic }
3624 { MNEM csMnem }
3625 { VISCII csVISCII }
3626 { VIQR csVIQR }
3627 { KOI8-R csKOI8R }
3628 { IBM00858 CCSID00858 CP00858 PC-Multilingual-850+euro }
3629 { IBM00924 CCSID00924 CP00924 ebcdic-Latin9--euro }
3630 { IBM01140 CCSID01140 CP01140 ebcdic-us-37+euro }
3631 { IBM01141 CCSID01141 CP01141 ebcdic-de-273+euro }
3632 { IBM01142 CCSID01142 CP01142 ebcdic-dk-277+euro ebcdic-no-277+euro }
3633 { IBM01143 CCSID01143 CP01143 ebcdic-fi-278+euro ebcdic-se-278+euro }
3634 { IBM01144 CCSID01144 CP01144 ebcdic-it-280+euro }
3635 { IBM01145 CCSID01145 CP01145 ebcdic-es-284+euro }
3636 { IBM01146 CCSID01146 CP01146 ebcdic-gb-285+euro }
3637 { IBM01147 CCSID01147 CP01147 ebcdic-fr-297+euro }
3638 { IBM01148 CCSID01148 CP01148 ebcdic-international-500+euro }
3639 { IBM01149 CCSID01149 CP01149 ebcdic-is-871+euro }
3640 { IBM1047 IBM-1047 }
3641 { PTCP154 csPTCP154 PT154 CP154 Cyrillic-Asian }
3642 { Amiga-1251 Ami1251 Amiga1251 Ami-1251 }
3643 { UNICODE-1-1 csUnicode11 }
3644 { CESU-8 csCESU-8 }
3645 { BOCU-1 csBOCU-1 }
3646 { UNICODE-1-1-UTF-7 csUnicode11UTF7 }
3647 { ISO-8859-14 iso-ir-199 ISO_8859-14:1998 ISO_8859-14 latin8 iso-celtic
3648 l8 }
3649 { ISO-8859-15 ISO_8859-15 Latin-9 }
3650 { ISO-8859-16 iso-ir-226 ISO_8859-16:2001 ISO_8859-16 latin10 l10 }
3651 { GBK CP936 MS936 windows-936 }
3652 { JIS_Encoding csJISEncoding }
3653 { Shift_JIS MS_Kanji csShiftJIS }
3654 { Extended_UNIX_Code_Packed_Format_for_Japanese csEUCPkdFmtJapanese
3655 EUC-JP }
3656 { Extended_UNIX_Code_Fixed_Width_for_Japanese csEUCFixWidJapanese }
3657 { ISO-10646-UCS-Basic csUnicodeASCII }
3658 { ISO-10646-Unicode-Latin1 csUnicodeLatin1 ISO-10646 }
3659 { ISO-Unicode-IBM-1261 csUnicodeIBM1261 }
3660 { ISO-Unicode-IBM-1268 csUnicodeIBM1268 }
3661 { ISO-Unicode-IBM-1276 csUnicodeIBM1276 }
3662 { ISO-Unicode-IBM-1264 csUnicodeIBM1264 }
3663 { ISO-Unicode-IBM-1265 csUnicodeIBM1265 }
3664 { ISO-8859-1-Windows-3.0-Latin-1 csWindows30Latin1 }
3665 { ISO-8859-1-Windows-3.1-Latin-1 csWindows31Latin1 }
3666 { ISO-8859-2-Windows-Latin-2 csWindows31Latin2 }
3667 { ISO-8859-9-Windows-Latin-5 csWindows31Latin5 }
3668 { Adobe-Standard-Encoding csAdobeStandardEncoding }
3669 { Ventura-US csVenturaUS }
3670 { Ventura-International csVenturaInternational }
3671 { PC8-Danish-Norwegian csPC8DanishNorwegian }
3672 { PC8-Turkish csPC8Turkish }
3673 { IBM-Symbols csIBMSymbols }
3674 { IBM-Thai csIBMThai }
3675 { HP-Legal csHPLegal }
3676 { HP-Pi-font csHPPiFont }
3677 { HP-Math8 csHPMath8 }
3678 { Adobe-Symbol-Encoding csHPPSMath }
3679 { HP-DeskTop csHPDesktop }
3680 { Ventura-Math csVenturaMath }
3681 { Microsoft-Publishing csMicrosoftPublishing }
3682 { Windows-31J csWindows31J }
3683 { GB2312 csGB2312 }
3684 { Big5 csBig5 }
3685}
3686
3687proc tcl_encoding {enc} {
3688 global encoding_aliases
3689 set names [encoding names]
3690 set lcnames [string tolower $names]
3691 set enc [string tolower $enc]
3692 set i [lsearch -exact $lcnames $enc]
3693 if {$i < 0} {
3694 # look for "isonnn" instead of "iso-nnn" or "iso_nnn"
3695 if {[regsub {^iso[-_]} $enc iso encx]} {
3696 set i [lsearch -exact $lcnames $encx]
3697 }
3698 }
3699 if {$i < 0} {
3700 foreach l $encoding_aliases {
3701 set ll [string tolower $l]
3702 if {[lsearch -exact $ll $enc] < 0} continue
3703 # look through the aliases for one that tcl knows about
3704 foreach e $ll {
3705 set i [lsearch -exact $lcnames $e]
3706 if {$i < 0} {
3707 if {[regsub {^iso[-_]} $e iso ex]} {
3708 set i [lsearch -exact $lcnames $ex]
3709 }
3710 }
3711 if {$i >= 0} break
3712 }
3713 break
3714 }
3715 }
3716 if {$i >= 0} {
3717 return [lindex $names $i]
3718 }
3719 return {}
3720}
3721
3722# defaults...
3723set datemode 0
3724set diffopts "-U 5 -p"
3725set wrcomcmd "git-diff-tree --stdin -p --pretty"
3726
3727set gitencoding {}
3728catch {
3729 set gitencoding [exec git-repo-config --get i18n.commitencoding]
3730}
3731if {$gitencoding == ""} {
3732 set gitencoding "utf-8"
3733}
3734set tclencoding [tcl_encoding $gitencoding]
3735if {$tclencoding == {}} {
3736 puts stderr "Warning: encoding $gitencoding is not supported by Tcl/Tk"
3737}
3738
3739set mainfont {Helvetica 9}
3740set textfont {Courier 9}
3741set findmergefiles 0
3742set maxgraphpct 50
3743set maxwidth 16
3744set revlistorder 0
3745set fastdate 0
3746set uparrowlen 7
3747set downarrowlen 7
3748set mingaplen 30
3749
3750set colors {green red blue magenta darkgrey brown orange}
3751
3752catch {source ~/.gitk}
3753
3754set namefont $mainfont
3755
3756font create optionfont -family sans-serif -size -12
3757
3758set revtreeargs {}
3759foreach arg $argv {
3760 switch -regexp -- $arg {
3761 "^$" { }
3762 "^-d" { set datemode 1 }
3763 default {
3764 lappend revtreeargs $arg
3765 }
3766 }
3767}
3768
3769# check that we can find a .git directory somewhere...
3770set gitdir [gitdir]
3771if {![file isdirectory $gitdir]} {
3772 error_popup "Cannot find the git directory \"$gitdir\"."
3773 exit 1
3774}
3775
3776set history {}
3777set historyindex 0
3778
3779set optim_delay 16
3780
3781set stopped 0
3782set stuffsaved 0
3783set patchnum 0
3784setcoords
3785makewindow $revtreeargs
3786readrefs
3787getcommits $revtreeargs