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