Merge updated git-gui and gitk that call each other
authorJunio C Hamano <gitster@pobox.com>
Sun, 7 Sep 2008 01:06:39 +0000 (18:06 -0700)
committerJunio C Hamano <gitster@pobox.com>
Sun, 7 Sep 2008 01:06:39 +0000 (18:06 -0700)
Merge git://repo.or.cz/git-gui and git://git.kernel.org/pub/scm/gitk/gitk

* git://repo.or.cz/git-gui:
git-gui: Show special diffs for complex conflict cases.
git-gui: Make F5 reselect a diff, if an untracked file is selected.
git-gui: Reimplement and enhance auto-selection of diffs.
git-gui: Support conflict states _U & UT.
git-gui: Support more merge tools.
git-gui: Don't allow staging files with conflicts.
git-gui: Support calling merge tools.
git-gui: Support resolving conflicts via the diff context menu.
git-gui: Mark forgotten strings for translation.
git-gui: Allow specifying an initial line for git gui blame.
git-gui: Better positioning in Blame Parent Commit
git-gui: Support passing blame to a parent commit.
git-gui: Support starting gitk from Gui Blame
git-gui: Teach git gui about file type changes

* git://git.kernel.org/pub/scm/gitk/gitk:
gitk: Add menu item for calling git gui blame
gitk: Add option to specify the default commit on command line

1  2  3 
git-gui/git-gui.sh
git-gui/lib/blame.tcl
git-gui/lib/browser.tcl
git-gui/lib/commit.tcl
git-gui/lib/diff.tcl
git-gui/lib/index.tcl
git-gui/lib/mergetool.tcl
git-gui/lib/option.tcl
gitk-git/gitk
index 86402d49f72c56d793f2f1958a9e3d3f8300f367,0000000000000000000000000000000000000000,0000000000000000000000000000000000000000..10d8a4422f151eda78a24ff533eb20c3d775d099
mode 100755,000000,000000..100755
--- /dev/null
--- /dev/null
@@@@ -1,3028 -1,0 -1,0 +1,3197 @@@@
-       show_diff $next_diff_p $next_diff_w $next_diff_i
 ++#!/bin/sh
 ++# Tcl ignores the next line -*- tcl -*- \
 ++ if test "z$*" = zversion \
 ++ || test "z$*" = z--version; \
 ++ then \
 ++     echo 'git-gui version @@GITGUI_VERSION@@'; \
 ++     exit; \
 ++ fi; \
 ++ argv0=$0; \
 ++ exec wish "$argv0" -- "$@"
 ++
 ++set appvers {@@GITGUI_VERSION@@}
 ++set copyright [encoding convertfrom utf-8 {
 ++Copyright © 2006, 2007 Shawn Pearce, et. al.
 ++
 ++This program is free software; you can redistribute it and/or modify
 ++it under the terms of the GNU General Public License as published by
 ++the Free Software Foundation; either version 2 of the License, or
 ++(at your option) any later version.
 ++
 ++This program is distributed in the hope that it will be useful,
 ++but WITHOUT ANY WARRANTY; without even the implied warranty of
 ++MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 ++GNU General Public License for more details.
 ++
 ++You should have received a copy of the GNU General Public License
 ++along with this program; if not, write to the Free Software
 ++Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA}]
 ++
 ++######################################################################
 ++##
 ++## Tcl/Tk sanity check
 ++
 ++if {[catch {package require Tcl 8.4} err]
 ++ || [catch {package require Tk  8.4} err]
 ++} {
 ++     catch {wm withdraw .}
 ++     tk_messageBox \
 ++             -icon error \
 ++             -type ok \
 ++             -title [mc "git-gui: fatal error"] \
 ++             -message $err
 ++     exit 1
 ++}
 ++
 ++catch {rename send {}} ; # What an evil concept...
 ++
 ++######################################################################
 ++##
 ++## locate our library
 ++
 ++set oguilib {@@GITGUI_LIBDIR@@}
 ++set oguirel {@@GITGUI_RELATIVE@@}
 ++if {$oguirel eq {1}} {
 ++     set oguilib [file dirname [file normalize $argv0]]
 ++     if {[file tail $oguilib] eq {git-core}} {
 ++             set oguilib [file dirname $oguilib]
 ++     }
 ++     set oguilib [file dirname $oguilib]
 ++     set oguilib [file join $oguilib share git-gui lib]
 ++     set oguimsg [file join $oguilib msgs]
 ++} elseif {[string match @@* $oguirel]} {
 ++     set oguilib [file join [file dirname [file normalize $argv0]] lib]
 ++     set oguimsg [file join [file dirname [file normalize $argv0]] po]
 ++} else {
 ++     set oguimsg [file join $oguilib msgs]
 ++}
 ++unset oguirel
 ++
 ++######################################################################
 ++##
 ++## enable verbose loading?
 ++
 ++if {![catch {set _verbose $env(GITGUI_VERBOSE)}]} {
 ++     unset _verbose
 ++     rename auto_load real__auto_load
 ++     proc auto_load {name args} {
 ++             puts stderr "auto_load $name"
 ++             return [uplevel 1 real__auto_load $name $args]
 ++     }
 ++     rename source real__source
 ++     proc source {name} {
 ++             puts stderr "source    $name"
 ++             uplevel 1 real__source $name
 ++     }
 ++}
 ++
 ++######################################################################
 ++##
 ++## Internationalization (i18n) through msgcat and gettext. See
 ++## http://www.gnu.org/software/gettext/manual/html_node/Tcl.html
 ++
 ++package require msgcat
 ++
 ++proc _mc_trim {fmt} {
 ++     set cmk [string first @@ $fmt]
 ++     if {$cmk > 0} {
 ++             return [string range $fmt 0 [expr {$cmk - 1}]]
 ++     }
 ++     return $fmt
 ++}
 ++
 ++proc mc {en_fmt args} {
 ++     set fmt [_mc_trim [::msgcat::mc $en_fmt]]
 ++     if {[catch {set msg [eval [list format $fmt] $args]} err]} {
 ++             set msg [eval [list format [_mc_trim $en_fmt]] $args]
 ++     }
 ++     return $msg
 ++}
 ++
 ++proc strcat {args} {
 ++     return [join $args {}]
 ++}
 ++
 ++::msgcat::mcload $oguimsg
 ++unset oguimsg
 ++
 ++######################################################################
 ++##
 ++## read only globals
 ++
 ++set _appname {Git Gui}
 ++set _gitdir {}
 ++set _gitexec {}
 ++set _reponame {}
 ++set _iscygwin {}
 ++set _search_path {}
 ++
 ++set _trace [lsearch -exact $argv --trace]
 ++if {$_trace >= 0} {
 ++     set argv [lreplace $argv $_trace $_trace]
 ++     set _trace 1
 ++} else {
 ++     set _trace 0
 ++}
 ++
 ++proc appname {} {
 ++     global _appname
 ++     return $_appname
 ++}
 ++
 ++proc gitdir {args} {
 ++     global _gitdir
 ++     if {$args eq {}} {
 ++             return $_gitdir
 ++     }
 ++     return [eval [list file join $_gitdir] $args]
 ++}
 ++
 ++proc gitexec {args} {
 ++     global _gitexec
 ++     if {$_gitexec eq {}} {
 ++             if {[catch {set _gitexec [git --exec-path]} err]} {
 ++                     error "Git not installed?\n\n$err"
 ++             }
 ++             if {[is_Cygwin]} {
 ++                     set _gitexec [exec cygpath \
 ++                             --windows \
 ++                             --absolute \
 ++                             $_gitexec]
 ++             } else {
 ++                     set _gitexec [file normalize $_gitexec]
 ++             }
 ++     }
 ++     if {$args eq {}} {
 ++             return $_gitexec
 ++     }
 ++     return [eval [list file join $_gitexec] $args]
 ++}
 ++
 ++proc reponame {} {
 ++     return $::_reponame
 ++}
 ++
 ++proc is_MacOSX {} {
 ++     if {[tk windowingsystem] eq {aqua}} {
 ++             return 1
 ++     }
 ++     return 0
 ++}
 ++
 ++proc is_Windows {} {
 ++     if {$::tcl_platform(platform) eq {windows}} {
 ++             return 1
 ++     }
 ++     return 0
 ++}
 ++
 ++proc is_Cygwin {} {
 ++     global _iscygwin
 ++     if {$_iscygwin eq {}} {
 ++             if {$::tcl_platform(platform) eq {windows}} {
 ++                     if {[catch {set p [exec cygpath --windir]} err]} {
 ++                             set _iscygwin 0
 ++                     } else {
 ++                             set _iscygwin 1
 ++                     }
 ++             } else {
 ++                     set _iscygwin 0
 ++             }
 ++     }
 ++     return $_iscygwin
 ++}
 ++
 ++proc is_enabled {option} {
 ++     global enabled_options
 ++     if {[catch {set on $enabled_options($option)}]} {return 0}
 ++     return $on
 ++}
 ++
 ++proc enable_option {option} {
 ++     global enabled_options
 ++     set enabled_options($option) 1
 ++}
 ++
 ++proc disable_option {option} {
 ++     global enabled_options
 ++     set enabled_options($option) 0
 ++}
 ++
 ++######################################################################
 ++##
 ++## config
 ++
 ++proc is_many_config {name} {
 ++     switch -glob -- $name {
 ++     gui.recentrepo -
 ++     remote.*.fetch -
 ++     remote.*.push
 ++             {return 1}
 ++     *
 ++             {return 0}
 ++     }
 ++}
 ++
 ++proc is_config_true {name} {
 ++     global repo_config
 ++     if {[catch {set v $repo_config($name)}]} {
 ++             return 0
 ++     } elseif {$v eq {true} || $v eq {1} || $v eq {yes}} {
 ++             return 1
 ++     } else {
 ++             return 0
 ++     }
 ++}
 ++
 ++proc get_config {name} {
 ++     global repo_config
 ++     if {[catch {set v $repo_config($name)}]} {
 ++             return {}
 ++     } else {
 ++             return $v
 ++     }
 ++}
 ++
 ++######################################################################
 ++##
 ++## handy utils
 ++
 ++proc _trace_exec {cmd} {
 ++     if {!$::_trace} return
 ++     set d {}
 ++     foreach v $cmd {
 ++             if {$d ne {}} {
 ++                     append d { }
 ++             }
 ++             if {[regexp {[ \t\r\n'"$?*]} $v]} {
 ++                     set v [sq $v]
 ++             }
 ++             append d $v
 ++     }
 ++     puts stderr $d
 ++}
 ++
 ++proc _git_cmd {name} {
 ++     global _git_cmd_path
 ++
 ++     if {[catch {set v $_git_cmd_path($name)}]} {
 ++             switch -- $name {
 ++               version   -
 ++             --version   -
 ++             --exec-path { return [list $::_git $name] }
 ++             }
 ++
 ++             set p [gitexec git-$name$::_search_exe]
 ++             if {[file exists $p]} {
 ++                     set v [list $p]
 ++             } elseif {[is_Windows] && [file exists [gitexec git-$name]]} {
 ++                     # Try to determine what sort of magic will make
 ++                     # git-$name go and do its thing, because native
 ++                     # Tcl on Windows doesn't know it.
 ++                     #
 ++                     set p [gitexec git-$name]
 ++                     set f [open $p r]
 ++                     set s [gets $f]
 ++                     close $f
 ++
 ++                     switch -glob -- [lindex $s 0] {
 ++                     #!*sh     { set i sh     }
 ++                     #!*perl   { set i perl   }
 ++                     #!*python { set i python }
 ++                     default   { error "git-$name is not supported: $s" }
 ++                     }
 ++
 ++                     upvar #0 _$i interp
 ++                     if {![info exists interp]} {
 ++                             set interp [_which $i]
 ++                     }
 ++                     if {$interp eq {}} {
 ++                             error "git-$name requires $i (not in PATH)"
 ++                     }
 ++                     set v [concat [list $interp] [lrange $s 1 end] [list $p]]
 ++             } else {
 ++                     # Assume it is builtin to git somehow and we
 ++                     # aren't actually able to see a file for it.
 ++                     #
 ++                     set v [list $::_git $name]
 ++             }
 ++             set _git_cmd_path($name) $v
 ++     }
 ++     return $v
 ++}
 ++
 ++proc _which {what args} {
 ++     global env _search_exe _search_path
 ++
 ++     if {$_search_path eq {}} {
 ++             if {[is_Cygwin] && [regexp {^(/|\.:)} $env(PATH)]} {
 ++                     set _search_path [split [exec cygpath \
 ++                             --windows \
 ++                             --path \
 ++                             --absolute \
 ++                             $env(PATH)] {;}]
 ++                     set _search_exe .exe
 ++             } elseif {[is_Windows]} {
 ++                     set gitguidir [file dirname [info script]]
 ++                     regsub -all ";" $gitguidir "\\;" gitguidir
 ++                     set env(PATH) "$gitguidir;$env(PATH)"
 ++                     set _search_path [split $env(PATH) {;}]
 ++                     set _search_exe .exe
 ++             } else {
 ++                     set _search_path [split $env(PATH) :]
 ++                     set _search_exe {}
 ++             }
 ++     }
 ++
 ++     if {[is_Windows] && [lsearch -exact $args -script] >= 0} {
 ++             set suffix {}
 ++     } else {
 ++             set suffix $_search_exe
 ++     }
 ++
 ++     foreach p $_search_path {
 ++             set p [file join $p $what$suffix]
 ++             if {[file exists $p]} {
 ++                     return [file normalize $p]
 ++             }
 ++     }
 ++     return {}
 ++}
 ++
 ++proc _lappend_nice {cmd_var} {
 ++     global _nice
 ++     upvar $cmd_var cmd
 ++
 ++     if {![info exists _nice]} {
 ++             set _nice [_which nice]
 ++     }
 ++     if {$_nice ne {}} {
 ++             lappend cmd $_nice
 ++     }
 ++}
 ++
 ++proc git {args} {
 ++     set opt [list]
 ++
 ++     while {1} {
 ++             switch -- [lindex $args 0] {
 ++             --nice {
 ++                     _lappend_nice opt
 ++             }
 ++
 ++             default {
 ++                     break
 ++             }
 ++
 ++             }
 ++
 ++             set args [lrange $args 1 end]
 ++     }
 ++
 ++     set cmdp [_git_cmd [lindex $args 0]]
 ++     set args [lrange $args 1 end]
 ++
 ++     _trace_exec [concat $opt $cmdp $args]
 ++     set result [eval exec $opt $cmdp $args]
 ++     if {$::_trace} {
 ++             puts stderr "< $result"
 ++     }
 ++     return $result
 ++}
 ++
 ++proc _open_stdout_stderr {cmd} {
 ++     _trace_exec $cmd
 ++     if {[catch {
 ++                     set fd [open [concat [list | ] $cmd] r]
 ++             } err]} {
 ++             if {   [lindex $cmd end] eq {2>@1}
 ++                 && $err eq {can not find channel named "1"}
 ++                     } {
 ++                     # Older versions of Tcl 8.4 don't have this 2>@1 IO
 ++                     # redirect operator.  Fallback to |& cat for those.
 ++                     # The command was not actually started, so its safe
 ++                     # to try to start it a second time.
 ++                     #
 ++                     set fd [open [concat \
 ++                             [list | ] \
 ++                             [lrange $cmd 0 end-1] \
 ++                             [list |& cat] \
 ++                             ] r]
 ++             } else {
 ++                     error $err
 ++             }
 ++     }
 ++     fconfigure $fd -eofchar {}
 ++     return $fd
 ++}
 ++
 ++proc git_read {args} {
 ++     set opt [list]
 ++
 ++     while {1} {
 ++             switch -- [lindex $args 0] {
 ++             --nice {
 ++                     _lappend_nice opt
 ++             }
 ++
 ++             --stderr {
 ++                     lappend args 2>@1
 ++             }
 ++
 ++             default {
 ++                     break
 ++             }
 ++
 ++             }
 ++
 ++             set args [lrange $args 1 end]
 ++     }
 ++
 ++     set cmdp [_git_cmd [lindex $args 0]]
 ++     set args [lrange $args 1 end]
 ++
 ++     return [_open_stdout_stderr [concat $opt $cmdp $args]]
 ++}
 ++
 ++proc git_write {args} {
 ++     set opt [list]
 ++
 ++     while {1} {
 ++             switch -- [lindex $args 0] {
 ++             --nice {
 ++                     _lappend_nice opt
 ++             }
 ++
 ++             default {
 ++                     break
 ++             }
 ++
 ++             }
 ++
 ++             set args [lrange $args 1 end]
 ++     }
 ++
 ++     set cmdp [_git_cmd [lindex $args 0]]
 ++     set args [lrange $args 1 end]
 ++
 ++     _trace_exec [concat $opt $cmdp $args]
 ++     return [open [concat [list | ] $opt $cmdp $args] w]
 ++}
 ++
 ++proc githook_read {hook_name args} {
 ++     set pchook [gitdir hooks $hook_name]
 ++     lappend args 2>@1
 ++
 ++     # On Windows [file executable] might lie so we need to ask
 ++     # the shell if the hook is executable.  Yes that's annoying.
 ++     #
 ++     if {[is_Windows]} {
 ++             upvar #0 _sh interp
 ++             if {![info exists interp]} {
 ++                     set interp [_which sh]
 ++             }
 ++             if {$interp eq {}} {
 ++                     error "hook execution requires sh (not in PATH)"
 ++             }
 ++
 ++             set scr {if test -x "$1";then exec "$@";fi}
 ++             set sh_c [list $interp -c $scr $interp $pchook]
 ++             return [_open_stdout_stderr [concat $sh_c $args]]
 ++     }
 ++
 ++     if {[file executable $pchook]} {
 ++             return [_open_stdout_stderr [concat [list $pchook] $args]]
 ++     }
 ++
 ++     return {}
 ++}
 ++
 ++proc kill_file_process {fd} {
 ++     set process [pid $fd]
 ++
 ++     catch {
 ++             if {[is_Windows]} {
 ++                     # Use a Cygwin-specific flag to allow killing
 ++                     # native Windows processes
 ++                     exec kill -f $process
 ++             } else {
 ++                     exec kill $process
 ++             }
 ++     }
 ++}
 ++
 ++proc sq {value} {
 ++     regsub -all ' $value "'\\''" value
 ++     return "'$value'"
 ++}
 ++
 ++proc load_current_branch {} {
 ++     global current_branch is_detached
 ++
 ++     set fd [open [gitdir HEAD] r]
 ++     if {[gets $fd ref] < 1} {
 ++             set ref {}
 ++     }
 ++     close $fd
 ++
 ++     set pfx {ref: refs/heads/}
 ++     set len [string length $pfx]
 ++     if {[string equal -length $len $pfx $ref]} {
 ++             # We're on a branch.  It might not exist.  But
 ++             # HEAD looks good enough to be a branch.
 ++             #
 ++             set current_branch [string range $ref $len end]
 ++             set is_detached 0
 ++     } else {
 ++             # Assume this is a detached head.
 ++             #
 ++             set current_branch HEAD
 ++             set is_detached 1
 ++     }
 ++}
 ++
 ++auto_load tk_optionMenu
 ++rename tk_optionMenu real__tkOptionMenu
 ++proc tk_optionMenu {w varName args} {
 ++     set m [eval real__tkOptionMenu $w $varName $args]
 ++     $m configure -font font_ui
 ++     $w configure -font font_ui
 ++     return $m
 ++}
 ++
 ++proc rmsel_tag {text} {
 ++     $text tag conf sel \
 ++             -background [$text cget -background] \
 ++             -foreground [$text cget -foreground] \
 ++             -borderwidth 0
 ++     $text tag conf in_sel -background lightgray
 ++     bind $text <Motion> break
 ++     return $text
 ++}
 ++
 ++set root_exists 0
 ++bind . <Visibility> {
 ++     bind . <Visibility> {}
 ++     set root_exists 1
 ++}
 ++
 ++if {[is_Windows]} {
 ++     wm iconbitmap . -default $oguilib/git-gui.ico
 ++}
 ++
 ++######################################################################
 ++##
 ++## config defaults
 ++
 ++set cursor_ptr arrow
 ++font create font_diff -family Courier -size 10
 ++font create font_ui
 ++catch {
 ++     label .dummy
 ++     eval font configure font_ui [font actual [.dummy cget -font]]
 ++     destroy .dummy
 ++}
 ++
 ++font create font_uiitalic
 ++font create font_uibold
 ++font create font_diffbold
 ++font create font_diffitalic
 ++
 ++foreach class {Button Checkbutton Entry Label
 ++             Labelframe Listbox Menu Message
 ++             Radiobutton Spinbox Text} {
 ++     option add *$class.font font_ui
 ++}
 ++unset class
 ++
 ++if {[is_Windows] || [is_MacOSX]} {
 ++     option add *Menu.tearOff 0
 ++}
 ++
 ++if {[is_MacOSX]} {
 ++     set M1B M1
 ++     set M1T Cmd
 ++} else {
 ++     set M1B Control
 ++     set M1T Ctrl
 ++}
 ++
 ++proc bind_button3 {w cmd} {
 ++     bind $w <Any-Button-3> $cmd
 ++     if {[is_MacOSX]} {
 ++             # Mac OS X sends Button-2 on right click through three-button mouse,
 ++             # or through trackpad right-clicking (two-finger touch + click).
 ++             bind $w <Any-Button-2> $cmd
 ++             bind $w <Control-Button-1> $cmd
 ++     }
 ++}
 ++
 ++proc apply_config {} {
 ++     global repo_config font_descs
 ++
 ++     foreach option $font_descs {
 ++             set name [lindex $option 0]
 ++             set font [lindex $option 1]
 ++             if {[catch {
 ++                     set need_weight 1
 ++                     foreach {cn cv} $repo_config(gui.$name) {
 ++                             if {$cn eq {-weight}} {
 ++                                     set need_weight 0
 ++                             }
 ++                             font configure $font $cn $cv
 ++                     }
 ++                     if {$need_weight} {
 ++                             font configure $font -weight normal
 ++                     }
 ++                     } err]} {
 ++                     error_popup [strcat [mc "Invalid font specified in %s:" "gui.$name"] "\n\n$err"]
 ++             }
 ++             foreach {cn cv} [font configure $font] {
 ++                     font configure ${font}bold $cn $cv
 ++                     font configure ${font}italic $cn $cv
 ++             }
 ++             font configure ${font}bold -weight bold
 ++             font configure ${font}italic -slant italic
 ++     }
 ++}
 ++
 ++set default_config(branch.autosetupmerge) true
+++set default_config(merge.tool) {}
+++set default_config(merge.keepbackup) true
 ++set default_config(merge.diffstat) true
 ++set default_config(merge.summary) false
 ++set default_config(merge.verbosity) 2
 ++set default_config(user.name) {}
 ++set default_config(user.email) {}
 ++
 ++set default_config(gui.matchtrackingbranch) false
 ++set default_config(gui.pruneduringfetch) false
 ++set default_config(gui.trustmtime) false
 ++set default_config(gui.fastcopyblame) false
 ++set default_config(gui.copyblamethreshold) 40
+++set default_config(gui.blamehistoryctx) 7
 ++set default_config(gui.diffcontext) 5
 ++set default_config(gui.commitmsgwidth) 75
 ++set default_config(gui.newbranchtemplate) {}
 ++set default_config(gui.spellingdictionary) {}
 ++set default_config(gui.fontui) [font configure font_ui]
 ++set default_config(gui.fontdiff) [font configure font_diff]
 ++set font_descs {
 ++     {fontui   font_ui   {mc "Main Font"}}
 ++     {fontdiff font_diff {mc "Diff/Console Font"}}
 ++}
 ++
 ++######################################################################
 ++##
 ++## find git
 ++
 ++set _git  [_which git]
 ++if {$_git eq {}} {
 ++     catch {wm withdraw .}
 ++     tk_messageBox \
 ++             -icon error \
 ++             -type ok \
 ++             -title [mc "git-gui: fatal error"] \
 ++             -message [mc "Cannot find git in PATH."]
 ++     exit 1
 ++}
 ++
 ++######################################################################
 ++##
 ++## version check
 ++
 ++if {[catch {set _git_version [git --version]} err]} {
 ++     catch {wm withdraw .}
 ++     tk_messageBox \
 ++             -icon error \
 ++             -type ok \
 ++             -title [mc "git-gui: fatal error"] \
 ++             -message "Cannot determine Git version:
 ++
 ++$err
 ++
 ++[appname] requires Git 1.5.0 or later."
 ++     exit 1
 ++}
 ++if {![regsub {^git version } $_git_version {} _git_version]} {
 ++     catch {wm withdraw .}
 ++     tk_messageBox \
 ++             -icon error \
 ++             -type ok \
 ++             -title [mc "git-gui: fatal error"] \
 ++             -message [strcat [mc "Cannot parse Git version string:"] "\n\n$_git_version"]
 ++     exit 1
 ++}
 ++
 ++set _real_git_version $_git_version
 ++regsub -- {[\-\.]dirty$} $_git_version {} _git_version
 ++regsub {\.[0-9]+\.g[0-9a-f]+$} $_git_version {} _git_version
 ++regsub {\.rc[0-9]+$} $_git_version {} _git_version
 ++regsub {\.GIT$} $_git_version {} _git_version
 ++regsub {\.[a-zA-Z]+\.[0-9]+$} $_git_version {} _git_version
 ++
 ++if {![regexp {^[1-9]+(\.[0-9]+)+$} $_git_version]} {
 ++     catch {wm withdraw .}
 ++     if {[tk_messageBox \
 ++             -icon warning \
 ++             -type yesno \
 ++             -default no \
 ++             -title "[appname]: warning" \
 ++              -message [mc "Git version cannot be determined.
 ++
 ++%s claims it is version '%s'.
 ++
 ++%s requires at least Git 1.5.0 or later.
 ++
 ++Assume '%s' is version 1.5.0?
 ++" $_git $_real_git_version [appname] $_real_git_version]] eq {yes}} {
 ++             set _git_version 1.5.0
 ++     } else {
 ++             exit 1
 ++     }
 ++}
 ++unset _real_git_version
 ++
 ++proc git-version {args} {
 ++     global _git_version
 ++
 ++     switch [llength $args] {
 ++     0 {
 ++             return $_git_version
 ++     }
 ++
 ++     2 {
 ++             set op [lindex $args 0]
 ++             set vr [lindex $args 1]
 ++             set cm [package vcompare $_git_version $vr]
 ++             return [expr $cm $op 0]
 ++     }
 ++
 ++     4 {
 ++             set type [lindex $args 0]
 ++             set name [lindex $args 1]
 ++             set parm [lindex $args 2]
 ++             set body [lindex $args 3]
 ++
 ++             if {($type ne {proc} && $type ne {method})} {
 ++                     error "Invalid arguments to git-version"
 ++             }
 ++             if {[llength $body] < 2 || [lindex $body end-1] ne {default}} {
 ++                     error "Last arm of $type $name must be default"
 ++             }
 ++
 ++             foreach {op vr cb} [lrange $body 0 end-2] {
 ++                     if {[git-version $op $vr]} {
 ++                             return [uplevel [list $type $name $parm $cb]]
 ++                     }
 ++             }
 ++
 ++             return [uplevel [list $type $name $parm [lindex $body end]]]
 ++     }
 ++
 ++     default {
 ++             error "git-version >= x"
 ++     }
 ++
 ++     }
 ++}
 ++
 ++if {[git-version < 1.5]} {
 ++     catch {wm withdraw .}
 ++     tk_messageBox \
 ++             -icon error \
 ++             -type ok \
 ++             -title [mc "git-gui: fatal error"] \
 ++             -message "[appname] requires Git 1.5.0 or later.
 ++
 ++You are using [git-version]:
 ++
 ++[git --version]"
 ++     exit 1
 ++}
 ++
 ++######################################################################
 ++##
 ++## configure our library
 ++
 ++set idx [file join $oguilib tclIndex]
 ++if {[catch {set fd [open $idx r]} err]} {
 ++     catch {wm withdraw .}
 ++     tk_messageBox \
 ++             -icon error \
 ++             -type ok \
 ++             -title [mc "git-gui: fatal error"] \
 ++             -message $err
 ++     exit 1
 ++}
 ++if {[gets $fd] eq {# Autogenerated by git-gui Makefile}} {
 ++     set idx [list]
 ++     while {[gets $fd n] >= 0} {
 ++             if {$n ne {} && ![string match #* $n]} {
 ++                     lappend idx $n
 ++             }
 ++     }
 ++} else {
 ++     set idx {}
 ++}
 ++close $fd
 ++
 ++if {$idx ne {}} {
 ++     set loaded [list]
 ++     foreach p $idx {
 ++             if {[lsearch -exact $loaded $p] >= 0} continue
 ++             source [file join $oguilib $p]
 ++             lappend loaded $p
 ++     }
 ++     unset loaded p
 ++} else {
 ++     set auto_path [concat [list $oguilib] $auto_path]
 ++}
 ++unset -nocomplain idx fd
 ++
 ++######################################################################
 ++##
 ++## config file parsing
 ++
 ++git-version proc _parse_config {arr_name args} {
 ++     >= 1.5.3 {
 ++             upvar $arr_name arr
 ++             array unset arr
 ++             set buf {}
 ++             catch {
 ++                     set fd_rc [eval \
 ++                             [list git_read config] \
 ++                             $args \
 ++                             [list --null --list]]
 ++                     fconfigure $fd_rc -translation binary
 ++                     set buf [read $fd_rc]
 ++                     close $fd_rc
 ++             }
 ++             foreach line [split $buf "\0"] {
 ++                     if {[regexp {^([^\n]+)\n(.*)$} $line line name value]} {
 ++                             if {[is_many_config $name]} {
 ++                                     lappend arr($name) $value
 ++                             } else {
 ++                                     set arr($name) $value
 ++                             }
 ++                     }
 ++             }
 ++     }
 ++     default {
 ++             upvar $arr_name arr
 ++             array unset arr
 ++             catch {
 ++                     set fd_rc [eval [list git_read config --list] $args]
 ++                     while {[gets $fd_rc line] >= 0} {
 ++                             if {[regexp {^([^=]+)=(.*)$} $line line name value]} {
 ++                                     if {[is_many_config $name]} {
 ++                                             lappend arr($name) $value
 ++                                     } else {
 ++                                             set arr($name) $value
 ++                                     }
 ++                             }
 ++                     }
 ++                     close $fd_rc
 ++             }
 ++     }
 ++}
 ++
 ++proc load_config {include_global} {
 ++     global repo_config global_config default_config
 ++
 ++     if {$include_global} {
 ++             _parse_config global_config --global
 ++     }
 ++     _parse_config repo_config
 ++
 ++     foreach name [array names default_config] {
 ++             if {[catch {set v $global_config($name)}]} {
 ++                     set global_config($name) $default_config($name)
 ++             }
 ++             if {[catch {set v $repo_config($name)}]} {
 ++                     set repo_config($name) $default_config($name)
 ++             }
 ++     }
 ++}
 ++
 ++######################################################################
 ++##
 ++## feature option selection
 ++
 ++if {[regexp {^git-(.+)$} [file tail $argv0] _junk subcommand]} {
 ++     unset _junk
 ++} else {
 ++     set subcommand gui
 ++}
 ++if {$subcommand eq {gui.sh}} {
 ++     set subcommand gui
 ++}
 ++if {$subcommand eq {gui} && [llength $argv] > 0} {
 ++     set subcommand [lindex $argv 0]
 ++     set argv [lrange $argv 1 end]
 ++}
 ++
 ++enable_option multicommit
 ++enable_option branch
 ++enable_option transport
 ++disable_option bare
 ++
 ++switch -- $subcommand {
 ++browser -
 ++blame {
 ++     enable_option bare
 ++
 ++     disable_option multicommit
 ++     disable_option branch
 ++     disable_option transport
 ++}
 ++citool {
 ++     enable_option singlecommit
 ++
 ++     disable_option multicommit
 ++     disable_option branch
 ++     disable_option transport
 ++}
 ++}
 ++
 ++######################################################################
 ++##
 ++## repository setup
 ++
 ++if {[catch {
 ++             set _gitdir $env(GIT_DIR)
 ++             set _prefix {}
 ++             }]
 ++     && [catch {
 ++             set _gitdir [git rev-parse --git-dir]
 ++             set _prefix [git rev-parse --show-prefix]
 ++     } err]} {
 ++     load_config 1
 ++     apply_config
 ++     choose_repository::pick
 ++}
 ++if {![file isdirectory $_gitdir] && [is_Cygwin]} {
 ++     catch {set _gitdir [exec cygpath --windows $_gitdir]}
 ++}
 ++if {![file isdirectory $_gitdir]} {
 ++     catch {wm withdraw .}
 ++     error_popup [strcat [mc "Git directory not found:"] "\n\n$_gitdir"]
 ++     exit 1
 ++}
 ++if {$_prefix ne {}} {
 ++     regsub -all {[^/]+/} $_prefix ../ cdup
 ++     if {[catch {cd $cdup} err]} {
 ++             catch {wm withdraw .}
 ++             error_popup [strcat [mc "Cannot move to top of working directory:"] "\n\n$err"]
 ++             exit 1
 ++     }
 ++     unset cdup
 ++} elseif {![is_enabled bare]} {
 ++     if {[lindex [file split $_gitdir] end] ne {.git}} {
 ++             catch {wm withdraw .}
 ++             error_popup [strcat [mc "Cannot use funny .git directory:"] "\n\n$_gitdir"]
 ++             exit 1
 ++     }
 ++     if {[catch {cd [file dirname $_gitdir]} err]} {
 ++             catch {wm withdraw .}
 ++             error_popup [strcat [mc "No working directory"] " [file dirname $_gitdir]:\n\n$err"]
 ++             exit 1
 ++     }
 ++}
 ++set _reponame [file split [file normalize $_gitdir]]
 ++if {[lindex $_reponame end] eq {.git}} {
 ++     set _reponame [lindex $_reponame end-1]
 ++} else {
 ++     set _reponame [lindex $_reponame end]
 ++}
 ++
 ++######################################################################
 ++##
 ++## global init
 ++
 ++set current_diff_path {}
 ++set current_diff_side {}
 ++set diff_actions [list]
 ++
 ++set HEAD {}
 ++set PARENT {}
 ++set MERGE_HEAD [list]
 ++set commit_type {}
 ++set empty_tree {}
 ++set current_branch {}
 ++set is_detached 0
 ++set current_diff_path {}
 ++set is_3way_diff 0
 ++set selected_commit_type new
 ++
 ++######################################################################
 ++##
 ++## task management
 ++
 ++set rescan_active 0
 ++set diff_active 0
 ++set last_clicked {}
 ++
 ++set disable_on_lock [list]
 ++set index_lock_type none
 ++
 ++proc lock_index {type} {
 ++     global index_lock_type disable_on_lock
 ++
 ++     if {$index_lock_type eq {none}} {
 ++             set index_lock_type $type
 ++             foreach w $disable_on_lock {
 ++                     uplevel #0 $w disabled
 ++             }
 ++             return 1
 ++     } elseif {$index_lock_type eq "begin-$type"} {
 ++             set index_lock_type $type
 ++             return 1
 ++     }
 ++     return 0
 ++}
 ++
 ++proc unlock_index {} {
 ++     global index_lock_type disable_on_lock
 ++
 ++     set index_lock_type none
 ++     foreach w $disable_on_lock {
 ++             uplevel #0 $w normal
 ++     }
 ++}
 ++
 ++######################################################################
 ++##
 ++## status
 ++
 ++proc repository_state {ctvar hdvar mhvar} {
 ++     global current_branch
 ++     upvar $ctvar ct $hdvar hd $mhvar mh
 ++
 ++     set mh [list]
 ++
 ++     load_current_branch
 ++     if {[catch {set hd [git rev-parse --verify HEAD]}]} {
 ++             set hd {}
 ++             set ct initial
 ++             return
 ++     }
 ++
 ++     set merge_head [gitdir MERGE_HEAD]
 ++     if {[file exists $merge_head]} {
 ++             set ct merge
 ++             set fd_mh [open $merge_head r]
 ++             while {[gets $fd_mh line] >= 0} {
 ++                     lappend mh $line
 ++             }
 ++             close $fd_mh
 ++             return
 ++     }
 ++
 ++     set ct normal
 ++}
 ++
 ++proc PARENT {} {
 ++     global PARENT empty_tree
 ++
 ++     set p [lindex $PARENT 0]
 ++     if {$p ne {}} {
 ++             return $p
 ++     }
 ++     if {$empty_tree eq {}} {
 ++             set empty_tree [git mktree << {}]
 ++     }
 ++     return $empty_tree
 ++}
 ++
 ++proc rescan {after {honor_trustmtime 1}} {
 ++     global HEAD PARENT MERGE_HEAD commit_type
 ++     global ui_index ui_workdir ui_comm
 ++     global rescan_active file_states
 ++     global repo_config
 ++
 ++     if {$rescan_active > 0 || ![lock_index read]} return
 ++
 ++     repository_state newType newHEAD newMERGE_HEAD
 ++     if {[string match amend* $commit_type]
 ++             && $newType eq {normal}
 ++             && $newHEAD eq $HEAD} {
 ++     } else {
 ++             set HEAD $newHEAD
 ++             set PARENT $newHEAD
 ++             set MERGE_HEAD $newMERGE_HEAD
 ++             set commit_type $newType
 ++     }
 ++
 ++     array unset file_states
 ++
 ++     if {!$::GITGUI_BCK_exists &&
 ++             (![$ui_comm edit modified]
 ++             || [string trim [$ui_comm get 0.0 end]] eq {})} {
 ++             if {[string match amend* $commit_type]} {
 ++             } elseif {[load_message GITGUI_MSG]} {
 ++             } elseif {[load_message MERGE_MSG]} {
 ++             } elseif {[load_message SQUASH_MSG]} {
 ++             }
 ++             $ui_comm edit reset
 ++             $ui_comm edit modified false
 ++     }
 ++
 ++     if {$honor_trustmtime && $repo_config(gui.trustmtime) eq {true}} {
 ++             rescan_stage2 {} $after
 ++     } else {
 ++             set rescan_active 1
 ++             ui_status [mc "Refreshing file status..."]
 ++             set fd_rf [git_read update-index \
 ++                     -q \
 ++                     --unmerged \
 ++                     --ignore-missing \
 ++                     --refresh \
 ++                     ]
 ++             fconfigure $fd_rf -blocking 0 -translation binary
 ++             fileevent $fd_rf readable \
 ++                     [list rescan_stage2 $fd_rf $after]
 ++     }
 ++}
 ++
 ++if {[is_Cygwin]} {
 ++     set is_git_info_exclude {}
 ++     proc have_info_exclude {} {
 ++             global is_git_info_exclude
 ++
 ++             if {$is_git_info_exclude eq {}} {
 ++                     if {[catch {exec test -f [gitdir info exclude]}]} {
 ++                             set is_git_info_exclude 0
 ++                     } else {
 ++                             set is_git_info_exclude 1
 ++                     }
 ++             }
 ++             return $is_git_info_exclude
 ++     }
 ++} else {
 ++     proc have_info_exclude {} {
 ++             return [file readable [gitdir info exclude]]
 ++     }
 ++}
 ++
 ++proc rescan_stage2 {fd after} {
 ++     global rescan_active buf_rdi buf_rdf buf_rlo
 ++
 ++     if {$fd ne {}} {
 ++             read $fd
 ++             if {![eof $fd]} return
 ++             close $fd
 ++     }
 ++
 ++     set ls_others [list --exclude-per-directory=.gitignore]
 ++     if {[have_info_exclude]} {
 ++             lappend ls_others "--exclude-from=[gitdir info exclude]"
 ++     }
 ++     set user_exclude [get_config core.excludesfile]
 ++     if {$user_exclude ne {} && [file readable $user_exclude]} {
 ++             lappend ls_others "--exclude-from=$user_exclude"
 ++     }
 ++
 ++     set buf_rdi {}
 ++     set buf_rdf {}
 ++     set buf_rlo {}
 ++
 ++     set rescan_active 3
 ++     ui_status [mc "Scanning for modified files ..."]
 ++     set fd_di [git_read diff-index --cached -z [PARENT]]
 ++     set fd_df [git_read diff-files -z]
 ++     set fd_lo [eval git_read ls-files --others -z $ls_others]
 ++
 ++     fconfigure $fd_di -blocking 0 -translation binary -encoding binary
 ++     fconfigure $fd_df -blocking 0 -translation binary -encoding binary
 ++     fconfigure $fd_lo -blocking 0 -translation binary -encoding binary
 ++     fileevent $fd_di readable [list read_diff_index $fd_di $after]
 ++     fileevent $fd_df readable [list read_diff_files $fd_df $after]
 ++     fileevent $fd_lo readable [list read_ls_others $fd_lo $after]
 ++}
 ++
 ++proc load_message {file} {
 ++     global ui_comm
 ++
 ++     set f [gitdir $file]
 ++     if {[file isfile $f]} {
 ++             if {[catch {set fd [open $f r]}]} {
 ++                     return 0
 ++             }
 ++             fconfigure $fd -eofchar {}
 ++             set content [string trim [read $fd]]
 ++             close $fd
 ++             regsub -all -line {[ \r\t]+$} $content {} content
 ++             $ui_comm delete 0.0 end
 ++             $ui_comm insert end $content
 ++             return 1
 ++     }
 ++     return 0
 ++}
 ++
 ++proc read_diff_index {fd after} {
 ++     global buf_rdi
 ++
 ++     append buf_rdi [read $fd]
 ++     set c 0
 ++     set n [string length $buf_rdi]
 ++     while {$c < $n} {
 ++             set z1 [string first "\0" $buf_rdi $c]
 ++             if {$z1 == -1} break
 ++             incr z1
 ++             set z2 [string first "\0" $buf_rdi $z1]
 ++             if {$z2 == -1} break
 ++
 ++             incr c
 ++             set i [split [string range $buf_rdi $c [expr {$z1 - 2}]] { }]
 ++             set p [string range $buf_rdi $z1 [expr {$z2 - 1}]]
 ++             merge_state \
 ++                     [encoding convertfrom $p] \
 ++                     [lindex $i 4]? \
 ++                     [list [lindex $i 0] [lindex $i 2]] \
 ++                     [list]
 ++             set c $z2
 ++             incr c
 ++     }
 ++     if {$c < $n} {
 ++             set buf_rdi [string range $buf_rdi $c end]
 ++     } else {
 ++             set buf_rdi {}
 ++     }
 ++
 ++     rescan_done $fd buf_rdi $after
 ++}
 ++
 ++proc read_diff_files {fd after} {
 ++     global buf_rdf
 ++
 ++     append buf_rdf [read $fd]
 ++     set c 0
 ++     set n [string length $buf_rdf]
 ++     while {$c < $n} {
 ++             set z1 [string first "\0" $buf_rdf $c]
 ++             if {$z1 == -1} break
 ++             incr z1
 ++             set z2 [string first "\0" $buf_rdf $z1]
 ++             if {$z2 == -1} break
 ++
 ++             incr c
 ++             set i [split [string range $buf_rdf $c [expr {$z1 - 2}]] { }]
 ++             set p [string range $buf_rdf $z1 [expr {$z2 - 1}]]
 ++             merge_state \
 ++                     [encoding convertfrom $p] \
 ++                     ?[lindex $i 4] \
 ++                     [list] \
 ++                     [list [lindex $i 0] [lindex $i 2]]
 ++             set c $z2
 ++             incr c
 ++     }
 ++     if {$c < $n} {
 ++             set buf_rdf [string range $buf_rdf $c end]
 ++     } else {
 ++             set buf_rdf {}
 ++     }
 ++
 ++     rescan_done $fd buf_rdf $after
 ++}
 ++
 ++proc read_ls_others {fd after} {
 ++     global buf_rlo
 ++
 ++     append buf_rlo [read $fd]
 ++     set pck [split $buf_rlo "\0"]
 ++     set buf_rlo [lindex $pck end]
 ++     foreach p [lrange $pck 0 end-1] {
 ++             set p [encoding convertfrom $p]
 ++             if {[string index $p end] eq {/}} {
 ++                     set p [string range $p 0 end-1]
 ++             }
 ++             merge_state $p ?O
 ++     }
 ++     rescan_done $fd buf_rlo $after
 ++}
 ++
 ++proc rescan_done {fd buf after} {
 ++     global rescan_active current_diff_path
 ++     global file_states repo_config
 ++     upvar $buf to_clear
 ++
 ++     if {![eof $fd]} return
 ++     set to_clear {}
 ++     close $fd
 ++     if {[incr rescan_active -1] > 0} return
 ++
 ++     prune_selection
 ++     unlock_index
 ++     display_all_files
 ++     if {$current_diff_path ne {}} reshow_diff
+++     if {$current_diff_path eq {}} select_first_diff
+++
 ++     uplevel #0 $after
 ++}
 ++
 ++proc prune_selection {} {
 ++     global file_states selected_paths
 ++
 ++     foreach path [array names selected_paths] {
 ++             if {[catch {set still_here $file_states($path)}]} {
 ++                     unset selected_paths($path)
 ++             }
 ++     }
 ++}
 ++
 ++######################################################################
 ++##
 ++## ui helpers
 ++
 ++proc mapicon {w state path} {
 ++     global all_icons
 ++
 ++     if {[catch {set r $all_icons($state$w)}]} {
 ++             puts "error: no icon for $w state={$state} $path"
 ++             return file_plain
 ++     }
 ++     return $r
 ++}
 ++
 ++proc mapdesc {state path} {
 ++     global all_descs
 ++
 ++     if {[catch {set r $all_descs($state)}]} {
 ++             puts "error: no desc for state={$state} $path"
 ++             return $state
 ++     }
 ++     return $r
 ++}
 ++
 ++proc ui_status {msg} {
 ++     global main_status
 ++     if {[info exists main_status]} {
 ++             $main_status show $msg
 ++     }
 ++}
 ++
 ++proc ui_ready {{test {}}} {
 ++     global main_status
 ++     if {[info exists main_status]} {
 ++             $main_status show [mc "Ready."] $test
 ++     }
 ++}
 ++
 ++proc escape_path {path} {
 ++     regsub -all {\\} $path "\\\\" path
 ++     regsub -all "\n" $path "\\n" path
 ++     return $path
 ++}
 ++
 ++proc short_path {path} {
 ++     return [escape_path [lindex [file split $path] end]]
 ++}
 ++
 ++set next_icon_id 0
 ++set null_sha1 [string repeat 0 40]
 ++
 ++proc merge_state {path new_state {head_info {}} {index_info {}}} {
 ++     global file_states next_icon_id null_sha1
 ++
 ++     set s0 [string index $new_state 0]
 ++     set s1 [string index $new_state 1]
 ++
 ++     if {[catch {set info $file_states($path)}]} {
 ++             set state __
 ++             set icon n[incr next_icon_id]
 ++     } else {
 ++             set state [lindex $info 0]
 ++             set icon [lindex $info 1]
 ++             if {$head_info eq {}}  {set head_info  [lindex $info 2]}
 ++             if {$index_info eq {}} {set index_info [lindex $info 3]}
 ++     }
 ++
 ++     if     {$s0 eq {?}} {set s0 [string index $state 0]} \
 ++     elseif {$s0 eq {_}} {set s0 _}
 ++
 ++     if     {$s1 eq {?}} {set s1 [string index $state 1]} \
 ++     elseif {$s1 eq {_}} {set s1 _}
 ++
 ++     if {$s0 eq {A} && $s1 eq {_} && $head_info eq {}} {
 ++             set head_info [list 0 $null_sha1]
 ++     } elseif {$s0 ne {_} && [string index $state 0] eq {_}
 ++             && $head_info eq {}} {
 ++             set head_info $index_info
 ++     }
 ++
 ++     set file_states($path) [list $s0$s1 $icon \
 ++             $head_info $index_info \
 ++             ]
 ++     return $state
 ++}
 ++
 ++proc display_file_helper {w path icon_name old_m new_m} {
 ++     global file_lists
 ++
 ++     if {$new_m eq {_}} {
 ++             set lno [lsearch -sorted -exact $file_lists($w) $path]
 ++             if {$lno >= 0} {
 ++                     set file_lists($w) [lreplace $file_lists($w) $lno $lno]
 ++                     incr lno
 ++                     $w conf -state normal
 ++                     $w delete $lno.0 [expr {$lno + 1}].0
 ++                     $w conf -state disabled
 ++             }
 ++     } elseif {$old_m eq {_} && $new_m ne {_}} {
 ++             lappend file_lists($w) $path
 ++             set file_lists($w) [lsort -unique $file_lists($w)]
 ++             set lno [lsearch -sorted -exact $file_lists($w) $path]
 ++             incr lno
 ++             $w conf -state normal
 ++             $w image create $lno.0 \
 ++                     -align center -padx 5 -pady 1 \
 ++                     -name $icon_name \
 ++                     -image [mapicon $w $new_m $path]
 ++             $w insert $lno.1 "[escape_path $path]\n"
 ++             $w conf -state disabled
 ++     } elseif {$old_m ne $new_m} {
 ++             $w conf -state normal
 ++             $w image conf $icon_name -image [mapicon $w $new_m $path]
 ++             $w conf -state disabled
 ++     }
 ++}
 ++
 ++proc display_file {path state} {
 ++     global file_states selected_paths
 ++     global ui_index ui_workdir
 ++
 ++     set old_m [merge_state $path $state]
 ++     set s $file_states($path)
 ++     set new_m [lindex $s 0]
 ++     set icon_name [lindex $s 1]
 ++
 ++     set o [string index $old_m 0]
 ++     set n [string index $new_m 0]
 ++     if {$o eq {U}} {
 ++             set o _
 ++     }
 ++     if {$n eq {U}} {
 ++             set n _
 ++     }
 ++     display_file_helper     $ui_index $path $icon_name $o $n
 ++
 ++     if {[string index $old_m 0] eq {U}} {
 ++             set o U
 ++     } else {
 ++             set o [string index $old_m 1]
 ++     }
 ++     if {[string index $new_m 0] eq {U}} {
 ++             set n U
 ++     } else {
 ++             set n [string index $new_m 1]
 ++     }
 ++     display_file_helper     $ui_workdir $path $icon_name $o $n
 ++
 ++     if {$new_m eq {__}} {
 ++             unset file_states($path)
 ++             catch {unset selected_paths($path)}
 ++     }
 ++}
 ++
 ++proc display_all_files_helper {w path icon_name m} {
 ++     global file_lists
 ++
 ++     lappend file_lists($w) $path
 ++     set lno [expr {[lindex [split [$w index end] .] 0] - 1}]
 ++     $w image create end \
 ++             -align center -padx 5 -pady 1 \
 ++             -name $icon_name \
 ++             -image [mapicon $w $m $path]
 ++     $w insert end "[escape_path $path]\n"
 ++}
 ++
 ++proc display_all_files {} {
 ++     global ui_index ui_workdir
 ++     global file_states file_lists
 ++     global last_clicked
 ++
 ++     $ui_index conf -state normal
 ++     $ui_workdir conf -state normal
 ++
 ++     $ui_index delete 0.0 end
 ++     $ui_workdir delete 0.0 end
 ++     set last_clicked {}
 ++
 ++     set file_lists($ui_index) [list]
 ++     set file_lists($ui_workdir) [list]
 ++
 ++     foreach path [lsort [array names file_states]] {
 ++             set s $file_states($path)
 ++             set m [lindex $s 0]
 ++             set icon_name [lindex $s 1]
 ++
 ++             set s [string index $m 0]
 ++             if {$s ne {U} && $s ne {_}} {
 ++                     display_all_files_helper $ui_index $path \
 ++                             $icon_name $s
 ++             }
 ++
 ++             if {[string index $m 0] eq {U}} {
 ++                     set s U
 ++             } else {
 ++                     set s [string index $m 1]
 ++             }
 ++             if {$s ne {_}} {
 ++                     display_all_files_helper $ui_workdir $path \
 ++                             $icon_name $s
 ++             }
 ++     }
 ++
 ++     $ui_index conf -state disabled
 ++     $ui_workdir conf -state disabled
 ++}
 ++
 ++######################################################################
 ++##
 ++## icons
 ++
 ++set filemask {
 ++#define mask_width 14
 ++#define mask_height 15
 ++static unsigned char mask_bits[] = {
 ++   0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f,
 ++   0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f,
 ++   0xfe, 0x1f, 0xfe, 0x1f, 0xfe, 0x1f};
 ++}
 ++
 ++image create bitmap file_plain -background white -foreground black -data {
 ++#define plain_width 14
 ++#define plain_height 15
 ++static unsigned char plain_bits[] = {
 ++   0xfe, 0x01, 0x02, 0x03, 0x02, 0x05, 0x02, 0x09, 0x02, 0x1f, 0x02, 0x10,
 ++   0x02, 0x10, 0x02, 0x10, 0x02, 0x10, 0x02, 0x10, 0x02, 0x10, 0x02, 0x10,
 ++   0x02, 0x10, 0x02, 0x10, 0xfe, 0x1f};
 ++} -maskdata $filemask
 ++
 ++image create bitmap file_mod -background white -foreground blue -data {
 ++#define mod_width 14
 ++#define mod_height 15
 ++static unsigned char mod_bits[] = {
 ++   0xfe, 0x01, 0x02, 0x03, 0x7a, 0x05, 0x02, 0x09, 0x7a, 0x1f, 0x02, 0x10,
 ++   0xfa, 0x17, 0x02, 0x10, 0xfa, 0x17, 0x02, 0x10, 0xfa, 0x17, 0x02, 0x10,
 ++   0xfa, 0x17, 0x02, 0x10, 0xfe, 0x1f};
 ++} -maskdata $filemask
 ++
 ++image create bitmap file_fulltick -background white -foreground "#007000" -data {
 ++#define file_fulltick_width 14
 ++#define file_fulltick_height 15
 ++static unsigned char file_fulltick_bits[] = {
 ++   0xfe, 0x01, 0x02, 0x1a, 0x02, 0x0c, 0x02, 0x0c, 0x02, 0x16, 0x02, 0x16,
 ++   0x02, 0x13, 0x00, 0x13, 0x86, 0x11, 0x8c, 0x11, 0xd8, 0x10, 0xf2, 0x10,
 ++   0x62, 0x10, 0x02, 0x10, 0xfe, 0x1f};
 ++} -maskdata $filemask
 ++
 ++image create bitmap file_parttick -background white -foreground "#005050" -data {
 ++#define parttick_width 14
 ++#define parttick_height 15
 ++static unsigned char parttick_bits[] = {
 ++   0xfe, 0x01, 0x02, 0x03, 0x7a, 0x05, 0x02, 0x09, 0x7a, 0x1f, 0x02, 0x10,
 ++   0x7a, 0x14, 0x02, 0x16, 0x02, 0x13, 0x8a, 0x11, 0xda, 0x10, 0x72, 0x10,
 ++   0x22, 0x10, 0x02, 0x10, 0xfe, 0x1f};
 ++} -maskdata $filemask
 ++
 ++image create bitmap file_question -background white -foreground black -data {
 ++#define file_question_width 14
 ++#define file_question_height 15
 ++static unsigned char file_question_bits[] = {
 ++   0xfe, 0x01, 0x02, 0x02, 0xe2, 0x04, 0xf2, 0x09, 0x1a, 0x1b, 0x0a, 0x13,
 ++   0x82, 0x11, 0xc2, 0x10, 0x62, 0x10, 0x62, 0x10, 0x02, 0x10, 0x62, 0x10,
 ++   0x62, 0x10, 0x02, 0x10, 0xfe, 0x1f};
 ++} -maskdata $filemask
 ++
 ++image create bitmap file_removed -background white -foreground red -data {
 ++#define file_removed_width 14
 ++#define file_removed_height 15
 ++static unsigned char file_removed_bits[] = {
 ++   0xfe, 0x01, 0x02, 0x03, 0x02, 0x05, 0x02, 0x09, 0x02, 0x1f, 0x02, 0x10,
 ++   0x1a, 0x16, 0x32, 0x13, 0xe2, 0x11, 0xc2, 0x10, 0xe2, 0x11, 0x32, 0x13,
 ++   0x1a, 0x16, 0x02, 0x10, 0xfe, 0x1f};
 ++} -maskdata $filemask
 ++
 ++image create bitmap file_merge -background white -foreground blue -data {
 ++#define file_merge_width 14
 ++#define file_merge_height 15
 ++static unsigned char file_merge_bits[] = {
 ++   0xfe, 0x01, 0x02, 0x03, 0x62, 0x05, 0x62, 0x09, 0x62, 0x1f, 0x62, 0x10,
 ++   0xfa, 0x11, 0xf2, 0x10, 0x62, 0x10, 0x02, 0x10, 0xfa, 0x17, 0x02, 0x10,
 ++   0xfa, 0x17, 0x02, 0x10, 0xfe, 0x1f};
 ++} -maskdata $filemask
 ++
+++image create bitmap file_statechange -background white -foreground green -data {
+++#define file_merge_width 14
+++#define file_merge_height 15
+++static unsigned char file_statechange_bits[] = {
+++   0xfe, 0x01, 0x02, 0x03, 0x02, 0x05, 0x02, 0x09, 0x02, 0x1f, 0x62, 0x10,
+++   0x62, 0x10, 0xba, 0x11, 0xba, 0x11, 0x62, 0x10, 0x62, 0x10, 0x02, 0x10,
+++   0x02, 0x10, 0x02, 0x10, 0xfe, 0x1f};
+++} -maskdata $filemask
+++
 ++set ui_index .vpane.files.index.list
 ++set ui_workdir .vpane.files.workdir.list
 ++
 ++set all_icons(_$ui_index)   file_plain
 ++set all_icons(A$ui_index)   file_fulltick
 ++set all_icons(M$ui_index)   file_fulltick
 ++set all_icons(D$ui_index)   file_removed
 ++set all_icons(U$ui_index)   file_merge
+++set all_icons(T$ui_index)   file_statechange
 ++
 ++set all_icons(_$ui_workdir) file_plain
 ++set all_icons(M$ui_workdir) file_mod
 ++set all_icons(D$ui_workdir) file_question
 ++set all_icons(U$ui_workdir) file_merge
 ++set all_icons(O$ui_workdir) file_plain
+++set all_icons(T$ui_workdir) file_statechange
 ++
 ++set max_status_desc 0
 ++foreach i {
 ++             {__ {mc "Unmodified"}}
 ++
 ++             {_M {mc "Modified, not staged"}}
 ++             {M_ {mc "Staged for commit"}}
 ++             {MM {mc "Portions staged for commit"}}
 ++             {MD {mc "Staged for commit, missing"}}
 ++
+++             {_T {mc "File type changed, not staged"}}
+++             {T_ {mc "File type changed, staged"}}
+++
 ++             {_O {mc "Untracked, not staged"}}
 ++             {A_ {mc "Staged for commit"}}
 ++             {AM {mc "Portions staged for commit"}}
 ++             {AD {mc "Staged for commit, missing"}}
 ++
 ++             {_D {mc "Missing"}}
 ++             {D_ {mc "Staged for removal"}}
 ++             {DO {mc "Staged for removal, still present"}}
 ++
+++             {_U {mc "Requires merge resolution"}}
 ++             {U_ {mc "Requires merge resolution"}}
 ++             {UU {mc "Requires merge resolution"}}
 ++             {UM {mc "Requires merge resolution"}}
 ++             {UD {mc "Requires merge resolution"}}
+++             {UT {mc "Requires merge resolution"}}
 ++     } {
 ++     set text [eval [lindex $i 1]]
 ++     if {$max_status_desc < [string length $text]} {
 ++             set max_status_desc [string length $text]
 ++     }
 ++     set all_descs([lindex $i 0]) $text
 ++}
 ++unset i
 ++
 ++######################################################################
 ++##
 ++## util
 ++
 ++proc scrollbar2many {list mode args} {
 ++     foreach w $list {eval $w $mode $args}
 ++}
 ++
 ++proc many2scrollbar {list mode sb top bottom} {
 ++     $sb set $top $bottom
 ++     foreach w $list {$w $mode moveto $top}
 ++}
 ++
 ++proc incr_font_size {font {amt 1}} {
 ++     set sz [font configure $font -size]
 ++     incr sz $amt
 ++     font configure $font -size $sz
 ++     font configure ${font}bold -size $sz
 ++     font configure ${font}italic -size $sz
 ++}
 ++
 ++######################################################################
 ++##
 ++## ui commands
 ++
 ++set starting_gitk_msg [mc "Starting gitk... please wait..."]
 ++
 ++proc do_gitk {revs} {
 ++     # -- Always start gitk through whatever we were loaded with.  This
 ++     #    lets us bypass using shell process on Windows systems.
 ++     #
 ++     set exe [_which gitk -script]
 ++     set cmd [list [info nameofexecutable] $exe]
 ++     if {$exe eq {}} {
 ++             error_popup [mc "Couldn't find gitk in PATH"]
 ++     } else {
 ++             global env
 ++
 ++             if {[info exists env(GIT_DIR)]} {
 ++                     set old_GIT_DIR $env(GIT_DIR)
 ++             } else {
 ++                     set old_GIT_DIR {}
 ++             }
 ++
 ++             set pwd [pwd]
 ++             cd [file dirname [gitdir]]
 ++             set env(GIT_DIR) [file tail [gitdir]]
 ++
 ++             eval exec $cmd $revs &
 ++
 ++             if {$old_GIT_DIR eq {}} {
 ++                     unset env(GIT_DIR)
 ++             } else {
 ++                     set env(GIT_DIR) $old_GIT_DIR
 ++             }
 ++             cd $pwd
 ++
 ++             ui_status $::starting_gitk_msg
 ++             after 10000 {
 ++                     ui_ready $starting_gitk_msg
 ++             }
 ++     }
 ++}
 ++
 ++set is_quitting 0
 ++
 ++proc do_quit {} {
 ++     global ui_comm is_quitting repo_config commit_type
 ++     global GITGUI_BCK_exists GITGUI_BCK_i
 ++     global ui_comm_spell
 ++
 ++     if {$is_quitting} return
 ++     set is_quitting 1
 ++
 ++     if {[winfo exists $ui_comm]} {
 ++             # -- Stash our current commit buffer.
 ++             #
 ++             set save [gitdir GITGUI_MSG]
 ++             if {$GITGUI_BCK_exists && ![$ui_comm edit modified]} {
 ++                     file rename -force [gitdir GITGUI_BCK] $save
 ++                     set GITGUI_BCK_exists 0
 ++             } else {
 ++                     set msg [string trim [$ui_comm get 0.0 end]]
 ++                     regsub -all -line {[ \r\t]+$} $msg {} msg
 ++                     if {(![string match amend* $commit_type]
 ++                             || [$ui_comm edit modified])
 ++                             && $msg ne {}} {
 ++                             catch {
 ++                                     set fd [open $save w]
 ++                                     puts -nonewline $fd $msg
 ++                                     close $fd
 ++                             }
 ++                     } else {
 ++                             catch {file delete $save}
 ++                     }
 ++             }
 ++
 ++             # -- Cancel our spellchecker if its running.
 ++             #
 ++             if {[info exists ui_comm_spell]} {
 ++                     $ui_comm_spell stop
 ++             }
 ++
 ++             # -- Remove our editor backup, its not needed.
 ++             #
 ++             after cancel $GITGUI_BCK_i
 ++             if {$GITGUI_BCK_exists} {
 ++                     catch {file delete [gitdir GITGUI_BCK]}
 ++             }
 ++
 ++             # -- Stash our current window geometry into this repository.
 ++             #
 ++             set cfg_geometry [list]
 ++             lappend cfg_geometry [wm geometry .]
 ++             lappend cfg_geometry [lindex [.vpane sash coord 0] 0]
 ++             lappend cfg_geometry [lindex [.vpane.files sash coord 0] 1]
 ++             if {[catch {set rc_geometry $repo_config(gui.geometry)}]} {
 ++                     set rc_geometry {}
 ++             }
 ++             if {$cfg_geometry ne $rc_geometry} {
 ++                     catch {git config gui.geometry $cfg_geometry}
 ++             }
 ++     }
 ++
 ++     destroy .
 ++}
 ++
 ++proc do_rescan {} {
 ++     rescan ui_ready
 ++}
 ++
+++proc ui_do_rescan {} {
+++     rescan {force_first_diff; ui_ready}
+++}
+++
 ++proc do_commit {} {
 ++     commit_tree
 ++}
 ++
 ++proc next_diff {} {
 ++     global next_diff_p next_diff_w next_diff_i
-       if {$col == 0 && $y > 1} {
-               set i [expr {$lno-1}]
-               set ll [expr {[llength $file_lists($w)]-1}]
-  
-               if {$i == $ll && $i == 0} {
-                       set after {reshow_diff;}
-               } else {
-                       global next_diff_p next_diff_w next_diff_i
-  
-                       set next_diff_w $w
-  
-                       if {$i < $ll} {
-                               set i [expr {$i + 1}]
-                               set next_diff_i $i
-                       } else {
-                               set next_diff_i $i
-                               set i [expr {$i - 1}]
-                       }
+++     show_diff $next_diff_p $next_diff_w {}
+++}
+++
+++proc find_anchor_pos {lst name} {
+++     set lid [lsearch -sorted -exact $lst $name]
+++
+++     if {$lid == -1} {
+++             set lid 0
+++             foreach lname $lst {
+++                     if {$lname >= $name} break
+++                     incr lid
+++             }
+++     }
+++
+++     return $lid
+++}
+++
+++proc find_file_from {flist idx delta path mmask} {
+++     global file_states
+++
+++     set len [llength $flist]
+++     while {$idx >= 0 && $idx < $len} {
+++             set name [lindex $flist $idx]
+++
+++             if {$name ne $path && [info exists file_states($name)]} {
+++                     set state [lindex $file_states($name) 0]
+++
+++                     if {$mmask eq {} || [regexp $mmask $state]} {
+++                             return $idx
+++                     }
+++             }
+++
+++             incr idx $delta
+++     }
+++
+++     return {}
+++}
+++
+++proc find_next_diff {w path {lno {}} {mmask {}}} {
+++     global next_diff_p next_diff_w next_diff_i
+++     global file_lists ui_index ui_workdir
+++
+++     set flist $file_lists($w)
+++     if {$lno eq {}} {
+++             set lno [find_anchor_pos $flist $path]
+++     } else {
+++             incr lno -1
+++     }
+++
+++     if {$mmask ne {} && ![regexp {(^\^)|(\$$)} $mmask]} {
+++             if {$w eq $ui_index} {
+++                     set mmask "^$mmask"
+++             } else {
+++                     set mmask "$mmask\$"
+++             }
+++     }
+++
+++     set idx [find_file_from $flist $lno 1 $path $mmask]
+++     if {$idx eq {}} {
+++             incr lno -1
+++             set idx [find_file_from $flist $lno -1 $path $mmask]
+++     }
+++
+++     if {$idx ne {}} {
+++             set next_diff_w $w
+++             set next_diff_p [lindex $flist $idx]
+++             set next_diff_i [expr {$idx+1}]
+++             return 1
+++     } else {
+++             return 0
+++     }
+++}
+++
+++proc next_diff_after_action {w path {lno {}} {mmask {}}} {
+++     global current_diff_path
+++
+++     if {$path ne $current_diff_path} {
+++             return {}
+++     } elseif {[find_next_diff $w $path $lno $mmask]} {
+++             return {next_diff;}
+++     } else {
+++             return {reshow_diff;}
+++     }
+++}
+++
+++proc select_first_diff {} {
+++     global ui_workdir
+++
+++     if {[find_next_diff $ui_workdir {} 1 {^_?U}] ||
+++         [find_next_diff $ui_workdir {} 1 {[^O]$}]} {
+++             next_diff
+++     }
+++}
+++
+++proc force_first_diff {} {
+++     global current_diff_path
+++
+++     if {[info exists file_states($current_diff_path)]} {
+++             set state [lindex $file_states($current_diff_path) 0]
+++
+++             if {[string index $state 1] ne {O}} return
+++     }
+++
+++     select_first_diff
 ++}
 ++
 ++proc toggle_or_diff {w x y} {
 ++     global file_states file_lists current_diff_path ui_index ui_workdir
 ++     global last_clicked selected_paths
 ++
 ++     set pos [split [$w index @$x,$y] .]
 ++     set lno [lindex $pos 0]
 ++     set col [lindex $pos 1]
 ++     set path [lindex $file_lists($w) [expr {$lno - 1}]]
 ++     if {$path eq {}} {
 ++             set last_clicked {}
 ++             return
 ++     }
 ++
 ++     set last_clicked [list $w $lno]
 ++     array unset selected_paths
 ++     $ui_index tag remove in_sel 0.0 end
 ++     $ui_workdir tag remove in_sel 0.0 end
 ++
-                       set next_diff_p [lindex $file_lists($w) $i]
+++     # Do not stage files with conflicts
+++     if {[info exists file_states($path)]} {
+++             set state [lindex $file_states($path) 0]
+++     } else {
+++             set state {__}
+++     }
 ++
-                       if {$next_diff_p ne {} && $current_diff_path ne {}} {
-                               set after {next_diff;}
-                       } else {
-                               set after {}
-                       }
+++     if {[string first {U} $state] >= 0} {
+++             set col 1
+++     }
 ++
-               -command do_rescan \
+++     # Restage the file, or simply show the diff
+++     if {$col == 0 && $y > 1} {
+++             if {[string index $state 1] eq {O}} {
+++                     set mmask {}
+++             } else {
+++                     set mmask {[^O]}
 ++             }
 ++
+++             set after [next_diff_after_action $w $path $lno $mmask]
+++
 ++             if {$w eq $ui_index} {
 ++                     update_indexinfo \
 ++                             "Unstaging [short_path $path] from commit" \
 ++                             [list $path] \
 ++                             [concat $after [list ui_ready]]
 ++             } elseif {$w eq $ui_workdir} {
 ++                     update_index \
 ++                             "Adding [short_path $path]" \
 ++                             [list $path] \
 ++                             [concat $after [list ui_ready]]
 ++             }
 ++     } else {
 ++             show_diff $path $w $lno
 ++     }
 ++}
 ++
 ++proc add_one_to_selection {w x y} {
 ++     global file_lists last_clicked selected_paths
 ++
 ++     set lno [lindex [split [$w index @$x,$y] .] 0]
 ++     set path [lindex $file_lists($w) [expr {$lno - 1}]]
 ++     if {$path eq {}} {
 ++             set last_clicked {}
 ++             return
 ++     }
 ++
 ++     if {$last_clicked ne {}
 ++             && [lindex $last_clicked 0] ne $w} {
 ++             array unset selected_paths
 ++             [lindex $last_clicked 0] tag remove in_sel 0.0 end
 ++     }
 ++
 ++     set last_clicked [list $w $lno]
 ++     if {[catch {set in_sel $selected_paths($path)}]} {
 ++             set in_sel 0
 ++     }
 ++     if {$in_sel} {
 ++             unset selected_paths($path)
 ++             $w tag remove in_sel $lno.0 [expr {$lno + 1}].0
 ++     } else {
 ++             set selected_paths($path) 1
 ++             $w tag add in_sel $lno.0 [expr {$lno + 1}].0
 ++     }
 ++}
 ++
 ++proc add_range_to_selection {w x y} {
 ++     global file_lists last_clicked selected_paths
 ++
 ++     if {[lindex $last_clicked 0] ne $w} {
 ++             toggle_or_diff $w $x $y
 ++             return
 ++     }
 ++
 ++     set lno [lindex [split [$w index @$x,$y] .] 0]
 ++     set lc [lindex $last_clicked 1]
 ++     if {$lc < $lno} {
 ++             set begin $lc
 ++             set end $lno
 ++     } else {
 ++             set begin $lno
 ++             set end $lc
 ++     }
 ++
 ++     foreach path [lrange $file_lists($w) \
 ++             [expr {$begin - 1}] \
 ++             [expr {$end - 1}]] {
 ++             set selected_paths($path) 1
 ++     }
 ++     $w tag add in_sel $begin.0 [expr {$end + 1}].0
 ++}
 ++
 ++proc show_more_context {} {
 ++     global repo_config
 ++     if {$repo_config(gui.diffcontext) < 99} {
 ++             incr repo_config(gui.diffcontext)
 ++             reshow_diff
 ++     }
 ++}
 ++
 ++proc show_less_context {} {
 ++     global repo_config
 ++     if {$repo_config(gui.diffcontext) > 1} {
 ++             incr repo_config(gui.diffcontext) -1
 ++             reshow_diff
 ++     }
 ++}
 ++
 ++######################################################################
 ++##
 ++## ui construction
 ++
 ++load_config 0
 ++apply_config
 ++set ui_comm {}
 ++
 ++# -- Menu Bar
 ++#
 ++menu .mbar -tearoff 0
 ++.mbar add cascade -label [mc Repository] -menu .mbar.repository
 ++.mbar add cascade -label [mc Edit] -menu .mbar.edit
 ++if {[is_enabled branch]} {
 ++     .mbar add cascade -label [mc Branch] -menu .mbar.branch
 ++}
 ++if {[is_enabled multicommit] || [is_enabled singlecommit]} {
 ++     .mbar add cascade -label [mc Commit@@noun] -menu .mbar.commit
 ++}
 ++if {[is_enabled transport]} {
 ++     .mbar add cascade -label [mc Merge] -menu .mbar.merge
 ++     .mbar add cascade -label [mc Remote] -menu .mbar.remote
 ++}
 ++. configure -menu .mbar
 ++
 ++# -- Repository Menu
 ++#
 ++menu .mbar.repository
 ++
 ++.mbar.repository add command \
 ++     -label [mc "Browse Current Branch's Files"] \
 ++     -command {browser::new $current_branch}
 ++set ui_browse_current [.mbar.repository index last]
 ++.mbar.repository add command \
 ++     -label [mc "Browse Branch Files..."] \
 ++     -command browser_open::dialog
 ++.mbar.repository add separator
 ++
 ++.mbar.repository add command \
 ++     -label [mc "Visualize Current Branch's History"] \
 ++     -command {do_gitk $current_branch}
 ++set ui_visualize_current [.mbar.repository index last]
 ++.mbar.repository add command \
 ++     -label [mc "Visualize All Branch History"] \
 ++     -command {do_gitk --all}
 ++.mbar.repository add separator
 ++
 ++proc current_branch_write {args} {
 ++     global current_branch
 ++     .mbar.repository entryconf $::ui_browse_current \
 ++             -label [mc "Browse %s's Files" $current_branch]
 ++     .mbar.repository entryconf $::ui_visualize_current \
 ++             -label [mc "Visualize %s's History" $current_branch]
 ++}
 ++trace add variable current_branch write current_branch_write
 ++
 ++if {[is_enabled multicommit]} {
 ++     .mbar.repository add command -label [mc "Database Statistics"] \
 ++             -command do_stats
 ++
 ++     .mbar.repository add command -label [mc "Compress Database"] \
 ++             -command do_gc
 ++
 ++     .mbar.repository add command -label [mc "Verify Database"] \
 ++             -command do_fsck_objects
 ++
 ++     .mbar.repository add separator
 ++
 ++     if {[is_Cygwin]} {
 ++             .mbar.repository add command \
 ++                     -label [mc "Create Desktop Icon"] \
 ++                     -command do_cygwin_shortcut
 ++     } elseif {[is_Windows]} {
 ++             .mbar.repository add command \
 ++                     -label [mc "Create Desktop Icon"] \
 ++                     -command do_windows_shortcut
 ++     } elseif {[is_MacOSX]} {
 ++             .mbar.repository add command \
 ++                     -label [mc "Create Desktop Icon"] \
 ++                     -command do_macosx_app
 ++     }
 ++}
 ++
 ++if {[is_MacOSX]} {
 ++     proc ::tk::mac::Quit {args} { do_quit }
 ++} else {
 ++     .mbar.repository add command -label [mc Quit] \
 ++             -command do_quit \
 ++             -accelerator $M1T-Q
 ++}
 ++
 ++# -- Edit Menu
 ++#
 ++menu .mbar.edit
 ++.mbar.edit add command -label [mc Undo] \
 ++     -command {catch {[focus] edit undo}} \
 ++     -accelerator $M1T-Z
 ++.mbar.edit add command -label [mc Redo] \
 ++     -command {catch {[focus] edit redo}} \
 ++     -accelerator $M1T-Y
 ++.mbar.edit add separator
 ++.mbar.edit add command -label [mc Cut] \
 ++     -command {catch {tk_textCut [focus]}} \
 ++     -accelerator $M1T-X
 ++.mbar.edit add command -label [mc Copy] \
 ++     -command {catch {tk_textCopy [focus]}} \
 ++     -accelerator $M1T-C
 ++.mbar.edit add command -label [mc Paste] \
 ++     -command {catch {tk_textPaste [focus]; [focus] see insert}} \
 ++     -accelerator $M1T-V
 ++.mbar.edit add command -label [mc Delete] \
 ++     -command {catch {[focus] delete sel.first sel.last}} \
 ++     -accelerator Del
 ++.mbar.edit add separator
 ++.mbar.edit add command -label [mc "Select All"] \
 ++     -command {catch {[focus] tag add sel 0.0 end}} \
 ++     -accelerator $M1T-A
 ++
 ++# -- Branch Menu
 ++#
 ++if {[is_enabled branch]} {
 ++     menu .mbar.branch
 ++
 ++     .mbar.branch add command -label [mc "Create..."] \
 ++             -command branch_create::dialog \
 ++             -accelerator $M1T-N
 ++     lappend disable_on_lock [list .mbar.branch entryconf \
 ++             [.mbar.branch index last] -state]
 ++
 ++     .mbar.branch add command -label [mc "Checkout..."] \
 ++             -command branch_checkout::dialog \
 ++             -accelerator $M1T-O
 ++     lappend disable_on_lock [list .mbar.branch entryconf \
 ++             [.mbar.branch index last] -state]
 ++
 ++     .mbar.branch add command -label [mc "Rename..."] \
 ++             -command branch_rename::dialog
 ++     lappend disable_on_lock [list .mbar.branch entryconf \
 ++             [.mbar.branch index last] -state]
 ++
 ++     .mbar.branch add command -label [mc "Delete..."] \
 ++             -command branch_delete::dialog
 ++     lappend disable_on_lock [list .mbar.branch entryconf \
 ++             [.mbar.branch index last] -state]
 ++
 ++     .mbar.branch add command -label [mc "Reset..."] \
 ++             -command merge::reset_hard
 ++     lappend disable_on_lock [list .mbar.branch entryconf \
 ++             [.mbar.branch index last] -state]
 ++}
 ++
 ++# -- Commit Menu
 ++#
 ++if {[is_enabled multicommit] || [is_enabled singlecommit]} {
 ++     menu .mbar.commit
 ++
 ++     .mbar.commit add radiobutton \
 ++             -label [mc "New Commit"] \
 ++             -command do_select_commit_type \
 ++             -variable selected_commit_type \
 ++             -value new
 ++     lappend disable_on_lock \
 ++             [list .mbar.commit entryconf [.mbar.commit index last] -state]
 ++
 ++     .mbar.commit add radiobutton \
 ++             -label [mc "Amend Last Commit"] \
 ++             -command do_select_commit_type \
 ++             -variable selected_commit_type \
 ++             -value amend
 ++     lappend disable_on_lock \
 ++             [list .mbar.commit entryconf [.mbar.commit index last] -state]
 ++
 ++     .mbar.commit add separator
 ++
 ++     .mbar.commit add command -label [mc Rescan] \
-       set subcommand_args {rev? path}
+++             -command ui_do_rescan \
 ++             -accelerator F5
 ++     lappend disable_on_lock \
 ++             [list .mbar.commit entryconf [.mbar.commit index last] -state]
 ++
 ++     .mbar.commit add command -label [mc "Stage To Commit"] \
 ++             -command do_add_selection \
 ++             -accelerator $M1T-T
 ++     lappend disable_on_lock \
 ++             [list .mbar.commit entryconf [.mbar.commit index last] -state]
 ++
 ++     .mbar.commit add command -label [mc "Stage Changed Files To Commit"] \
 ++             -command do_add_all \
 ++             -accelerator $M1T-I
 ++     lappend disable_on_lock \
 ++             [list .mbar.commit entryconf [.mbar.commit index last] -state]
 ++
 ++     .mbar.commit add command -label [mc "Unstage From Commit"] \
 ++             -command do_unstage_selection
 ++     lappend disable_on_lock \
 ++             [list .mbar.commit entryconf [.mbar.commit index last] -state]
 ++
 ++     .mbar.commit add command -label [mc "Revert Changes"] \
 ++             -command do_revert_selection
 ++     lappend disable_on_lock \
 ++             [list .mbar.commit entryconf [.mbar.commit index last] -state]
 ++
 ++     .mbar.commit add separator
 ++
 ++     .mbar.commit add command -label [mc "Show Less Context"] \
 ++             -command show_less_context \
 ++             -accelerator $M1T-\-
 ++
 ++     .mbar.commit add command -label [mc "Show More Context"] \
 ++             -command show_more_context \
 ++             -accelerator $M1T-=
 ++
 ++     .mbar.commit add separator
 ++
 ++     .mbar.commit add command -label [mc "Sign Off"] \
 ++             -command do_signoff \
 ++             -accelerator $M1T-S
 ++
 ++     .mbar.commit add command -label [mc Commit@@verb] \
 ++             -command do_commit \
 ++             -accelerator $M1T-Return
 ++     lappend disable_on_lock \
 ++             [list .mbar.commit entryconf [.mbar.commit index last] -state]
 ++}
 ++
 ++# -- Merge Menu
 ++#
 ++if {[is_enabled branch]} {
 ++     menu .mbar.merge
 ++     .mbar.merge add command -label [mc "Local Merge..."] \
 ++             -command merge::dialog \
 ++             -accelerator $M1T-M
 ++     lappend disable_on_lock \
 ++             [list .mbar.merge entryconf [.mbar.merge index last] -state]
 ++     .mbar.merge add command -label [mc "Abort Merge..."] \
 ++             -command merge::reset_hard
 ++     lappend disable_on_lock \
 ++             [list .mbar.merge entryconf [.mbar.merge index last] -state]
 ++}
 ++
 ++# -- Transport Menu
 ++#
 ++if {[is_enabled transport]} {
 ++     menu .mbar.remote
 ++
 ++     .mbar.remote add command \
 ++             -label [mc "Push..."] \
 ++             -command do_push_anywhere \
 ++             -accelerator $M1T-P
 ++     .mbar.remote add command \
 ++             -label [mc "Delete..."] \
 ++             -command remote_branch_delete::dialog
 ++}
 ++
 ++if {[is_MacOSX]} {
 ++     # -- Apple Menu (Mac OS X only)
 ++     #
 ++     .mbar add cascade -label Apple -menu .mbar.apple
 ++     menu .mbar.apple
 ++
 ++     .mbar.apple add command -label [mc "About %s" [appname]] \
 ++             -command do_about
 ++     .mbar.apple add separator
 ++     .mbar.apple add command \
 ++             -label [mc "Preferences..."] \
 ++             -command do_options \
 ++             -accelerator $M1T-,
 ++     bind . <$M1B-,> do_options
 ++} else {
 ++     # -- Edit Menu
 ++     #
 ++     .mbar.edit add separator
 ++     .mbar.edit add command -label [mc "Options..."] \
 ++             -command do_options
 ++}
 ++
 ++# -- Help Menu
 ++#
 ++.mbar add cascade -label [mc Help] -menu .mbar.help
 ++menu .mbar.help
 ++
 ++if {![is_MacOSX]} {
 ++     .mbar.help add command -label [mc "About %s" [appname]] \
 ++             -command do_about
 ++}
 ++
 ++set browser {}
 ++catch {set browser $repo_config(instaweb.browser)}
 ++set doc_path [file dirname [gitexec]]
 ++set doc_path [file join $doc_path Documentation index.html]
 ++
 ++if {[is_Cygwin]} {
 ++     set doc_path [exec cygpath --mixed $doc_path]
 ++}
 ++
 ++if {$browser eq {}} {
 ++     if {[is_MacOSX]} {
 ++             set browser open
 ++     } elseif {[is_Cygwin]} {
 ++             set program_files [file dirname [exec cygpath --windir]]
 ++             set program_files [file join $program_files {Program Files}]
 ++             set firefox [file join $program_files {Mozilla Firefox} firefox.exe]
 ++             set ie [file join $program_files {Internet Explorer} IEXPLORE.EXE]
 ++             if {[file exists $firefox]} {
 ++                     set browser $firefox
 ++             } elseif {[file exists $ie]} {
 ++                     set browser $ie
 ++             }
 ++             unset program_files firefox ie
 ++     }
 ++}
 ++
 ++if {[file isfile $doc_path]} {
 ++     set doc_url "file:$doc_path"
 ++} else {
 ++     set doc_url {http://www.kernel.org/pub/software/scm/git/docs/}
 ++}
 ++
 ++if {$browser ne {}} {
 ++     .mbar.help add command -label [mc "Online Documentation"] \
 ++             -command [list exec $browser $doc_url &]
 ++}
 ++unset browser doc_path doc_url
 ++
 ++# -- Standard bindings
 ++#
 ++wm protocol . WM_DELETE_WINDOW do_quit
 ++bind all <$M1B-Key-q> do_quit
 ++bind all <$M1B-Key-Q> do_quit
 ++bind all <$M1B-Key-w> {destroy [winfo toplevel %W]}
 ++bind all <$M1B-Key-W> {destroy [winfo toplevel %W]}
 ++
 ++set subcommand_args {}
 ++proc usage {} {
 ++     puts stderr "usage: $::argv0 $::subcommand $::subcommand_args"
 ++     exit 1
 ++}
 ++
 ++# -- Not a normal commit type invocation?  Do that instead!
 ++#
 ++switch -- $subcommand {
 ++browser -
 ++blame {
-               blame::new $head $path
+++     if {$subcommand eq "blame"} {
+++             set subcommand_args {[--line=<num>] rev? path}
+++     } else {
+++             set subcommand_args {rev? path}
+++     }
 ++     if {$argv eq {}} usage
 ++     set head {}
 ++     set path {}
+++     set jump_spec {}
 ++     set is_path 0
 ++     foreach a $argv {
 ++             if {$is_path || [file exists $_prefix$a]} {
 ++                     if {$path ne {}} usage
 ++                     set path $_prefix$a
 ++                     break
 ++             } elseif {$a eq {--}} {
 ++                     if {$path ne {}} {
 ++                             if {$head ne {}} usage
 ++                             set head $path
 ++                             set path {}
 ++                     }
 ++                     set is_path 1
+++             } elseif {[regexp {^--line=(\d+)$} $a a lnum]} {
+++                     if {$jump_spec ne {} || $head ne {}} usage
+++                     set jump_spec [list $lnum]
 ++             } elseif {$head eq {}} {
 ++                     if {$head ne {}} usage
 ++                     set head $a
 ++                     set is_path 1
 ++             } else {
 ++                     usage
 ++             }
 ++     }
 ++     unset is_path
 ++
 ++     if {$head ne {} && $path eq {}} {
 ++             set path $_prefix$head
 ++             set head {}
 ++     }
 ++
 ++     if {$head eq {}} {
 ++             load_current_branch
 ++     } else {
 ++             if {[regexp {^[0-9a-f]{1,39}$} $head]} {
 ++                     if {[catch {
 ++                                     set head [git rev-parse --verify $head]
 ++                             } err]} {
 ++                             puts stderr $err
 ++                             exit 1
 ++                     }
 ++             }
 ++             set current_branch $head
 ++     }
 ++
 ++     switch -- $subcommand {
 ++     browser {
+++             if {$jump_spec ne {}} usage
 ++             if {$head eq {}} {
 ++                     if {$path ne {} && [file isdirectory $path]} {
 ++                             set head $current_branch
 ++                     } else {
 ++                             set head $path
 ++                             set path {}
 ++                     }
 ++             }
 ++             browser::new $head $path
 ++     }
 ++     blame   {
 ++             if {$head eq {} && ![file exists $path]} {
 ++                     puts stderr [mc "fatal: cannot stat path %s: No such file or directory" $path]
 ++                     exit 1
 ++             }
-       -command do_rescan
+++             blame::new $head $path $jump_spec
 ++     }
 ++     }
 ++     return
 ++}
 ++citool -
 ++gui {
 ++     if {[llength $argv] != 0} {
 ++             puts -nonewline stderr "usage: $argv0"
 ++             if {$subcommand ne {gui}
 ++                     && [file tail $argv0] ne "git-$subcommand"} {
 ++                     puts -nonewline stderr " $subcommand"
 ++             }
 ++             puts stderr {}
 ++             exit 1
 ++     }
 ++     # fall through to setup UI for commits
 ++}
 ++default {
 ++     puts stderr "usage: $argv0 \[{blame|browser|citool}\]"
 ++     exit 1
 ++}
 ++}
 ++
 ++# -- Branch Control
 ++#
 ++frame .branch \
 ++     -borderwidth 1 \
 ++     -relief sunken
 ++label .branch.l1 \
 ++     -text [mc "Current Branch:"] \
 ++     -anchor w \
 ++     -justify left
 ++label .branch.cb \
 ++     -textvariable current_branch \
 ++     -anchor w \
 ++     -justify left
 ++pack .branch.l1 -side left
 ++pack .branch.cb -side left -fill x
 ++pack .branch -side top -fill x
 ++
 ++# -- Main Window Layout
 ++#
 ++panedwindow .vpane -orient horizontal
 ++panedwindow .vpane.files -orient vertical
 ++.vpane add .vpane.files -sticky nsew -height 100 -width 200
 ++pack .vpane -anchor n -side top -fill both -expand 1
 ++
 ++# -- Index File List
 ++#
 ++frame .vpane.files.index -height 100 -width 200
 ++label .vpane.files.index.title -text [mc "Staged Changes (Will Commit)"] \
 ++     -background lightgreen -foreground black
 ++text $ui_index -background white -foreground black \
 ++     -borderwidth 0 \
 ++     -width 20 -height 10 \
 ++     -wrap none \
 ++     -cursor $cursor_ptr \
 ++     -xscrollcommand {.vpane.files.index.sx set} \
 ++     -yscrollcommand {.vpane.files.index.sy set} \
 ++     -state disabled
 ++scrollbar .vpane.files.index.sx -orient h -command [list $ui_index xview]
 ++scrollbar .vpane.files.index.sy -orient v -command [list $ui_index yview]
 ++pack .vpane.files.index.title -side top -fill x
 ++pack .vpane.files.index.sx -side bottom -fill x
 ++pack .vpane.files.index.sy -side right -fill y
 ++pack $ui_index -side left -fill both -expand 1
 ++
 ++# -- Working Directory File List
 ++#
 ++frame .vpane.files.workdir -height 100 -width 200
 ++label .vpane.files.workdir.title -text [mc "Unstaged Changes"] \
 ++     -background lightsalmon -foreground black
 ++text $ui_workdir -background white -foreground black \
 ++     -borderwidth 0 \
 ++     -width 20 -height 10 \
 ++     -wrap none \
 ++     -cursor $cursor_ptr \
 ++     -xscrollcommand {.vpane.files.workdir.sx set} \
 ++     -yscrollcommand {.vpane.files.workdir.sy set} \
 ++     -state disabled
 ++scrollbar .vpane.files.workdir.sx -orient h -command [list $ui_workdir xview]
 ++scrollbar .vpane.files.workdir.sy -orient v -command [list $ui_workdir yview]
 ++pack .vpane.files.workdir.title -side top -fill x
 ++pack .vpane.files.workdir.sx -side bottom -fill x
 ++pack .vpane.files.workdir.sy -side right -fill y
 ++pack $ui_workdir -side left -fill both -expand 1
 ++
 ++.vpane.files add .vpane.files.workdir -sticky nsew
 ++.vpane.files add .vpane.files.index -sticky nsew
 ++
 ++foreach i [list $ui_index $ui_workdir] {
 ++     rmsel_tag $i
 ++     $i tag conf in_diff -background [$i tag cget in_sel -background]
 ++}
 ++unset i
 ++
 ++# -- Diff and Commit Area
 ++#
 ++frame .vpane.lower -height 300 -width 400
 ++frame .vpane.lower.commarea
 ++frame .vpane.lower.diff -relief sunken -borderwidth 1
 ++pack .vpane.lower.diff -fill both -expand 1
 ++pack .vpane.lower.commarea -side bottom -fill x
 ++.vpane add .vpane.lower -sticky nsew
 ++
 ++# -- Commit Area Buttons
 ++#
 ++frame .vpane.lower.commarea.buttons
 ++label .vpane.lower.commarea.buttons.l -text {} \
 ++     -anchor w \
 ++     -justify left
 ++pack .vpane.lower.commarea.buttons.l -side top -fill x
 ++pack .vpane.lower.commarea.buttons -side left -fill y
 ++
 ++button .vpane.lower.commarea.buttons.rescan -text [mc Rescan] \
-  $ctxm add command \
-       -label [mc "Show Less Context"] \
-       -command show_less_context
-  lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
-  $ctxm add command \
-       -label [mc "Show More Context"] \
-       -command show_more_context
-  lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
-  $ctxm add separator
-  $ctxm add command \
-       -label [mc Refresh] \
-       -command reshow_diff
-  lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
-  $ctxm add command \
-       -label [mc Copy] \
-       -command {tk_textCopy $ui_diff}
-  lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
-  $ctxm add command \
-       -label [mc "Select All"] \
-       -command {focus $ui_diff;$ui_diff tag add sel 0.0 end}
-  lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
-  $ctxm add command \
-       -label [mc "Copy All"] \
-       -command {
-               $ui_diff tag add sel 0.0 end
-               tk_textCopy $ui_diff
-               $ui_diff tag remove sel 0.0 end
-       }
-  lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
-  $ctxm add separator
-  $ctxm add command \
-       -label [mc "Decrease Font Size"] \
-       -command {incr_font_size font_diff -1}
-  lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
-  $ctxm add command \
-       -label [mc "Increase Font Size"] \
-       -command {incr_font_size font_diff 1}
-  lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
-  $ctxm add separator
-  $ctxm add command -label [mc "Options..."] \
-       -command do_options
-  proc popup_diff_menu {ctxm x y X Y} {
+++     -command ui_do_rescan
 ++pack .vpane.lower.commarea.buttons.rescan -side top -fill x
 ++lappend disable_on_lock \
 ++     {.vpane.lower.commarea.buttons.rescan conf -state}
 ++
 ++button .vpane.lower.commarea.buttons.incall -text [mc "Stage Changed"] \
 ++     -command do_add_all
 ++pack .vpane.lower.commarea.buttons.incall -side top -fill x
 ++lappend disable_on_lock \
 ++     {.vpane.lower.commarea.buttons.incall conf -state}
 ++
 ++button .vpane.lower.commarea.buttons.signoff -text [mc "Sign Off"] \
 ++     -command do_signoff
 ++pack .vpane.lower.commarea.buttons.signoff -side top -fill x
 ++
 ++button .vpane.lower.commarea.buttons.commit -text [mc Commit@@verb] \
 ++     -command do_commit
 ++pack .vpane.lower.commarea.buttons.commit -side top -fill x
 ++lappend disable_on_lock \
 ++     {.vpane.lower.commarea.buttons.commit conf -state}
 ++
 ++button .vpane.lower.commarea.buttons.push -text [mc Push] \
 ++     -command do_push_anywhere
 ++pack .vpane.lower.commarea.buttons.push -side top -fill x
 ++
 ++# -- Commit Message Buffer
 ++#
 ++frame .vpane.lower.commarea.buffer
 ++frame .vpane.lower.commarea.buffer.header
 ++set ui_comm .vpane.lower.commarea.buffer.t
 ++set ui_coml .vpane.lower.commarea.buffer.header.l
 ++radiobutton .vpane.lower.commarea.buffer.header.new \
 ++     -text [mc "New Commit"] \
 ++     -command do_select_commit_type \
 ++     -variable selected_commit_type \
 ++     -value new
 ++lappend disable_on_lock \
 ++     [list .vpane.lower.commarea.buffer.header.new conf -state]
 ++radiobutton .vpane.lower.commarea.buffer.header.amend \
 ++     -text [mc "Amend Last Commit"] \
 ++     -command do_select_commit_type \
 ++     -variable selected_commit_type \
 ++     -value amend
 ++lappend disable_on_lock \
 ++     [list .vpane.lower.commarea.buffer.header.amend conf -state]
 ++label $ui_coml \
 ++     -anchor w \
 ++     -justify left
 ++proc trace_commit_type {varname args} {
 ++     global ui_coml commit_type
 ++     switch -glob -- $commit_type {
 ++     initial       {set txt [mc "Initial Commit Message:"]}
 ++     amend         {set txt [mc "Amended Commit Message:"]}
 ++     amend-initial {set txt [mc "Amended Initial Commit Message:"]}
 ++     amend-merge   {set txt [mc "Amended Merge Commit Message:"]}
 ++     merge         {set txt [mc "Merge Commit Message:"]}
 ++     *             {set txt [mc "Commit Message:"]}
 ++     }
 ++     $ui_coml conf -text $txt
 ++}
 ++trace add variable commit_type write trace_commit_type
 ++pack $ui_coml -side left -fill x
 ++pack .vpane.lower.commarea.buffer.header.amend -side right
 ++pack .vpane.lower.commarea.buffer.header.new -side right
 ++
 ++text $ui_comm -background white -foreground black \
 ++     -borderwidth 1 \
 ++     -undo true \
 ++     -maxundo 20 \
 ++     -autoseparators true \
 ++     -relief sunken \
 ++     -width $repo_config(gui.commitmsgwidth) -height 9 -wrap none \
 ++     -font font_diff \
 ++     -yscrollcommand {.vpane.lower.commarea.buffer.sby set}
 ++scrollbar .vpane.lower.commarea.buffer.sby \
 ++     -command [list $ui_comm yview]
 ++pack .vpane.lower.commarea.buffer.header -side top -fill x
 ++pack .vpane.lower.commarea.buffer.sby -side right -fill y
 ++pack $ui_comm -side left -fill y
 ++pack .vpane.lower.commarea.buffer -side left -fill y
 ++
 ++# -- Commit Message Buffer Context Menu
 ++#
 ++set ctxm .vpane.lower.commarea.buffer.ctxm
 ++menu $ctxm -tearoff 0
 ++$ctxm add command \
 ++     -label [mc Cut] \
 ++     -command {tk_textCut $ui_comm}
 ++$ctxm add command \
 ++     -label [mc Copy] \
 ++     -command {tk_textCopy $ui_comm}
 ++$ctxm add command \
 ++     -label [mc Paste] \
 ++     -command {tk_textPaste $ui_comm}
 ++$ctxm add command \
 ++     -label [mc Delete] \
 ++     -command {$ui_comm delete sel.first sel.last}
 ++$ctxm add separator
 ++$ctxm add command \
 ++     -label [mc "Select All"] \
 ++     -command {focus $ui_comm;$ui_comm tag add sel 0.0 end}
 ++$ctxm add command \
 ++     -label [mc "Copy All"] \
 ++     -command {
 ++             $ui_comm tag add sel 0.0 end
 ++             tk_textCopy $ui_comm
 ++             $ui_comm tag remove sel 0.0 end
 ++     }
 ++$ctxm add separator
 ++$ctxm add command \
 ++     -label [mc "Sign Off"] \
 ++     -command do_signoff
 ++set ui_comm_ctxm $ctxm
 ++
 ++# -- Diff Header
 ++#
 ++proc trace_current_diff_path {varname args} {
 ++     global current_diff_path diff_actions file_states
 ++     if {$current_diff_path eq {}} {
 ++             set s {}
 ++             set f {}
 ++             set p {}
 ++             set o disabled
 ++     } else {
 ++             set p $current_diff_path
 ++             set s [mapdesc [lindex $file_states($p) 0] $p]
 ++             set f [mc "File:"]
 ++             set p [escape_path $p]
 ++             set o normal
 ++     }
 ++
 ++     .vpane.lower.diff.header.status configure -text $s
 ++     .vpane.lower.diff.header.file configure -text $f
 ++     .vpane.lower.diff.header.path configure -text $p
 ++     foreach w $diff_actions {
 ++             uplevel #0 $w $o
 ++     }
 ++}
 ++trace add variable current_diff_path write trace_current_diff_path
 ++
 ++frame .vpane.lower.diff.header -background gold
 ++label .vpane.lower.diff.header.status \
 ++     -background gold \
 ++     -foreground black \
 ++     -width $max_status_desc \
 ++     -anchor w \
 ++     -justify left
 ++label .vpane.lower.diff.header.file \
 ++     -background gold \
 ++     -foreground black \
 ++     -anchor w \
 ++     -justify left
 ++label .vpane.lower.diff.header.path \
 ++     -background gold \
 ++     -foreground black \
 ++     -anchor w \
 ++     -justify left
 ++pack .vpane.lower.diff.header.status -side left
 ++pack .vpane.lower.diff.header.file -side left
 ++pack .vpane.lower.diff.header.path -fill x
 ++set ctxm .vpane.lower.diff.header.ctxm
 ++menu $ctxm -tearoff 0
 ++$ctxm add command \
 ++     -label [mc Copy] \
 ++     -command {
 ++             clipboard clear
 ++             clipboard append \
 ++                     -format STRING \
 ++                     -type STRING \
 ++                     -- $current_diff_path
 ++     }
 ++lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
 ++bind_button3 .vpane.lower.diff.header.path "tk_popup $ctxm %X %Y"
 ++
 ++# -- Diff Body
 ++#
 ++frame .vpane.lower.diff.body
 ++set ui_diff .vpane.lower.diff.body.t
 ++text $ui_diff -background white -foreground black \
 ++     -borderwidth 0 \
 ++     -width 80 -height 15 -wrap none \
 ++     -font font_diff \
 ++     -xscrollcommand {.vpane.lower.diff.body.sbx set} \
 ++     -yscrollcommand {.vpane.lower.diff.body.sby set} \
 ++     -state disabled
 ++scrollbar .vpane.lower.diff.body.sbx -orient horizontal \
 ++     -command [list $ui_diff xview]
 ++scrollbar .vpane.lower.diff.body.sby -orient vertical \
 ++     -command [list $ui_diff yview]
 ++pack .vpane.lower.diff.body.sbx -side bottom -fill x
 ++pack .vpane.lower.diff.body.sby -side right -fill y
 ++pack $ui_diff -side left -fill both -expand 1
 ++pack .vpane.lower.diff.header -side top -fill x
 ++pack .vpane.lower.diff.body -side bottom -fill both -expand 1
 ++
 ++$ui_diff tag conf d_cr -elide true
 ++$ui_diff tag conf d_@ -foreground blue -font font_diffbold
 ++$ui_diff tag conf d_+ -foreground {#00a000}
 ++$ui_diff tag conf d_- -foreground red
 ++
 ++$ui_diff tag conf d_++ -foreground {#00a000}
 ++$ui_diff tag conf d_-- -foreground red
 ++$ui_diff tag conf d_+s \
 ++     -foreground {#00a000} \
 ++     -background {#e2effa}
 ++$ui_diff tag conf d_-s \
 ++     -foreground red \
 ++     -background {#e2effa}
 ++$ui_diff tag conf d_s+ \
 ++     -foreground {#00a000} \
 ++     -background ivory1
 ++$ui_diff tag conf d_s- \
 ++     -foreground red \
 ++     -background ivory1
 ++
 ++$ui_diff tag conf d<<<<<<< \
 ++     -foreground orange \
 ++     -font font_diffbold
 ++$ui_diff tag conf d======= \
 ++     -foreground orange \
 ++     -font font_diffbold
 ++$ui_diff tag conf d>>>>>>> \
 ++     -foreground orange \
 ++     -font font_diffbold
 ++
 ++$ui_diff tag raise sel
 ++
 ++# -- Diff Body Context Menu
 ++#
+++
+++proc create_common_diff_popup {ctxm} {
+++     $ctxm add command \
+++             -label [mc "Show Less Context"] \
+++             -command show_less_context
+++     lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
+++     $ctxm add command \
+++             -label [mc "Show More Context"] \
+++             -command show_more_context
+++     lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
+++     $ctxm add separator
+++     $ctxm add command \
+++             -label [mc Refresh] \
+++             -command reshow_diff
+++     lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
+++     $ctxm add command \
+++             -label [mc Copy] \
+++             -command {tk_textCopy $ui_diff}
+++     lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
+++     $ctxm add command \
+++             -label [mc "Select All"] \
+++             -command {focus $ui_diff;$ui_diff tag add sel 0.0 end}
+++     lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
+++     $ctxm add command \
+++             -label [mc "Copy All"] \
+++             -command {
+++                     $ui_diff tag add sel 0.0 end
+++                     tk_textCopy $ui_diff
+++                     $ui_diff tag remove sel 0.0 end
+++             }
+++     lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
+++     $ctxm add separator
+++     $ctxm add command \
+++             -label [mc "Decrease Font Size"] \
+++             -command {incr_font_size font_diff -1}
+++     lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
+++     $ctxm add command \
+++             -label [mc "Increase Font Size"] \
+++             -command {incr_font_size font_diff 1}
+++     lappend diff_actions [list $ctxm entryconf [$ctxm index last] -state]
+++     $ctxm add separator
+++     $ctxm add command -label [mc "Options..."] \
+++             -command do_options
+++}
+++
 ++set ctxm .vpane.lower.diff.body.ctxm
 ++menu $ctxm -tearoff 0
 ++$ctxm add command \
 ++     -label [mc "Apply/Reverse Hunk"] \
 ++     -command {apply_hunk $cursorX $cursorY}
 ++set ui_diff_applyhunk [$ctxm index last]
 ++lappend diff_actions [list $ctxm entryconf $ui_diff_applyhunk -state]
 ++$ctxm add command \
 ++     -label [mc "Apply/Reverse Line"] \
 ++     -command {apply_line $cursorX $cursorY; do_rescan}
 ++set ui_diff_applyline [$ctxm index last]
 ++lappend diff_actions [list $ctxm entryconf $ui_diff_applyline -state]
 ++$ctxm add separator
-       if {$::ui_index eq $::current_diff_side} {
-               set l [mc "Unstage Hunk From Commit"]
-               set t [mc "Unstage Line From Commit"]
+++create_common_diff_popup $ctxm
+++
+++set ctxmmg .vpane.lower.diff.body.ctxmmg
+++menu $ctxmmg -tearoff 0
+++$ctxmmg add command \
+++     -label [mc "Run Merge Tool"] \
+++     -command {merge_resolve_tool}
+++lappend diff_actions [list $ctxmmg entryconf [$ctxmmg index last] -state]
+++$ctxmmg add separator
+++$ctxmmg add command \
+++     -label [mc "Use Remote Version"] \
+++     -command {merge_resolve_one 3}
+++lappend diff_actions [list $ctxmmg entryconf [$ctxmmg index last] -state]
+++$ctxmmg add command \
+++     -label [mc "Use Local Version"] \
+++     -command {merge_resolve_one 2}
+++lappend diff_actions [list $ctxmmg entryconf [$ctxmmg index last] -state]
+++$ctxmmg add command \
+++     -label [mc "Revert To Base"] \
+++     -command {merge_resolve_one 1}
+++lappend diff_actions [list $ctxmmg entryconf [$ctxmmg index last] -state]
+++$ctxmmg add separator
+++create_common_diff_popup $ctxmmg
+++
+++proc popup_diff_menu {ctxm ctxmmg x y X Y} {
 ++     global current_diff_path file_states
 ++     set ::cursorX $x
 ++     set ::cursorY $y
-               set l [mc "Stage Hunk For Commit"]
-               set t [mc "Stage Line For Commit"]
-       }
-       if {$::is_3way_diff
-               || $current_diff_path eq {}
-               || ![info exists file_states($current_diff_path)]
-               || {_O} eq [lindex $file_states($current_diff_path) 0]} {
-               set s disabled
+++     if {[info exists file_states($current_diff_path)]} {
+++             set state [lindex $file_states($current_diff_path) 0]
 ++     } else {
-               set s normal
+++             set state {__}
+++     }
+++     if {[string first {U} $state] >= 0} {
+++             tk_popup $ctxmmg $X $Y
 ++     } else {
-       $ctxm entryconf $::ui_diff_applyhunk -state $s -label $l
-       $ctxm entryconf $::ui_diff_applyline -state $s -label $t
-       tk_popup $ctxm $X $Y
+++             if {$::ui_index eq $::current_diff_side} {
+++                     set l [mc "Unstage Hunk From Commit"]
+++                     set t [mc "Unstage Line From Commit"]
+++             } else {
+++                     set l [mc "Stage Hunk For Commit"]
+++                     set t [mc "Stage Line For Commit"]
+++             }
+++             if {$::is_3way_diff
+++                     || $current_diff_path eq {}
+++                     || {__} eq $state
+++                     || {_O} eq $state
+++                     || {_T} eq $state
+++                     || {T_} eq $state} {
+++                     set s disabled
+++             } else {
+++                     set s normal
+++             }
+++             $ctxm entryconf $::ui_diff_applyhunk -state $s -label $l
+++             $ctxm entryconf $::ui_diff_applyline -state $s -label $t
+++             tk_popup $ctxm $X $Y
 ++     }
-  bind_button3 $ui_diff [list popup_diff_menu $ctxm %x %y %X %Y]
 ++}
-  bind .   <Key-F5>     do_rescan
-  bind .   <$M1B-Key-r> do_rescan
-  bind .   <$M1B-Key-R> do_rescan
+++bind_button3 $ui_diff [list popup_diff_menu $ctxm $ctxmmg %x %y %X %Y]
 ++
 ++# -- Status Bar
 ++#
 ++set main_status [::status_bar::new .status]
 ++pack .status -anchor w -side bottom -fill x
 ++$main_status show [mc "Initializing..."]
 ++
 ++# -- Load geometry
 ++#
 ++catch {
 ++set gm $repo_config(gui.geometry)
 ++wm geometry . [lindex $gm 0]
 ++.vpane sash place 0 \
 ++     [lindex $gm 1] \
 ++     [lindex [.vpane sash coord 0] 1]
 ++.vpane.files sash place 0 \
 ++     [lindex [.vpane.files sash coord 0] 0] \
 ++     [lindex $gm 2]
 ++unset gm
 ++}
 ++
 ++# -- Key Bindings
 ++#
 ++bind $ui_comm <$M1B-Key-Return> {do_commit;break}
 ++bind $ui_comm <$M1B-Key-t> {do_add_selection;break}
 ++bind $ui_comm <$M1B-Key-T> {do_add_selection;break}
 ++bind $ui_comm <$M1B-Key-i> {do_add_all;break}
 ++bind $ui_comm <$M1B-Key-I> {do_add_all;break}
 ++bind $ui_comm <$M1B-Key-x> {tk_textCut %W;break}
 ++bind $ui_comm <$M1B-Key-X> {tk_textCut %W;break}
 ++bind $ui_comm <$M1B-Key-c> {tk_textCopy %W;break}
 ++bind $ui_comm <$M1B-Key-C> {tk_textCopy %W;break}
 ++bind $ui_comm <$M1B-Key-v> {tk_textPaste %W; %W see insert; break}
 ++bind $ui_comm <$M1B-Key-V> {tk_textPaste %W; %W see insert; break}
 ++bind $ui_comm <$M1B-Key-a> {%W tag add sel 0.0 end;break}
 ++bind $ui_comm <$M1B-Key-A> {%W tag add sel 0.0 end;break}
 ++bind $ui_comm <$M1B-Key-minus> {show_less_context;break}
 ++bind $ui_comm <$M1B-Key-KP_Subtract> {show_less_context;break}
 ++bind $ui_comm <$M1B-Key-equal> {show_more_context;break}
 ++bind $ui_comm <$M1B-Key-plus> {show_more_context;break}
 ++bind $ui_comm <$M1B-Key-KP_Add> {show_more_context;break}
 ++
 ++bind $ui_diff <$M1B-Key-x> {tk_textCopy %W;break}
 ++bind $ui_diff <$M1B-Key-X> {tk_textCopy %W;break}
 ++bind $ui_diff <$M1B-Key-c> {tk_textCopy %W;break}
 ++bind $ui_diff <$M1B-Key-C> {tk_textCopy %W;break}
 ++bind $ui_diff <$M1B-Key-v> {break}
 ++bind $ui_diff <$M1B-Key-V> {break}
 ++bind $ui_diff <$M1B-Key-a> {%W tag add sel 0.0 end;break}
 ++bind $ui_diff <$M1B-Key-A> {%W tag add sel 0.0 end;break}
 ++bind $ui_diff <Key-Up>     {catch {%W yview scroll -1 units};break}
 ++bind $ui_diff <Key-Down>   {catch {%W yview scroll  1 units};break}
 ++bind $ui_diff <Key-Left>   {catch {%W xview scroll -1 units};break}
 ++bind $ui_diff <Key-Right>  {catch {%W xview scroll  1 units};break}
 ++bind $ui_diff <Key-k>         {catch {%W yview scroll -1 units};break}
 ++bind $ui_diff <Key-j>         {catch {%W yview scroll  1 units};break}
 ++bind $ui_diff <Key-h>         {catch {%W xview scroll -1 units};break}
 ++bind $ui_diff <Key-l>         {catch {%W xview scroll  1 units};break}
 ++bind $ui_diff <Control-Key-b> {catch {%W yview scroll -1 pages};break}
 ++bind $ui_diff <Control-Key-f> {catch {%W yview scroll  1 pages};break}
 ++bind $ui_diff <Button-1>   {focus %W}
 ++
 ++if {[is_enabled branch]} {
 ++     bind . <$M1B-Key-n> branch_create::dialog
 ++     bind . <$M1B-Key-N> branch_create::dialog
 ++     bind . <$M1B-Key-o> branch_checkout::dialog
 ++     bind . <$M1B-Key-O> branch_checkout::dialog
 ++     bind . <$M1B-Key-m> merge::dialog
 ++     bind . <$M1B-Key-M> merge::dialog
 ++}
 ++if {[is_enabled transport]} {
 ++     bind . <$M1B-Key-p> do_push_anywhere
 ++     bind . <$M1B-Key-P> do_push_anywhere
 ++}
 ++
+++bind .   <Key-F5>     ui_do_rescan
+++bind .   <$M1B-Key-r> ui_do_rescan
+++bind .   <$M1B-Key-R> ui_do_rescan
 ++bind .   <$M1B-Key-s> do_signoff
 ++bind .   <$M1B-Key-S> do_signoff
 ++bind .   <$M1B-Key-t> do_add_selection
 ++bind .   <$M1B-Key-T> do_add_selection
 ++bind .   <$M1B-Key-i> do_add_all
 ++bind .   <$M1B-Key-I> do_add_all
 ++bind .   <$M1B-Key-minus> {show_less_context;break}
 ++bind .   <$M1B-Key-KP_Subtract> {show_less_context;break}
 ++bind .   <$M1B-Key-equal> {show_more_context;break}
 ++bind .   <$M1B-Key-plus> {show_more_context;break}
 ++bind .   <$M1B-Key-KP_Add> {show_more_context;break}
 ++bind .   <$M1B-Key-Return> do_commit
 ++foreach i [list $ui_index $ui_workdir] {
 ++     bind $i <Button-1>       "toggle_or_diff         $i %x %y; break"
 ++     bind $i <$M1B-Button-1>  "add_one_to_selection   $i %x %y; break"
 ++     bind $i <Shift-Button-1> "add_range_to_selection $i %x %y; break"
 ++}
 ++unset i
 ++
 ++set file_lists($ui_index) [list]
 ++set file_lists($ui_workdir) [list]
 ++
 ++wm title . "[appname] ([reponame]) [file normalize [file dirname [gitdir]]]"
 ++focus -force $ui_comm
 ++
 ++# -- Warn the user about environmental problems.  Cygwin's Tcl
 ++#    does *not* pass its env array onto any processes it spawns.
 ++#    This means that git processes get none of our environment.
 ++#
 ++if {[is_Cygwin]} {
 ++     set ignored_env 0
 ++     set suggest_user {}
 ++     set msg [mc "Possible environment issues exist.
 ++
 ++The following environment variables are probably
 ++going to be ignored by any Git subprocess run
 ++by %s:
 ++
 ++" [appname]]
 ++     foreach name [array names env] {
 ++             switch -regexp -- $name {
 ++             {^GIT_INDEX_FILE$} -
 ++             {^GIT_OBJECT_DIRECTORY$} -
 ++             {^GIT_ALTERNATE_OBJECT_DIRECTORIES$} -
 ++             {^GIT_DIFF_OPTS$} -
 ++             {^GIT_EXTERNAL_DIFF$} -
 ++             {^GIT_PAGER$} -
 ++             {^GIT_TRACE$} -
 ++             {^GIT_CONFIG$} -
 ++             {^GIT_CONFIG_LOCAL$} -
 ++             {^GIT_(AUTHOR|COMMITTER)_DATE$} {
 ++                     append msg " - $name\n"
 ++                     incr ignored_env
 ++             }
 ++             {^GIT_(AUTHOR|COMMITTER)_(NAME|EMAIL)$} {
 ++                     append msg " - $name\n"
 ++                     incr ignored_env
 ++                     set suggest_user $name
 ++             }
 ++             }
 ++     }
 ++     if {$ignored_env > 0} {
 ++             append msg [mc "
 ++This is due to a known issue with the
 ++Tcl binary distributed by Cygwin."]
 ++
 ++             if {$suggest_user ne {}} {
 ++                     append msg [mc "
 ++
 ++A good replacement for %s
 ++is placing values for the user.name and
 ++user.email settings into your personal
 ++~/.gitconfig file.
 ++" $suggest_user]
 ++             }
 ++             warn_popup $msg
 ++     }
 ++     unset ignored_env msg suggest_user name
 ++}
 ++
 ++# -- Only initialize complex UI if we are going to stay running.
 ++#
 ++if {[is_enabled transport]} {
 ++     load_all_remotes
 ++
 ++     set n [.mbar.remote index end]
 ++     populate_push_menu
 ++     populate_fetch_menu
 ++     set n [expr {[.mbar.remote index end] - $n}]
 ++     if {$n > 0} {
 ++             if {[.mbar.remote type 0] eq "tearoff"} { incr n }
 ++             .mbar.remote insert $n separator
 ++     }
 ++     unset n
 ++}
 ++
 ++if {[winfo exists $ui_comm]} {
 ++     set GITGUI_BCK_exists [load_message GITGUI_BCK]
 ++
 ++     # -- If both our backup and message files exist use the
 ++     #    newer of the two files to initialize the buffer.
 ++     #
 ++     if {$GITGUI_BCK_exists} {
 ++             set m [gitdir GITGUI_MSG]
 ++             if {[file isfile $m]} {
 ++                     if {[file mtime [gitdir GITGUI_BCK]] > [file mtime $m]} {
 ++                             catch {file delete [gitdir GITGUI_MSG]}
 ++                     } else {
 ++                             $ui_comm delete 0.0 end
 ++                             $ui_comm edit reset
 ++                             $ui_comm edit modified false
 ++                             catch {file delete [gitdir GITGUI_BCK]}
 ++                             set GITGUI_BCK_exists 0
 ++                     }
 ++             }
 ++             unset m
 ++     }
 ++
 ++     proc backup_commit_buffer {} {
 ++             global ui_comm GITGUI_BCK_exists
 ++
 ++             set m [$ui_comm edit modified]
 ++             if {$m || $GITGUI_BCK_exists} {
 ++                     set msg [string trim [$ui_comm get 0.0 end]]
 ++                     regsub -all -line {[ \r\t]+$} $msg {} msg
 ++
 ++                     if {$msg eq {}} {
 ++                             if {$GITGUI_BCK_exists} {
 ++                                     catch {file delete [gitdir GITGUI_BCK]}
 ++                                     set GITGUI_BCK_exists 0
 ++                             }
 ++                     } elseif {$m} {
 ++                             catch {
 ++                                     set fd [open [gitdir GITGUI_BCK] w]
 ++                                     puts -nonewline $fd $msg
 ++                                     close $fd
 ++                                     set GITGUI_BCK_exists 1
 ++                             }
 ++                     }
 ++
 ++                     $ui_comm edit modified false
 ++             }
 ++
 ++             set ::GITGUI_BCK_i [after 2000 backup_commit_buffer]
 ++     }
 ++
 ++     backup_commit_buffer
 ++
 ++     # -- If the user has aspell available we can drive it
 ++     #    in pipe mode to spellcheck the commit message.
 ++     #
 ++     set spell_cmd [list |]
 ++     set spell_dict [get_config gui.spellingdictionary]
 ++     lappend spell_cmd aspell
 ++     if {$spell_dict ne {}} {
 ++             lappend spell_cmd --master=$spell_dict
 ++     }
 ++     lappend spell_cmd --mode=none
 ++     lappend spell_cmd --encoding=utf-8
 ++     lappend spell_cmd pipe
 ++     if {$spell_dict eq {none}
 ++      || [catch {set spell_fd [open $spell_cmd r+]} spell_err]} {
 ++             bind_button3 $ui_comm [list tk_popup $ui_comm_ctxm %X %Y]
 ++     } else {
 ++             set ui_comm_spell [spellcheck::init \
 ++                     $spell_fd \
 ++                     $ui_comm \
 ++                     $ui_comm_ctxm \
 ++             ]
 ++     }
 ++     unset -nocomplain spell_cmd spell_fd spell_err spell_dict
 ++}
 ++
 ++lock_index begin-read
 ++if {![winfo ismapped .]} {
 ++     wm deiconify .
 ++}
 ++after 1 do_rescan
 ++if {[is_enabled multicommit]} {
 ++     after 1000 hint_gc
 ++}
index b6e42cbc8fe0a49c301335f78cc2941bd9d59870,0000000000000000000000000000000000000000,0000000000000000000000000000000000000000..827c85d67f0b0f18d22b1399624e6819cf93f23f
mode 100644,000000,000000..100644
--- /dev/null
--- /dev/null
@@@@ -1,1076 -1,0 -1,0 +1,1212 @@@@
-  constructor new {i_commit i_path} {
 ++# git-gui blame viewer
 ++# Copyright (C) 2006, 2007 Shawn Pearce
 ++
 ++class blame {
 ++
 ++image create photo ::blame::img_back_arrow -data {R0lGODlhGAAYAIUAAPwCBEzKXFTSZIz+nGzmhGzqfGTidIT+nEzGXHTqhGzmfGzifFzadETCVES+VARWDFzWbHzyjAReDGTadFTOZDSyRDyyTCymPARaFGTedFzSbDy2TCyqRCyqPARaDAyCHES6VDy6VCyiPAR6HCSeNByWLARyFARiDARqFGTifARiFAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACH5BAEAAAAALAAAAAAYABgAAAajQIBwSCwaj8ikcsk0BppJwRPqHEypQwHBis0WDAdEFyBIKBaMAKLBdjQeSkFBYTBAIvgEoS6JmhUTEwIUDQ4VFhcMGEhyCgoZExoUaxsWHB0THkgfAXUGAhoBDSAVFR0XBnCbDRmgog0hpSIiDJpJIyEQhBUcJCIlwA22SSYVogknEg8eD82qSigdDSknY0IqJQXPYxIl1dZCGNvWw+Dm510GQQAh/mhDcmVhdGVkIGJ5IEJNUFRvR0lGIFBybyB2ZXJzaW9uIDIuNQ0KqSBEZXZlbENvciAxOTk3LDE5OTguIEFsbCByaWdodHMgcmVzZXJ2ZWQuDQpodHRwOi8vd3d3LmRldmVsY29yLmNvbQA7}
 ++
 ++# Persistant data (survives loads)
 ++#
 ++field history {}; # viewer history: {commit path}
 ++field header    ; # array commit,key -> header field
 ++
 ++# Tk UI control paths
 ++#
 ++field w          ; # top window in this viewer
 ++field w_back     ; # our back button
 ++field w_path     ; # label showing the current file path
 ++field w_columns  ; # list of all column widgets in the viewer
 ++field w_line     ; # text column: all line numbers
 ++field w_amov     ; # text column: annotations + move tracking
 ++field w_asim     ; # text column: annotations (simple computation)
 ++field w_file     ; # text column: actual file data
 ++field w_cviewer  ; # pane showing commit message
 ++field status     ; # status mega-widget instance
 ++field old_height ; # last known height of $w.file_pane
 ++
 ++# Tk UI colors
 ++#
 ++variable active_color #c0edc5
 ++variable group_colors {
 ++     #d6d6d6
 ++     #e1e1e1
 ++     #ececec
 ++}
 ++
 ++# Current blame data; cleared/reset on each load
 ++#
 ++field commit               ; # input commit to blame
 ++field path                 ; # input filename to view in $commit
 ++
 ++field current_fd        {} ; # background process running
 ++field highlight_line    -1 ; # current line selected
 ++field highlight_column  {} ; # current commit column selected
 ++field highlight_commit  {} ; # sha1 of commit selected
 ++
 ++field total_lines       0  ; # total length of file
 ++field blame_lines       0  ; # number of lines computed
 ++field amov_data            ; # list of {commit origfile origline}
 ++field asim_data            ; # list of {commit origfile origline}
 ++
 ++field r_commit             ; # commit currently being parsed
 ++field r_orig_line          ; # original line number
 ++field r_final_line         ; # final line number
 ++field r_line_count         ; # lines in this region
 ++
 ++field tooltip_wm        {} ; # Current tooltip toplevel, if open
 ++field tooltip_t         {} ; # Text widget in $tooltip_wm
 ++field tooltip_timer     {} ; # Current timer event for our tooltip
 ++field tooltip_commit    {} ; # Commit(s) in tooltip
 ++
-       _load $this {}
+++constructor new {i_commit i_path i_jump} {
 ++     global cursor_ptr
 ++     variable active_color
 ++     variable group_colors
 ++
 ++     set commit $i_commit
 ++     set path   $i_path
 ++
 ++     make_toplevel top w
 ++     wm title $top [append "[appname] ([reponame]): " [mc "File Viewer"]]
 ++
 ++     frame $w.header -background gold
 ++     label $w.header.commit_l \
 ++             -text [mc "Commit:"] \
 ++             -background gold \
 ++             -foreground black \
 ++             -anchor w \
 ++             -justify left
 ++     set w_back $w.header.commit_b
 ++     label $w_back \
 ++             -image ::blame::img_back_arrow \
 ++             -borderwidth 0 \
 ++             -relief flat \
 ++             -state disabled \
 ++             -background gold \
 ++             -foreground black \
 ++             -activebackground gold
 ++     bind $w_back <Button-1> "
 ++             if {\[$w_back cget -state\] eq {normal}} {
 ++                     [cb _history_menu]
 ++             }
 ++             "
 ++     label $w.header.commit \
 ++             -textvariable @commit \
 ++             -background gold \
 ++             -foreground black \
 ++             -anchor w \
 ++             -justify left
 ++     label $w.header.path_l \
 ++             -text [mc "File:"] \
 ++             -background gold \
 ++             -foreground black \
 ++             -anchor w \
 ++             -justify left
 ++     set w_path $w.header.path
 ++     label $w_path \
 ++             -background gold \
 ++             -foreground black \
 ++             -anchor w \
 ++             -justify left
 ++     pack $w.header.commit_l -side left
 ++     pack $w_back -side left
 ++     pack $w.header.commit -side left
 ++     pack $w_path -fill x -side right
 ++     pack $w.header.path_l -side right
 ++
 ++     panedwindow $w.file_pane -orient vertical
 ++     frame $w.file_pane.out
 ++     frame $w.file_pane.cm
 ++     $w.file_pane add $w.file_pane.out \
 ++             -sticky nsew \
 ++             -minsize 100 \
 ++             -height 100 \
 ++             -width 100
 ++     $w.file_pane add $w.file_pane.cm \
 ++             -sticky nsew \
 ++             -minsize 25 \
 ++             -height 25 \
 ++             -width 100
 ++
 ++     set w_line $w.file_pane.out.linenumber_t
 ++     text $w_line \
 ++             -takefocus 0 \
 ++             -highlightthickness 0 \
 ++             -padx 0 -pady 0 \
 ++             -background white \
 ++             -foreground black \
 ++             -borderwidth 0 \
 ++             -state disabled \
 ++             -wrap none \
 ++             -height 40 \
 ++             -width 6 \
 ++             -font font_diff
 ++     $w_line tag conf linenumber -justify right -rmargin 5
 ++
 ++     set w_amov $w.file_pane.out.amove_t
 ++     text $w_amov \
 ++             -takefocus 0 \
 ++             -highlightthickness 0 \
 ++             -padx 0 -pady 0 \
 ++             -background white \
 ++             -foreground black \
 ++             -borderwidth 0 \
 ++             -state disabled \
 ++             -wrap none \
 ++             -height 40 \
 ++             -width 5 \
 ++             -font font_diff
 ++     $w_amov tag conf author_abbr -justify right -rmargin 5
 ++     $w_amov tag conf curr_commit
 ++     $w_amov tag conf prior_commit -foreground blue -underline 1
 ++     $w_amov tag bind prior_commit \
 ++             <Button-1> \
 ++             "[cb _load_commit $w_amov @amov_data @%x,%y];break"
 ++
 ++     set w_asim $w.file_pane.out.asimple_t
 ++     text $w_asim \
 ++             -takefocus 0 \
 ++             -highlightthickness 0 \
 ++             -padx 0 -pady 0 \
 ++             -background white \
 ++             -foreground black \
 ++             -borderwidth 0 \
 ++             -state disabled \
 ++             -wrap none \
 ++             -height 40 \
 ++             -width 4 \
 ++             -font font_diff
 ++     $w_asim tag conf author_abbr -justify right
 ++     $w_asim tag conf curr_commit
 ++     $w_asim tag conf prior_commit -foreground blue -underline 1
 ++     $w_asim tag bind prior_commit \
 ++             <Button-1> \
 ++             "[cb _load_commit $w_asim @asim_data @%x,%y];break"
 ++
 ++     set w_file $w.file_pane.out.file_t
 ++     text $w_file \
 ++             -takefocus 0 \
 ++             -highlightthickness 0 \
 ++             -padx 0 -pady 0 \
 ++             -background white \
 ++             -foreground black \
 ++             -borderwidth 0 \
 ++             -state disabled \
 ++             -wrap none \
 ++             -height 40 \
 ++             -width 80 \
 ++             -xscrollcommand [list $w.file_pane.out.sbx set] \
 ++             -font font_diff
 ++
 ++     set w_columns [list $w_amov $w_asim $w_line $w_file]
 ++
 ++     scrollbar $w.file_pane.out.sbx \
 ++             -orient h \
 ++             -command [list $w_file xview]
 ++     scrollbar $w.file_pane.out.sby \
 ++             -orient v \
 ++             -command [list scrollbar2many $w_columns yview]
 ++     eval grid $w_columns $w.file_pane.out.sby -sticky nsew
 ++     grid conf \
 ++             $w.file_pane.out.sbx \
 ++             -column [expr {[llength $w_columns] - 1}] \
 ++             -sticky we
 ++     grid columnconfigure \
 ++             $w.file_pane.out \
 ++             [expr {[llength $w_columns] - 1}] \
 ++             -weight 1
 ++     grid rowconfigure $w.file_pane.out 0 -weight 1
 ++
 ++     set w_cviewer $w.file_pane.cm.t
 ++     text $w_cviewer \
 ++             -background white \
 ++             -foreground black \
 ++             -borderwidth 0 \
 ++             -state disabled \
 ++             -wrap none \
 ++             -height 10 \
 ++             -width 80 \
 ++             -xscrollcommand [list $w.file_pane.cm.sbx set] \
 ++             -yscrollcommand [list $w.file_pane.cm.sby set] \
 ++             -font font_diff
 ++     $w_cviewer tag conf still_loading \
 ++             -font font_uiitalic \
 ++             -justify center
 ++     $w_cviewer tag conf header_key \
 ++             -tabs {3c} \
 ++             -background $active_color \
 ++             -font font_uibold
 ++     $w_cviewer tag conf header_val \
 ++             -background $active_color \
 ++             -font font_ui
 ++     $w_cviewer tag raise sel
 ++     scrollbar $w.file_pane.cm.sbx \
 ++             -orient h \
 ++             -command [list $w_cviewer xview]
 ++     scrollbar $w.file_pane.cm.sby \
 ++             -orient v \
 ++             -command [list $w_cviewer yview]
 ++     pack $w.file_pane.cm.sby -side right -fill y
 ++     pack $w.file_pane.cm.sbx -side bottom -fill x
 ++     pack $w_cviewer -expand 1 -fill both
 ++
 ++     set status [::status_bar::new $w.status]
 ++
 ++     menu $w.ctxm -tearoff 0
 ++     $w.ctxm add command \
 ++             -label [mc "Copy Commit"] \
 ++             -command [cb _copycommit]
 ++     $w.ctxm add command \
 ++             -label [mc "Do Full Copy Detection"] \
 ++             -command [cb _fullcopyblame]
+++     $w.ctxm add command \
+++             -label [mc "Show History Context"] \
+++             -command [cb _gitkcommit]
+++     $w.ctxm add command \
+++             -label [mc "Blame Parent Commit"] \
+++             -command [cb _blameparent]
 ++
 ++     foreach i $w_columns {
 ++             for {set g 0} {$g < [llength $group_colors]} {incr g} {
 ++                     $i tag conf color$g -background [lindex $group_colors $g]
 ++             }
 ++
 ++             $i conf -cursor $cursor_ptr
 ++             $i conf -yscrollcommand [list many2scrollbar \
 ++                     $w_columns yview $w.file_pane.out.sby]
 ++             bind $i <Button-1> "
 ++                     [cb _hide_tooltip]
 ++                     [cb _click $i @%x,%y]
 ++                     focus $i
 ++             "
 ++             bind $i <Any-Motion>  [cb _show_tooltip $i @%x,%y]
 ++             bind $i <Any-Enter>   [cb _hide_tooltip]
 ++             bind $i <Any-Leave>   [cb _hide_tooltip]
 ++             bind_button3 $i "
 ++                     [cb _hide_tooltip]
 ++                     set cursorX %x
 ++                     set cursorY %y
 ++                     set cursorW %W
 ++                     tk_popup $w.ctxm %X %Y
 ++             "
 ++             bind $i <Shift-Tab> "[list focus $w_cviewer];break"
 ++             bind $i <Tab>       "[list focus $w_cviewer];break"
 ++     }
 ++
 ++     foreach i [concat $w_columns $w_cviewer] {
 ++             bind $i <Key-Up>        {catch {%W yview scroll -1 units};break}
 ++             bind $i <Key-Down>      {catch {%W yview scroll  1 units};break}
 ++             bind $i <Key-Left>      {catch {%W xview scroll -1 units};break}
 ++             bind $i <Key-Right>     {catch {%W xview scroll  1 units};break}
 ++             bind $i <Key-k>         {catch {%W yview scroll -1 units};break}
 ++             bind $i <Key-j>         {catch {%W yview scroll  1 units};break}
 ++             bind $i <Key-h>         {catch {%W xview scroll -1 units};break}
 ++             bind $i <Key-l>         {catch {%W xview scroll  1 units};break}
 ++             bind $i <Control-Key-b> {catch {%W yview scroll -1 pages};break}
 ++             bind $i <Control-Key-f> {catch {%W yview scroll  1 pages};break}
 ++     }
 ++
 ++     bind $w_cviewer <Shift-Tab> "[list focus $w_file];break"
 ++     bind $w_cviewer <Tab>       "[list focus $w_file];break"
 ++     bind $w_cviewer <Button-1> [list focus $w_cviewer]
 ++     bind $w_file    <Visibility> [list focus $w_file]
 ++
 ++     grid configure $w.header -sticky ew
 ++     grid configure $w.file_pane -sticky nsew
 ++     grid configure $w.status -sticky ew
 ++     grid columnconfigure $top 0 -weight 1
 ++     grid rowconfigure $top 0 -weight 0
 ++     grid rowconfigure $top 1 -weight 1
 ++     grid rowconfigure $top 2 -weight 0
 ++
 ++     set req_w [winfo reqwidth  $top]
 ++     set req_h [winfo reqheight $top]
 ++     set scr_h [expr {[winfo screenheight $top] - 100}]
 ++     if {$req_w < 600} {set req_w 600}
 ++     if {$req_h < $scr_h} {set req_h $scr_h}
 ++     set g "${req_w}x${req_h}"
 ++     wm geometry $top $g
 ++     update
 ++
 ++     set old_height [winfo height $w.file_pane]
 ++     $w.file_pane sash place 0 \
 ++             [lindex [$w.file_pane sash coord 0] 0] \
 ++             [expr {int($old_height * 0.70)}]
 ++     bind $w.file_pane <Configure> \
 ++     "if {{$w.file_pane} eq {%W}} {[cb _resize %h]}"
 ++
 ++     wm protocol $top WM_DELETE_WINDOW "destroy $top"
 ++     bind $top <Destroy> [cb _kill]
 ++
-               lappend history [list \
-                       $commit $path \
-                       $highlight_column \
-                       $highlight_line \
-                       [lindex [$w_file xview] 0] \
-                       [lindex [$w_file yview] 0] \
-                       ]
-               set commit [lindex $dat 0]
-               set path   [lindex $dat 1]
-               _load $this [list [lindex $dat 2]]
+++     _load $this $i_jump
 ++}
 ++
 ++method _kill {} {
 ++     if {$current_fd ne {}} {
 ++             kill_file_process $current_fd
 ++             catch {close $current_fd}
 ++             set current_fd {}
 ++     }
 ++}
 ++
 ++method _load {jump} {
 ++     variable group_colors
 ++
 ++     _hide_tooltip $this
 ++
 ++     if {$total_lines != 0 || $current_fd ne {}} {
 ++             _kill $this
 ++
 ++             foreach i $w_columns {
 ++                     $i conf -state normal
 ++                     $i delete 0.0 end
 ++                     foreach g [$i tag names] {
 ++                             if {[regexp {^g[0-9a-f]{40}$} $g]} {
 ++                                     $i tag delete $g
 ++                             }
 ++                     }
 ++                     $i conf -state disabled
 ++             }
 ++
 ++             $w_cviewer conf -state normal
 ++             $w_cviewer delete 0.0 end
 ++             $w_cviewer conf -state disabled
 ++
 ++             set highlight_line -1
 ++             set highlight_column {}
 ++             set highlight_commit {}
 ++             set total_lines 0
 ++     }
 ++
 ++     if {$history eq {}} {
 ++             $w_back conf -state disabled
 ++     } else {
 ++             $w_back conf -state normal
 ++     }
 ++
 ++     # Index 0 is always empty.  There is never line 0 as
 ++     # we use only 1 based lines, as that matches both with
 ++     # git-blame output and with Tk's text widget.
 ++     #
 ++     set amov_data [list [list]]
 ++     set asim_data [list [list]]
 ++
 ++     $status show [mc "Reading %s..." "$commit:[escape_path $path]"]
 ++     $w_path conf -text [escape_path $path]
 ++     if {$commit eq {}} {
 ++             set fd [open $path r]
 ++             fconfigure $fd -eofchar {}
 ++     } else {
 ++             set fd [git_read cat-file blob "$commit:$path"]
 ++     }
 ++     fconfigure $fd -blocking 0 -translation lf -encoding binary
 ++     fileevent $fd readable [cb _read_file $fd $jump]
 ++     set current_fd $fd
 ++}
 ++
 ++method _history_menu {} {
 ++     set m $w.backmenu
 ++     if {[winfo exists $m]} {
 ++             $m delete 0 end
 ++     } else {
 ++             menu $m -tearoff 0
 ++     }
 ++
 ++     for {set i [expr {[llength $history] - 1}]
 ++             } {$i >= 0} {incr i -1} {
 ++             set e [lindex $history $i]
 ++             set c [lindex $e 0]
 ++             set f [lindex $e 1]
 ++
 ++             if {[regexp {^[0-9a-f]{40}$} $c]} {
 ++                     set t [string range $c 0 8]...
 ++             } elseif {$c eq {}} {
 ++                     set t {Working Directory}
 ++             } else {
 ++                     set t $c
 ++             }
 ++             if {![catch {set summary $header($c,summary)}]} {
 ++                     append t " $summary"
 ++                     if {[string length $t] > 70} {
 ++                             set t [string range $t 0 66]...
 ++                     }
 ++             }
 ++
 ++             $m add command -label $t -command [cb _goback $i]
 ++     }
 ++     set X [winfo rootx $w_back]
 ++     set Y [expr {[winfo rooty $w_back] + [winfo height $w_back]}]
 ++     tk_popup $m $X $Y
 ++}
 ++
 ++method _goback {i} {
 ++     set dat [lindex $history $i]
 ++     set history [lrange $history 0 [expr {$i - 1}]]
 ++     set commit [lindex $dat 0]
 ++     set path [lindex $dat 1]
 ++     _load $this [lrange $dat 2 5]
 ++}
 ++
 ++method _read_file {fd jump} {
 ++     if {$fd ne $current_fd} {
 ++             catch {close $fd}
 ++             return
 ++     }
 ++
 ++     foreach i $w_columns {$i conf -state normal}
 ++     while {[gets $fd line] >= 0} {
 ++             regsub "\r\$" $line {} line
 ++             incr total_lines
 ++             lappend amov_data {}
 ++             lappend asim_data {}
 ++
 ++             if {$total_lines > 1} {
 ++                     foreach i $w_columns {$i insert end "\n"}
 ++             }
 ++
 ++             $w_line insert end "$total_lines" linenumber
 ++             $w_file insert end "$line"
 ++     }
 ++
 ++     set ln_wc [expr {[string length $total_lines] + 2}]
 ++     if {[$w_line cget -width] < $ln_wc} {
 ++             $w_line conf -width $ln_wc
 ++     }
 ++
 ++     foreach i $w_columns {$i conf -state disabled}
 ++
 ++     if {[eof $fd]} {
 ++             close $fd
 ++
 ++             # If we don't force Tk to update the widgets *right now*
 ++             # none of our jump commands will cause a change in the UI.
 ++             #
 ++             update
 ++
 ++             if {[llength $jump] == 1} {
 ++                     set highlight_line [lindex $jump 0]
 ++                     $w_file see "$highlight_line.0"
 ++             } elseif {[llength $jump] == 4} {
 ++                     set highlight_column [lindex $jump 0]
 ++                     set highlight_line [lindex $jump 1]
 ++                     $w_file xview moveto [lindex $jump 2]
 ++                     $w_file yview moveto [lindex $jump 3]
 ++             }
 ++
 ++             _exec_blame $this $w_asim @asim_data \
 ++                     [list] \
 ++                     [mc "Loading copy/move tracking annotations..."]
 ++     }
 ++} ifdeleted { catch {close $fd} }
 ++
 ++method _exec_blame {cur_w cur_d options cur_s} {
 ++     lappend options --incremental
 ++     if {$commit eq {}} {
 ++             lappend options --contents $path
 ++     } else {
 ++             lappend options $commit
 ++     }
 ++     lappend options -- $path
 ++     set fd [eval git_read --nice blame $options]
 ++     fconfigure $fd -blocking 0 -translation lf -encoding binary
 ++     fileevent $fd readable [cb _read_blame $fd $cur_w $cur_d]
 ++     set current_fd $fd
 ++     set blame_lines 0
 ++
 ++     $status start \
 ++             $cur_s \
 ++             [mc "lines annotated"]
 ++}
 ++
 ++method _read_blame {fd cur_w cur_d} {
 ++     upvar #0 $cur_d line_data
 ++     variable group_colors
 ++
 ++     if {$fd ne $current_fd} {
 ++             catch {close $fd}
 ++             return
 ++     }
 ++
 ++     $cur_w conf -state normal
 ++     while {[gets $fd line] >= 0} {
 ++             if {[regexp {^([a-z0-9]{40}) (\d+) (\d+) (\d+)$} $line line \
 ++                     cmit original_line final_line line_count]} {
 ++                     set r_commit     $cmit
 ++                     set r_orig_line  $original_line
 ++                     set r_final_line $final_line
 ++                     set r_line_count $line_count
 ++             } elseif {[string match {filename *} $line]} {
 ++                     set file [string range $line 9 end]
 ++                     set n    $r_line_count
 ++                     set lno  $r_final_line
 ++                     set oln  $r_orig_line
 ++                     set cmit $r_commit
 ++
 ++                     if {[regexp {^0{40}$} $cmit]} {
 ++                             set commit_abbr work
 ++                             set commit_type curr_commit
 ++                     } elseif {$cmit eq $commit} {
 ++                             set commit_abbr this
 ++                             set commit_type curr_commit
 ++                     } else {
 ++                             set commit_type prior_commit
 ++                             set commit_abbr [string range $cmit 0 3]
 ++                     }
 ++
 ++                     set author_abbr {}
 ++                     set a_name {}
 ++                     catch {set a_name $header($cmit,author)}
 ++                     while {$a_name ne {}} {
 ++                             if {$author_abbr ne {}
 ++                                     && [string index $a_name 0] eq {'}} {
 ++                                     regsub {^'[^']+'\s+} $a_name {} a_name
 ++                             }
 ++                             if {![regexp {^([[:upper:]])} $a_name _a]} break
 ++                             append author_abbr $_a
 ++                             unset _a
 ++                             if {![regsub \
 ++                                     {^[[:upper:]][^\s]*\s+} \
 ++                                     $a_name {} a_name ]} break
 ++                     }
 ++                     if {$author_abbr eq {}} {
 ++                             set author_abbr { |}
 ++                     } else {
 ++                             set author_abbr [string range $author_abbr 0 3]
 ++                     }
 ++                     unset a_name
 ++
 ++                     set first_lno $lno
 ++                     while {
 ++                        $first_lno > 1
 ++                     && $cmit eq [lindex $line_data [expr {$first_lno - 1}] 0]
 ++                     && $file eq [lindex $line_data [expr {$first_lno - 1}] 1]
 ++                     } {
 ++                             incr first_lno -1
 ++                     }
 ++
 ++                     set color {}
 ++                     if {$first_lno < $lno} {
 ++                             foreach g [$w_file tag names $first_lno.0] {
 ++                                     if {[regexp {^color[0-9]+$} $g]} {
 ++                                             set color $g
 ++                                             break
 ++                                     }
 ++                             }
 ++                     } else {
 ++                             set i [lsort [concat \
 ++                                     [$w_file tag names "[expr {$first_lno - 1}].0"] \
 ++                                     [$w_file tag names "[expr {$lno + $n}].0"] \
 ++                                     ]]
 ++                             for {set g 0} {$g < [llength $group_colors]} {incr g} {
 ++                                     if {[lsearch -sorted -exact $i color$g] == -1} {
 ++                                             set color color$g
 ++                                             break
 ++                                     }
 ++                             }
 ++                     }
 ++                     if {$color eq {}} {
 ++                             set color color0
 ++                     }
 ++
 ++                     while {$n > 0} {
 ++                             set lno_e "$lno.0 lineend + 1c"
 ++                             if {[lindex $line_data $lno] ne {}} {
 ++                                     set g [lindex $line_data $lno 0]
 ++                                     foreach i $w_columns {
 ++                                             $i tag remove g$g $lno.0 $lno_e
 ++                                     }
 ++                             }
 ++                             lset line_data $lno [list $cmit $file $oln]
 ++
 ++                             $cur_w delete $lno.0 "$lno.0 lineend"
 ++                             if {$lno == $first_lno} {
 ++                                     $cur_w insert $lno.0 $commit_abbr $commit_type
 ++                             } elseif {$lno == [expr {$first_lno + 1}]} {
 ++                                     $cur_w insert $lno.0 $author_abbr author_abbr
 ++                             } else {
 ++                                     $cur_w insert $lno.0 { |}
 ++                             }
 ++
 ++                             foreach i $w_columns {
 ++                                     if {$cur_w eq $w_amov} {
 ++                                             for {set g 0} \
 ++                                                     {$g < [llength $group_colors]} \
 ++                                                     {incr g} {
 ++                                                     $i tag remove color$g $lno.0 $lno_e
 ++                                             }
 ++                                             $i tag add $color $lno.0 $lno_e
 ++                                     }
 ++                                     $i tag add g$cmit $lno.0 $lno_e
 ++                             }
 ++
 ++                             if {$highlight_column eq $cur_w} {
 ++                                     if {$highlight_line == -1
 ++                                      && [lindex [$w_file yview] 0] == 0} {
 ++                                             $w_file see $lno.0
 ++                                             set highlight_line $lno
 ++                                     }
 ++                                     if {$highlight_line == $lno} {
 ++                                             _showcommit $this $cur_w $lno
 ++                                     }
 ++                             }
 ++
 ++                             incr n -1
 ++                             incr lno
 ++                             incr oln
 ++                             incr blame_lines
 ++                     }
 ++
 ++                     while {
 ++                        $cmit eq [lindex $line_data $lno 0]
 ++                     && $file eq [lindex $line_data $lno 1]
 ++                     } {
 ++                             $cur_w delete $lno.0 "$lno.0 lineend"
 ++
 ++                             if {$lno == $first_lno} {
 ++                                     $cur_w insert $lno.0 $commit_abbr $commit_type
 ++                             } elseif {$lno == [expr {$first_lno + 1}]} {
 ++                                     $cur_w insert $lno.0 $author_abbr author_abbr
 ++                             } else {
 ++                                     $cur_w insert $lno.0 { |}
 ++                             }
 ++
 ++                             if {$cur_w eq $w_amov} {
 ++                                     foreach i $w_columns {
 ++                                             for {set g 0} \
 ++                                                     {$g < [llength $group_colors]} \
 ++                                                     {incr g} {
 ++                                                     $i tag remove color$g $lno.0 $lno_e
 ++                                             }
 ++                                             $i tag add $color $lno.0 $lno_e
 ++                                     }
 ++                             }
 ++
 ++                             incr lno
 ++                     }
 ++
 ++             } elseif {[regexp {^([a-z-]+) (.*)$} $line line key data]} {
 ++                     set header($r_commit,$key) $data
 ++             }
 ++     }
 ++     $cur_w conf -state disabled
 ++
 ++     if {[eof $fd]} {
 ++             close $fd
 ++             if {$cur_w eq $w_asim} {
 ++                     # Switches for original location detection
 ++                     set threshold [get_config gui.copyblamethreshold]
 ++                     set original_options [list "-C$threshold"]
 ++
 ++                     if {![is_config_true gui.fastcopyblame]} {
 ++                             # thorough copy search; insert before the threshold
 ++                             set original_options [linsert $original_options 0 -C]
 ++                     }
 ++                     if {[git-version >= 1.5.3]} {
 ++                             lappend original_options -w ; # ignore indentation changes
 ++                     }
 ++
 ++                     _exec_blame $this $w_amov @amov_data \
 ++                             $original_options \
 ++                             [mc "Loading original location annotations..."]
 ++             } else {
 ++                     set current_fd {}
 ++                     $status stop [mc "Annotation complete."]
 ++             }
 ++     } else {
 ++             $status update $blame_lines $total_lines
 ++     }
 ++} ifdeleted { catch {close $fd} }
 ++
 ++method _find_commit_bound {data_list start_idx delta} {
 ++     upvar #0 $data_list line_data
 ++     set pos $start_idx
 ++     set limit       [expr {[llength $line_data] - 1}]
 ++     set base_commit [lindex $line_data $pos 0]
 ++
 ++     while {$pos > 0 && $pos < $limit} {
 ++             set new_pos [expr {$pos + $delta}]
 ++             if {[lindex $line_data $new_pos 0] ne $base_commit} {
 ++                     return $pos
 ++             }
 ++
 ++             set pos $new_pos
 ++     }
 ++
 ++     return $pos
 ++}
 ++
 ++method _fullcopyblame {} {
 ++     if {$current_fd ne {}} {
 ++             tk_messageBox \
 ++                     -icon error \
 ++                     -type ok \
 ++                     -title [mc "Busy"] \
 ++                     -message [mc "Annotation process is already running."]
 ++
 ++             return
 ++     }
 ++
 ++     # Switches for original location detection
 ++     set threshold [get_config gui.copyblamethreshold]
 ++     set original_options [list -C -C "-C$threshold"]
 ++
 ++     if {[git-version >= 1.5.3]} {
 ++             lappend original_options -w ; # ignore indentation changes
 ++     }
 ++
 ++     # Find the line range
 ++     set pos @$::cursorX,$::cursorY
 ++     set lno [lindex [split [$::cursorW index $pos] .] 0]
 ++     set min_amov_lno [_find_commit_bound $this @amov_data $lno -1]
 ++     set max_amov_lno [_find_commit_bound $this @amov_data $lno 1]
 ++     set min_asim_lno [_find_commit_bound $this @asim_data $lno -1]
 ++     set max_asim_lno [_find_commit_bound $this @asim_data $lno 1]
 ++
 ++     if {$min_asim_lno < $min_amov_lno} {
 ++             set min_amov_lno $min_asim_lno
 ++     }
 ++
 ++     if {$max_asim_lno > $max_amov_lno} {
 ++             set max_amov_lno $max_asim_lno
 ++     }
 ++
 ++     lappend original_options -L "$min_amov_lno,$max_amov_lno"
 ++
 ++     # Clear lines
 ++     for {set i $min_amov_lno} {$i <= $max_amov_lno} {incr i} {
 ++             lset amov_data $i [list ]
 ++     }
 ++
 ++     # Start the back-end process
 ++     _exec_blame $this $w_amov @amov_data \
 ++             $original_options \
 ++             [mc "Running thorough copy detection..."]
 ++}
 ++
 ++method _click {cur_w pos} {
 ++     set lno [lindex [split [$cur_w index $pos] .] 0]
 ++     _showcommit $this $cur_w $lno
 ++}
 ++
 ++method _load_commit {cur_w cur_d pos} {
 ++     upvar #0 $cur_d line_data
 ++     set lno [lindex [split [$cur_w index $pos] .] 0]
 ++     set dat [lindex $line_data $lno]
 ++     if {$dat ne {}} {
-  method _copycommit {} {
+++             _load_new_commit $this  \
+++                     [lindex $dat 0] \
+++                     [lindex $dat 1] \
+++                     [list [lindex $dat 2]]
 ++     }
 ++}
 ++
+++method _load_new_commit {new_commit new_path jump} {
+++     lappend history [list \
+++             $commit $path \
+++             $highlight_column \
+++             $highlight_line \
+++             [lindex [$w_file xview] 0] \
+++             [lindex [$w_file yview] 0] \
+++             ]
+++
+++     set commit $new_commit
+++     set path   $new_path
+++     _load $this $jump
+++}
+++
 ++method _showcommit {cur_w lno} {
 ++     global repo_config
 ++     variable active_color
 ++
 ++     if {$highlight_commit ne {}} {
 ++             foreach i $w_columns {
 ++                     $i tag conf g$highlight_commit -background {}
 ++                     $i tag lower g$highlight_commit
 ++             }
 ++     }
 ++
 ++     if {$cur_w eq $w_asim} {
 ++             set dat [lindex $asim_data $lno]
 ++             set highlight_column $w_asim
 ++     } else {
 ++             set dat [lindex $amov_data $lno]
 ++             set highlight_column $w_amov
 ++     }
 ++
 ++     $w_cviewer conf -state normal
 ++     $w_cviewer delete 0.0 end
 ++
 ++     if {$dat eq {}} {
 ++             set cmit {}
 ++             $w_cviewer insert end [mc "Loading annotation..."] still_loading
 ++     } else {
 ++             set cmit [lindex $dat 0]
 ++             set file [lindex $dat 1]
 ++
 ++             foreach i $w_columns {
 ++                     $i tag conf g$cmit -background $active_color
 ++                     $i tag raise g$cmit
 ++             }
 ++
 ++             set author_name {}
 ++             set author_email {}
 ++             set author_time {}
 ++             catch {set author_name $header($cmit,author)}
 ++             catch {set author_email $header($cmit,author-mail)}
 ++             catch {set author_time [format_date $header($cmit,author-time)]}
 ++
 ++             set committer_name {}
 ++             set committer_email {}
 ++             set committer_time {}
 ++             catch {set committer_name $header($cmit,committer)}
 ++             catch {set committer_email $header($cmit,committer-mail)}
 ++             catch {set committer_time [format_date $header($cmit,committer-time)]}
 ++
 ++             if {[catch {set msg $header($cmit,message)}]} {
 ++                     set msg {}
 ++                     catch {
 ++                             set fd [git_read cat-file commit $cmit]
 ++                             fconfigure $fd -encoding binary -translation lf
 ++                             if {[catch {set enc $repo_config(i18n.commitencoding)}]} {
 ++                                     set enc utf-8
 ++                             }
 ++                             while {[gets $fd line] > 0} {
 ++                                     if {[string match {encoding *} $line]} {
 ++                                             set enc [string tolower [string range $line 9 end]]
 ++                                     }
 ++                             }
 ++                             set msg [read $fd]
 ++                             close $fd
 ++
 ++                             set enc [tcl_encoding $enc]
 ++                             if {$enc ne {}} {
 ++                                     set msg [encoding convertfrom $enc $msg]
 ++                                     set author_name [encoding convertfrom $enc $author_name]
 ++                                     set committer_name [encoding convertfrom $enc $committer_name]
 ++                                     set header($cmit,author) $author_name
 ++                                     set header($cmit,committer) $committer_name
 ++                                     set header($cmit,summary) \
 ++                                     [encoding convertfrom $enc $header($cmit,summary)]
 ++                             }
 ++                             set msg [string trim $msg]
 ++                     }
 ++                     set header($cmit,message) $msg
 ++             }
 ++
 ++             $w_cviewer insert end "commit $cmit\n" header_key
 ++             $w_cviewer insert end [strcat [mc "Author:"] "\t"] header_key
 ++             $w_cviewer insert end "$author_name $author_email" header_val
 ++             $w_cviewer insert end "  $author_time\n" header_val
 ++
 ++             $w_cviewer insert end [strcat [mc "Committer:"] "\t"] header_key
 ++             $w_cviewer insert end "$committer_name $committer_email" header_val
 ++             $w_cviewer insert end "  $committer_time\n" header_val
 ++
 ++             if {$file ne $path} {
 ++                     $w_cviewer insert end [strcat [mc "Original File:"] "\t"] header_key
 ++                     $w_cviewer insert end "[escape_path $file]\n" header_val
 ++             }
 ++
 ++             $w_cviewer insert end "\n$msg"
 ++     }
 ++     $w_cviewer conf -state disabled
 ++
 ++     set highlight_line $lno
 ++     set highlight_commit $cmit
 ++
 ++     if {[lsearch -exact $tooltip_commit $highlight_commit] != -1} {
 ++             _hide_tooltip $this
 ++     }
 ++}
 ++
-       set dat [lindex $amov_data $lno]
+++method _get_click_amov_info {} {
 ++     set pos @$::cursorX,$::cursorY
 ++     set lno [lindex [split [$::cursorW index $pos] .] 0]
+++     return [lindex $amov_data $lno]
+++}
+++
+++method _copycommit {} {
+++     set dat [_get_click_amov_info $this]
 ++     if {$dat ne {}} {
 ++             clipboard clear
 ++             clipboard append \
 ++                     -format STRING \
 ++                     -type STRING \
 ++                     -- [lindex $dat 0]
 ++     }
 ++}
 ++
+++method _format_offset_date {base offset} {
+++     set exval [expr {$base + $offset*24*60*60}]
+++     return [clock format $exval -format {%Y-%m-%d}]
+++}
+++
+++method _gitkcommit {} {
+++     set dat [_get_click_amov_info $this]
+++     if {$dat ne {}} {
+++             set cmit [lindex $dat 0]
+++             set radius [get_config gui.blamehistoryctx]
+++             set cmdline [list --select-commit=$cmit]
+++
+++                if {$radius > 0} {
+++                     set author_time {}
+++                     set committer_time {}
+++
+++                     catch {set author_time $header($cmit,author-time)}
+++                     catch {set committer_time $header($cmit,committer-time)}
+++
+++                     if {$committer_time eq {}} {
+++                             set committer_time $author_time
+++                     }
+++
+++                     set after_time [_format_offset_date $this $committer_time [expr {-$radius}]]
+++                     set before_time [_format_offset_date $this $committer_time $radius]
+++
+++                     lappend cmdline --after=$after_time --before=$before_time
+++             }
+++
+++             lappend cmdline $cmit
+++
+++             set base_rev "HEAD"
+++             if {$commit ne {}} {
+++                     set base_rev $commit
+++             }
+++
+++             if {$base_rev ne $cmit} {
+++                     lappend cmdline $base_rev
+++             }
+++
+++             do_gitk $cmdline
+++     }
+++}
+++
+++method _blameparent {} {
+++     set dat [_get_click_amov_info $this]
+++     if {$dat ne {}} {
+++             set cmit [lindex $dat 0]
+++             set new_path [lindex $dat 1]
+++
+++             if {[catch {set cparent [git rev-parse --verify "$cmit^"]}]} {
+++                     error_popup [strcat [mc "Cannot find parent commit:"] "\n\n$err"]
+++                     return;
+++             }
+++
+++             _kill $this
+++
+++             # Generate a diff between the commit and its parent,
+++             # and use the hunks to update the line number.
+++             # Request zero context to simplify calculations.
+++             if {[catch {set fd [eval git_read diff-tree \
+++                             --unified=0 $cparent $cmit $new_path]} err]} {
+++                     $status stop [mc "Unable to display parent"]
+++                     error_popup [strcat [mc "Error loading diff:"] "\n\n$err"]
+++                     return
+++             }
+++
+++             set r_orig_line [lindex $dat 2]
+++
+++             fconfigure $fd \
+++                     -blocking 0 \
+++                     -encoding binary \
+++                     -translation binary
+++             fileevent $fd readable [cb _read_diff_load_commit \
+++                     $fd $cparent $new_path $r_orig_line]
+++             set current_fd $fd
+++     }
+++}
+++
+++method _read_diff_load_commit {fd cparent new_path tline} {
+++     if {$fd ne $current_fd} {
+++             catch {close $fd}
+++             return
+++     }
+++
+++     while {[gets $fd line] >= 0} {
+++             if {[regexp {^@@ -(\d+)(,(\d+))? \+(\d+)(,(\d+))? @@} $line line \
+++                     old_line osz old_size new_line nsz new_size]} {
+++
+++                     if {$osz eq {}} { set old_size 1 }
+++                     if {$nsz eq {}} { set new_size 1 }
+++
+++                     if {$new_line <= $tline} {
+++                             if {[expr {$new_line + $new_size}] > $tline} {
+++                                     # Target line within the hunk
+++                                     set line_shift [expr {
+++                                             ($new_size-$old_size)*($tline-$new_line)/$new_size
+++                                             }]
+++                             } else {
+++                                     set line_shift [expr {$new_size-$old_size}]
+++                             }
+++
+++                             set r_orig_line [expr {$r_orig_line - $line_shift}]
+++                     }
+++             }
+++     }
+++
+++     if {[eof $fd]} {
+++             close $fd;
+++             set current_fd {}
+++
+++             _load_new_commit $this  \
+++                     $cparent        \
+++                     $new_path       \
+++                     [list $r_orig_line]
+++     }
+++} ifdeleted { catch {close $fd} }
+++
 ++method _show_tooltip {cur_w pos} {
 ++     if {$tooltip_wm ne {}} {
 ++             _open_tooltip $this $cur_w
 ++     } elseif {$tooltip_timer eq {}} {
 ++             set tooltip_timer [after 1000 [cb _open_tooltip $cur_w]]
 ++     }
 ++}
 ++
 ++method _open_tooltip {cur_w} {
 ++     set tooltip_timer {}
 ++     set pos_x [winfo pointerx $cur_w]
 ++     set pos_y [winfo pointery $cur_w]
 ++     if {[winfo containing $pos_x $pos_y] ne $cur_w} {
 ++             _hide_tooltip $this
 ++             return
 ++     }
 ++
 ++     if {$tooltip_wm ne "$cur_w.tooltip"} {
 ++             _hide_tooltip $this
 ++
 ++             set tooltip_wm [toplevel $cur_w.tooltip -borderwidth 1]
 ++             wm overrideredirect $tooltip_wm 1
 ++             wm transient $tooltip_wm [winfo toplevel $cur_w]
 ++             set tooltip_t $tooltip_wm.label
 ++             text $tooltip_t \
 ++                     -takefocus 0 \
 ++                     -highlightthickness 0 \
 ++                     -relief flat \
 ++                     -borderwidth 0 \
 ++                     -wrap none \
 ++                     -background lightyellow \
 ++                     -foreground black
 ++             $tooltip_t tag conf section_header -font font_uibold
 ++             pack $tooltip_t
 ++     } else {
 ++             $tooltip_t conf -state normal
 ++             $tooltip_t delete 0.0 end
 ++     }
 ++
 ++     set pos @[join [list \
 ++             [expr {$pos_x - [winfo rootx $cur_w]}] \
 ++             [expr {$pos_y - [winfo rooty $cur_w]}]] ,]
 ++     set lno [lindex [split [$cur_w index $pos] .] 0]
 ++     if {$cur_w eq $w_amov} {
 ++             set dat [lindex $amov_data $lno]
 ++             set org {}
 ++     } else {
 ++             set dat [lindex $asim_data $lno]
 ++             set org [lindex $amov_data $lno]
 ++     }
 ++
 ++     if {$dat eq {}} {
 ++             _hide_tooltip $this
 ++             return
 ++     }
 ++
 ++     set cmit [lindex $dat 0]
 ++     set tooltip_commit [list $cmit]
 ++
 ++     set author_name {}
 ++     set summary     {}
 ++     set author_time {}
 ++     catch {set author_name $header($cmit,author)}
 ++     catch {set summary     $header($cmit,summary)}
 ++     catch {set author_time [format_date $header($cmit,author-time)]}
 ++
 ++     $tooltip_t insert end "commit $cmit\n"
 ++     $tooltip_t insert end "$author_name  $author_time\n"
 ++     $tooltip_t insert end "$summary"
 ++
 ++     if {$org ne {} && [lindex $org 0] ne $cmit} {
 ++             set save [$tooltip_t get 0.0 end]
 ++             $tooltip_t delete 0.0 end
 ++
 ++             set cmit [lindex $org 0]
 ++             set file [lindex $org 1]
 ++             lappend tooltip_commit $cmit
 ++
 ++             set author_name {}
 ++             set summary     {}
 ++             set author_time {}
 ++             catch {set author_name $header($cmit,author)}
 ++             catch {set summary     $header($cmit,summary)}
 ++             catch {set author_time [format_date $header($cmit,author-time)]}
 ++
 ++             $tooltip_t insert end [strcat [mc "Originally By:"] "\n"] section_header
 ++             $tooltip_t insert end "commit $cmit\n"
 ++             $tooltip_t insert end "$author_name  $author_time\n"
 ++             $tooltip_t insert end "$summary\n"
 ++
 ++             if {$file ne $path} {
 ++                     $tooltip_t insert end [strcat [mc "In File:"] " "] section_header
 ++                     $tooltip_t insert end "$file\n"
 ++             }
 ++
 ++             $tooltip_t insert end "\n"
 ++             $tooltip_t insert end [strcat [mc "Copied Or Moved Here By:"] "\n"] section_header
 ++             $tooltip_t insert end $save
 ++     }
 ++
 ++     $tooltip_t conf -state disabled
 ++     _position_tooltip $this
 ++}
 ++
 ++method _position_tooltip {} {
 ++     set max_h [lindex [split [$tooltip_t index end] .] 0]
 ++     set max_w 0
 ++     for {set i 1} {$i <= $max_h} {incr i} {
 ++             set c [lindex [split [$tooltip_t index "$i.0 lineend"] .] 1]
 ++             if {$c > $max_w} {set max_w $c}
 ++     }
 ++     $tooltip_t conf -width $max_w -height $max_h
 ++
 ++     set req_w [winfo reqwidth  $tooltip_t]
 ++     set req_h [winfo reqheight $tooltip_t]
 ++     set pos_x [expr {[winfo pointerx .] +  5}]
 ++     set pos_y [expr {[winfo pointery .] + 10}]
 ++
 ++     set g "${req_w}x${req_h}"
 ++     if {$pos_x >= 0} {append g +}
 ++     append g $pos_x
 ++     if {$pos_y >= 0} {append g +}
 ++     append g $pos_y
 ++
 ++     wm geometry $tooltip_wm $g
 ++     raise $tooltip_wm
 ++}
 ++
 ++method _hide_tooltip {} {
 ++     if {$tooltip_wm ne {}} {
 ++             destroy $tooltip_wm
 ++             set tooltip_wm {}
 ++             set tooltip_commit {}
 ++     }
 ++     if {$tooltip_timer ne {}} {
 ++             after cancel $tooltip_timer
 ++             set tooltip_timer {}
 ++     }
 ++}
 ++
 ++method _resize {new_height} {
 ++     set diff [expr {$new_height - $old_height}]
 ++     if {$diff == 0} return
 ++
 ++     set my [expr {[winfo height $w.file_pane] - 25}]
 ++     set o [$w.file_pane sash coord 0]
 ++     set ox [lindex $o 0]
 ++     set oy [expr {[lindex $o 1] + $diff}]
 ++     if {$oy < 0}   {set oy 0}
 ++     if {$oy > $my} {set oy $my}
 ++     $w.file_pane sash place 0 $ox $oy
 ++
 ++     set old_height $new_height
 ++}
 ++
 ++}
index ab470d12648c1dd3560b411e70ea210cc4381e3e,0000000000000000000000000000000000000000,0000000000000000000000000000000000000000..0410cc68df186d6e8e0080058be94126243fd4d3
mode 100644,000000,000000..100644
--- /dev/null
--- /dev/null
@@@@ -1,311 -1,0 -1,0 +1,311 @@@@
-                       blame::new $browser_commit $p
 ++# git-gui tree browser
 ++# Copyright (C) 2006, 2007 Shawn Pearce
 ++
 ++class browser {
 ++
 ++image create photo ::browser::img_parent  -data {R0lGODlhEAAQAIUAAPwCBBxSHBxOHMTSzNzu3KzCtBRGHCSKFIzCjLzSxBQ2FAxGHDzCLCyeHBQ+FHSmfAwuFBxKLDSCNMzizISyjJzOnDSyLAw+FAQSDAQeDBxWJAwmDAQOBKzWrDymNAQaDAQODAwaDDyKTFSyXFTGTEy6TAQCBAQKDAwiFBQyHAwSFAwmHAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACH5BAEAAAAALAAAAAAQABAAAAZ1QIBwSCwaj0hiQCBICpcDQsFgGAaIguhhi0gohIsrQEDYMhiNrRfgeAQC5fMCAolIDhD2hFI5WC4YRBkaBxsOE2l/RxsHHA4dHmkfRyAbIQ4iIyQlB5NFGCAACiakpSZEJyinTgAcKSesACorgU4mJ6uxR35BACH+aENyZWF0ZWQgYnkgQk1QVG9HSUYgUHJvIHZlcnNpb24gMi41DQqpIERldmVsQ29yIDE5OTcsMTk5OC4gQWxsIHJpZ2h0cyByZXNlcnZlZC4NCmh0dHA6Ly93d3cuZGV2ZWxjb3IuY29tADs=}
 ++image create photo ::browser::img_rblob   -data {R0lGODlhEAAQAIUAAPwCBFxaXNze3Ly2rJSWjPz+/Ozq7GxqbJyanPT29HRydMzOzDQyNIyKjERCROTi3Pz69PTy7Pzy7PTu5Ozm3LyqlJyWlJSSjJSOhOzi1LyulPz27PTq3PTm1OzezLyqjIyKhJSKfOzaxPz29OzizLyidIyGdIyCdOTOpLymhOzavOTStMTCtMS+rMS6pMSynMSulLyedAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACH5BAEAAAAALAAAAAAQABAAAAaQQIAQECgajcNkQMBkDgKEQFK4LFgLhkMBIVUKroWEYlEgMLxbBKLQUBwc52HgAQ4LBo049atWQyIPA3pEdFcQEhMUFYNVagQWFxgZGoxfYRsTHB0eH5UJCJAYICEinUoPIxIcHCQkIiIllQYEGCEhJicoKYwPmiQeKisrKLFKLCwtLi8wHyUlMYwM0tPUDH5BACH+aENyZWF0ZWQgYnkgQk1QVG9HSUYgUHJvIHZlcnNpb24gMi41DQqpIERldmVsQ29yIDE5OTcsMTk5OC4gQWxsIHJpZ2h0cyByZXNlcnZlZC4NCmh0dHA6Ly93d3cuZGV2ZWxjb3IuY29tADs=}
 ++image create photo ::browser::img_xblob   -data {R0lGODlhEAAQAIYAAPwCBFRWVFxaXNza3OTi3Nze3Ly2tJyanPz+/Ozq7GxubNzSxMzOzMTGxHRybDQyNLy+vHRydHx6fKSipISChIyKjGxqbERCRCwuLLy6vGRiZExKTCQiJAwKDLSytLy2rJSSlHx+fDw6PKyqrBQWFPTu5Ozm3LyulLS2tCQmJAQCBPTq3Ozi1MSynCwqLAQGBOTazOzizOzezLyqjBweHNzSvOzaxKyurHRuZNzOtLymhDw+PIyCdOzWvOTOpLyidNzKtOTStLyifMTCtMS+rLyedAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACH5BAEAAAAALAAAAAAQABAAAAfZgACCAAEChYeGg4oCAwQFjgYBBwGKggEECJkICQoIkwADCwwNDY2mDA4Lng8QDhESsLARExQVDhYXGBkWExIaGw8cHR4SCQQfFQ8eFgUgIQEiwiMSBMYfGB4atwEXDyQd0wQlJicPKAHoFyIpJCoeDgMrLC0YKBsX6i4kL+4OMDEyZijr5oLGNxUqUCioEcPGDAwjPNyI6MEDChQjcOSwsUDHgw07RIgI4KCkAgs8cvTw8eOBogAxQtXIASTISiEuBwUYMoRIixYnZggpUgTDywdIkWJIitRPIAAh/mhDcmVhdGVkIGJ5IEJNUFRvR0lGIFBybyB2ZXJzaW9uIDIuNQ0KqSBEZXZlbENvciAxOTk3LDE5OTguIEFsbCByaWdodHMgcmVzZXJ2ZWQuDQpodHRwOi8vd3d3LmRldmVsY29yLmNvbQA7}
 ++image create photo ::browser::img_tree    -data {R0lGODlhEAAQAIYAAPwCBAQCBExKTBwWHMzKzOzq7ERCRExGTCwqLARqnAQ+ZHR2dKyqrNTOzHx2fCQiJMTi9NTu9HzC3AxmnAQ+XPTm7Dy67DymzITC3IzG5AxypHRydKymrMzOzOzu7BweHByy9AyGtFyy1IzG3NTu/ARupFRSVByazBR6rAyGvFyuzJTK3MTm9BR+tAxWhHS61MTi7Pz+/IymvCxulBRelAx2rHS63Pz6/PTy9PTu9Nza3ISitBRupFSixNTS1CxqnDQyNMzGzOTi5MTCxMTGxGxubGxqbLy2vLSutGRiZLy6vLSytKyurDQuNFxaXKSipDw6PAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACH5BAEAAAAALAAAAAAQABAAAAfDgACCAAECg4eIAAMEBQYHCImDBgkKCwwNBQIBBw4Bhw8QERITFJYEFQUFnoIPFhcYoRkaFBscHR4Ggh8gIRciEiMQJBkltCa6JyUoKSkXKhIrLCQYuQAPLS4TEyUhKb0qLzDVAjEFMjMuNBMoNcw21QY3ODkFOjs82RM1PfDzFRU3fOggcM7Fj2pAgggRokOHDx9DhhAZUqQaISBGhjwMEvEIkiIHEgUAkgSJkiNLmFSMJChAEydPGBSBwvJQgAc0/QQCACH+aENyZWF0ZWQgYnkgQk1QVG9HSUYgUHJvIHZlcnNpb24gMi41DQqpIERldmVsQ29yIDE5OTcsMTk5OC4gQWxsIHJpZ2h0cyByZXNlcnZlZC4NCmh0dHA6Ly93d3cuZGV2ZWxjb3IuY29tADs=}
 ++image create photo ::browser::img_symlink -data {R0lGODlhEAAQAIQAAPwCBCwqLLSytLy+vERGRFRWVDQ2NKSmpAQCBKyurMTGxISChJyanHR2dIyKjGxubHRydGRmZIyOjFxeXHx6fAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACH5BAEAAAAALAAAAAAQABAAAAVbICACwWieY1CibCCsrBkMb0zchSEcNYskCtqBBzshFkOGQFk0IRqOxqPBODRHCMhCQKteRc9FI/KQWGOIyFYgkDC+gPR4snCcfRGKOIKIgSMQE31+f4OEYCZ+IQAh/mhDcmVhdGVkIGJ5IEJNUFRvR0lGIFBybyB2ZXJzaW9uIDIuNQ0KqSBEZXZlbENvciAxOTk3LDE5OTguIEFsbCByaWdodHMgcmVzZXJ2ZWQuDQpodHRwOi8vd3d3LmRldmVsY29yLmNvbQA7}
 ++image create photo ::browser::img_unknown -data {R0lGODlhEAAQAIUAAPwCBFxaXIyKjNTW1Nze3LS2tJyanER2RGS+VPz+/PTu5GxqbPz69BQ6BCxeLFSqRPT29HRydMzOzDQyNERmPKSypCRWHIyKhERCRDyGPKz2nESiLBxGHCyCHGxubPz6/PTy7Ozi1Ly2rKSipOzm3LyqlKSWhCRyFOzizLymhNTKtNzOvOzaxOTStPz27OzWvOTOpLSupLyedMS+rMS6pMSulLyqjLymfLyifAAAAAAAAAAAAAAAAAAAAAAAAAAAACH5BAEAAAAALAAAAAAQABAAAAamQIAQECgajcOkYEBoDgoBQyAJOCCuiENCsWBIh9aGw9F4HCARiXciRDQoBUnlYRlcIgsMG5CxXAgMGhscBRAEBRd7AB0eBBoIgxUfICEiikSPgyMMIAokJZcBkBybJgomIaBJAZoMpyCmqkMBFCcVCrgKKAwpoSorKqchKCwtvasIFBIhLiYvLzDHsxQNMcMKLDAwMqEz3jQ1NTY3ONyrE+jp6hN+QQAh/mhDcmVhdGVkIGJ5IEJNUFRvR0lGIFBybyB2ZXJzaW9uIDIuNQ0KqSBEZXZlbENvciAxOTk3LDE5OTguIEFsbCByaWdodHMgcmVzZXJ2ZWQuDQpodHRwOi8vd3d3LmRldmVsY29yLmNvbQA7}
 ++
 ++field w
 ++field browser_commit
 ++field browser_path
 ++field browser_files  {}
 ++field browser_status [mc "Starting..."]
 ++field browser_stack  {}
 ++field browser_busy   1
 ++
 ++field ls_buf     {}; # Buffered record output from ls-tree
 ++
 ++constructor new {commit {path {}}} {
 ++     global cursor_ptr M1B
 ++     make_toplevel top w
 ++     wm title $top [append "[appname] ([reponame]): " [mc "File Browser"]]
 ++
 ++     set browser_commit $commit
 ++     set browser_path $browser_commit:$path
 ++
 ++     label $w.path \
 ++             -textvariable @browser_path \
 ++             -anchor w \
 ++             -justify left \
 ++             -borderwidth 1 \
 ++             -relief sunken \
 ++             -font font_uibold
 ++     pack $w.path -anchor w -side top -fill x
 ++
 ++     frame $w.list
 ++     set w_list $w.list.l
 ++     text $w_list -background white -foreground black \
 ++             -borderwidth 0 \
 ++             -cursor $cursor_ptr \
 ++             -state disabled \
 ++             -wrap none \
 ++             -height 20 \
 ++             -width 70 \
 ++             -xscrollcommand [list $w.list.sbx set] \
 ++             -yscrollcommand [list $w.list.sby set]
 ++     rmsel_tag $w_list
 ++     scrollbar $w.list.sbx -orient h -command [list $w_list xview]
 ++     scrollbar $w.list.sby -orient v -command [list $w_list yview]
 ++     pack $w.list.sbx -side bottom -fill x
 ++     pack $w.list.sby -side right -fill y
 ++     pack $w_list -side left -fill both -expand 1
 ++     pack $w.list -side top -fill both -expand 1
 ++
 ++     label $w.status \
 ++             -textvariable @browser_status \
 ++             -anchor w \
 ++             -justify left \
 ++             -borderwidth 1 \
 ++             -relief sunken
 ++     pack $w.status -anchor w -side bottom -fill x
 ++
 ++     bind $w_list <Button-1>        "[cb _click 0 @%x,%y];break"
 ++     bind $w_list <Double-Button-1> "[cb _click 1 @%x,%y];break"
 ++     bind $w_list <$M1B-Up>         "[cb _parent]        ;break"
 ++     bind $w_list <$M1B-Left>       "[cb _parent]        ;break"
 ++     bind $w_list <Up>              "[cb _move -1]       ;break"
 ++     bind $w_list <Down>            "[cb _move  1]       ;break"
 ++     bind $w_list <$M1B-Right>      "[cb _enter]         ;break"
 ++     bind $w_list <Return>          "[cb _enter]         ;break"
 ++     bind $w_list <Prior>           "[cb _page -1]       ;break"
 ++     bind $w_list <Next>            "[cb _page  1]       ;break"
 ++     bind $w_list <Left>            break
 ++     bind $w_list <Right>           break
 ++
 ++     bind $w_list <Visibility> [list focus $w_list]
 ++     set w $w_list
 ++     if {$path ne {}} {
 ++             _ls $this $browser_commit:$path $path
 ++     } else {
 ++             _ls $this $browser_commit $path
 ++     }
 ++     return $this
 ++}
 ++
 ++method _move {dir} {
 ++     if {$browser_busy} return
 ++     set lno [lindex [split [$w index in_sel.first] .] 0]
 ++     incr lno $dir
 ++     if {[lindex $browser_files [expr {$lno - 1}]] ne {}} {
 ++             $w tag remove in_sel 0.0 end
 ++             $w tag add in_sel $lno.0 [expr {$lno + 1}].0
 ++             $w see $lno.0
 ++     }
 ++}
 ++
 ++method _page {dir} {
 ++     if {$browser_busy} return
 ++     $w yview scroll $dir pages
 ++     set lno [expr {int(
 ++               [lindex [$w yview] 0]
 ++             * [llength $browser_files]
 ++             + 1)}]
 ++     if {[lindex $browser_files [expr {$lno - 1}]] ne {}} {
 ++             $w tag remove in_sel 0.0 end
 ++             $w tag add in_sel $lno.0 [expr {$lno + 1}].0
 ++             $w see $lno.0
 ++     }
 ++}
 ++
 ++method _parent {} {
 ++     if {$browser_busy} return
 ++     set info [lindex $browser_files 0]
 ++     if {[lindex $info 0] eq {parent}} {
 ++             set parent [lindex $browser_stack end-1]
 ++             set browser_stack [lrange $browser_stack 0 end-2]
 ++             if {$browser_stack eq {}} {
 ++                     regsub {:.*$} $browser_path {:} browser_path
 ++             } else {
 ++                     regsub {/[^/]+$} $browser_path {} browser_path
 ++             }
 ++             set browser_status [mc "Loading %s..." $browser_path]
 ++             _ls $this [lindex $parent 0] [lindex $parent 1]
 ++     }
 ++}
 ++
 ++method _enter {} {
 ++     if {$browser_busy} return
 ++     set lno [lindex [split [$w index in_sel.first] .] 0]
 ++     set info [lindex $browser_files [expr {$lno - 1}]]
 ++     if {$info ne {}} {
 ++             switch -- [lindex $info 0] {
 ++             parent {
 ++                     _parent $this
 ++             }
 ++             tree {
 ++                     set name [lindex $info 2]
 ++                     set escn [escape_path $name]
 ++                     set browser_status [mc "Loading %s..." $escn]
 ++                     append browser_path $escn
 ++                     _ls $this [lindex $info 1] $name
 ++             }
 ++             blob {
 ++                     set name [lindex $info 2]
 ++                     set p {}
 ++                     foreach n $browser_stack {
 ++                             append p [lindex $n 1]
 ++                     }
 ++                     append p $name
+++                     blame::new $browser_commit $p {}
 ++             }
 ++             }
 ++     }
 ++}
 ++
 ++method _click {was_double_click pos} {
 ++     if {$browser_busy} return
 ++     set lno [lindex [split [$w index $pos] .] 0]
 ++     focus $w
 ++
 ++     if {[lindex $browser_files [expr {$lno - 1}]] ne {}} {
 ++             $w tag remove in_sel 0.0 end
 ++             $w tag add in_sel $lno.0 [expr {$lno + 1}].0
 ++             if {$was_double_click} {
 ++                     _enter $this
 ++             }
 ++     }
 ++}
 ++
 ++method _ls {tree_id {name {}}} {
 ++     set ls_buf {}
 ++     set browser_files {}
 ++     set browser_busy 1
 ++
 ++     $w conf -state normal
 ++     $w tag remove in_sel 0.0 end
 ++     $w delete 0.0 end
 ++     if {$browser_stack ne {}} {
 ++             $w image create end \
 ++                     -align center -padx 5 -pady 1 \
 ++                     -name icon0 \
 ++                     -image ::browser::img_parent
 ++             $w insert end [mc "\[Up To Parent\]"]
 ++             lappend browser_files parent
 ++     }
 ++     lappend browser_stack [list $tree_id $name]
 ++     $w conf -state disabled
 ++
 ++     set fd [git_read ls-tree -z $tree_id]
 ++     fconfigure $fd -blocking 0 -translation binary -encoding binary
 ++     fileevent $fd readable [cb _read $fd]
 ++}
 ++
 ++method _read {fd} {
 ++     append ls_buf [read $fd]
 ++     set pck [split $ls_buf "\0"]
 ++     set ls_buf [lindex $pck end]
 ++
 ++     set n [llength $browser_files]
 ++     $w conf -state normal
 ++     foreach p [lrange $pck 0 end-1] {
 ++             set tab [string first "\t" $p]
 ++             if {$tab == -1} continue
 ++
 ++             set info [split [string range $p 0 [expr {$tab - 1}]] { }]
 ++             set path [string range $p [expr {$tab + 1}] end]
 ++             set type   [lindex $info 1]
 ++             set object [lindex $info 2]
 ++
 ++             switch -- $type {
 ++             blob {
 ++                     scan [lindex $info 0] %o mode
 ++                     if {$mode == 0120000} {
 ++                             set image ::browser::img_symlink
 ++                     } elseif {($mode & 0100) != 0} {
 ++                             set image ::browser::img_xblob
 ++                     } else {
 ++                             set image ::browser::img_rblob
 ++                     }
 ++             }
 ++             tree {
 ++                     set image ::browser::img_tree
 ++                     append path /
 ++             }
 ++             default {
 ++                     set image ::browser::img_unknown
 ++             }
 ++             }
 ++
 ++             if {$n > 0} {$w insert end "\n"}
 ++             $w image create end \
 ++                     -align center -padx 5 -pady 1 \
 ++                     -name icon[incr n] \
 ++                     -image $image
 ++             $w insert end [escape_path $path]
 ++             lappend browser_files [list $type $object $path]
 ++     }
 ++     $w conf -state disabled
 ++
 ++     if {[eof $fd]} {
 ++             close $fd
 ++             set browser_status [mc "Ready."]
 ++             set browser_busy 0
 ++             set ls_buf {}
 ++             if {$n > 0} {
 ++                     $w tag add in_sel 1.0 2.0
 ++                     focus -force $w
 ++             }
 ++     }
 ++} ifdeleted {
 ++     catch {close $fd}
 ++}
 ++
 ++}
 ++
 ++class browser_open {
 ++
 ++field w              ; # widget path
 ++field w_rev          ; # mega-widget to pick the initial revision
 ++
 ++constructor dialog {} {
 ++     make_toplevel top w
 ++     wm title $top [append "[appname] ([reponame]): " [mc "Browse Branch Files"]]
 ++     if {$top ne {.}} {
 ++             wm geometry $top "+[winfo rootx .]+[winfo rooty .]"
 ++     }
 ++
 ++     label $w.header \
 ++             -text [mc "Browse Branch Files"] \
 ++             -font font_uibold
 ++     pack $w.header -side top -fill x
 ++
 ++     frame $w.buttons
 ++     button $w.buttons.browse -text [mc Browse] \
 ++             -default active \
 ++             -command [cb _open]
 ++     pack $w.buttons.browse -side right
 ++     button $w.buttons.cancel -text [mc Cancel] \
 ++             -command [list destroy $w]
 ++     pack $w.buttons.cancel -side right -padx 5
 ++     pack $w.buttons -side bottom -fill x -pady 10 -padx 10
 ++
 ++     set w_rev [::choose_rev::new $w.rev [mc Revision]]
 ++     $w_rev bind_listbox <Double-Button-1> [cb _open]
 ++     pack $w.rev -anchor nw -fill both -expand 1 -pady 5 -padx 5
 ++
 ++     bind $w <Visibility> [cb _visible]
 ++     bind $w <Key-Escape> [list destroy $w]
 ++     bind $w <Key-Return> [cb _open]\;break
 ++     tkwait window $w
 ++}
 ++
 ++method _open {} {
 ++     if {[catch {$w_rev commit_or_die} err]} {
 ++             return
 ++     }
 ++     set name [$w_rev get]
 ++     destroy $w
 ++     browser::new $name
 ++}
 ++
 ++method _visible {} {
 ++     grab $w
 ++     $w_rev focus_filter
 ++}
 ++
 ++}
index 40a710355751836e78b65101592b753266f507ca,0000000000000000000000000000000000000000,0000000000000000000000000000000000000000..2977315624fc5deeecff275d8dc8b3f2e3210a70
mode 100644,000000,000000..100644
--- /dev/null
--- /dev/null
@@@@ -1,470 -1,0 -1,0 +1,473 @@@@
 ++# git-gui misc. commit reading/writing support
 ++# Copyright (C) 2006, 2007 Shawn Pearce
 ++
 ++proc load_last_commit {} {
 ++     global HEAD PARENT MERGE_HEAD commit_type ui_comm
 ++     global repo_config
 ++
 ++     if {[llength $PARENT] == 0} {
 ++             error_popup [mc "There is nothing to amend.
 ++
 ++You are about to create the initial commit.  There is no commit before this to amend.
 ++"]
 ++             return
 ++     }
 ++
 ++     repository_state curType curHEAD curMERGE_HEAD
 ++     if {$curType eq {merge}} {
 ++             error_popup [mc "Cannot amend while merging.
 ++
 ++You are currently in the middle of a merge that has not been fully completed.  You cannot amend the prior commit unless you first abort the current merge activity.
 ++"]
 ++             return
 ++     }
 ++
 ++     set msg {}
 ++     set parents [list]
 ++     if {[catch {
 ++                     set fd [git_read cat-file commit $curHEAD]
 ++                     fconfigure $fd -encoding binary -translation lf
 ++                     if {[catch {set enc $repo_config(i18n.commitencoding)}]} {
 ++                             set enc utf-8
 ++                     }
 ++                     while {[gets $fd line] > 0} {
 ++                             if {[string match {parent *} $line]} {
 ++                                     lappend parents [string range $line 7 end]
 ++                             } elseif {[string match {encoding *} $line]} {
 ++                                     set enc [string tolower [string range $line 9 end]]
 ++                             }
 ++                     }
 ++                     set msg [read $fd]
 ++                     close $fd
 ++
 ++                     set enc [tcl_encoding $enc]
 ++                     if {$enc ne {}} {
 ++                             set msg [encoding convertfrom $enc $msg]
 ++                     }
 ++                     set msg [string trim $msg]
 ++             } err]} {
 ++             error_popup [strcat [mc "Error loading commit data for amend:"] "\n\n$err"]
 ++             return
 ++     }
 ++
 ++     set HEAD $curHEAD
 ++     set PARENT $parents
 ++     set MERGE_HEAD [list]
 ++     switch -- [llength $parents] {
 ++     0       {set commit_type amend-initial}
 ++     1       {set commit_type amend}
 ++     default {set commit_type amend-merge}
 ++     }
 ++
 ++     $ui_comm delete 0.0 end
 ++     $ui_comm insert end $msg
 ++     $ui_comm edit reset
 ++     $ui_comm edit modified false
 ++     rescan ui_ready
 ++}
 ++
 ++set GIT_COMMITTER_IDENT {}
 ++
 ++proc committer_ident {} {
 ++     global GIT_COMMITTER_IDENT
 ++
 ++     if {$GIT_COMMITTER_IDENT eq {}} {
 ++             if {[catch {set me [git var GIT_COMMITTER_IDENT]} err]} {
 ++                     error_popup [strcat [mc "Unable to obtain your identity:"] "\n\n$err"]
 ++                     return {}
 ++             }
 ++             if {![regexp {^(.*) [0-9]+ [-+0-9]+$} \
 ++                     $me me GIT_COMMITTER_IDENT]} {
 ++                     error_popup [strcat [mc "Invalid GIT_COMMITTER_IDENT:"] "\n\n$me"]
 ++                     return {}
 ++             }
 ++     }
 ++
 ++     return $GIT_COMMITTER_IDENT
 ++}
 ++
 ++proc do_signoff {} {
 ++     global ui_comm
 ++
 ++     set me [committer_ident]
 ++     if {$me eq {}} return
 ++
 ++     set sob "Signed-off-by: $me"
 ++     set last [$ui_comm get {end -1c linestart} {end -1c}]
 ++     if {$last ne $sob} {
 ++             $ui_comm edit separator
 ++             if {$last ne {}
 ++                     && ![regexp {^[A-Z][A-Za-z]*-[A-Za-z-]+: *} $last]} {
 ++                     $ui_comm insert end "\n"
 ++             }
 ++             $ui_comm insert end "\n$sob"
 ++             $ui_comm edit separator
 ++             $ui_comm see end
 ++     }
 ++}
 ++
 ++proc create_new_commit {} {
 ++     global commit_type ui_comm
 ++
 ++     set commit_type normal
 ++     $ui_comm delete 0.0 end
 ++     $ui_comm edit reset
 ++     $ui_comm edit modified false
 ++     rescan ui_ready
 ++}
 ++
 ++proc commit_tree {} {
 ++     global HEAD commit_type file_states ui_comm repo_config
 ++     global pch_error
 ++
 ++     if {[committer_ident] eq {}} return
 ++     if {![lock_index update]} return
 ++
 ++     # -- Our in memory state should match the repository.
 ++     #
 ++     repository_state curType curHEAD curMERGE_HEAD
 ++     if {[string match amend* $commit_type]
 ++             && $curType eq {normal}
 ++             && $curHEAD eq $HEAD} {
 ++     } elseif {$commit_type ne $curType || $HEAD ne $curHEAD} {
 ++             info_popup [mc "Last scanned state does not match repository state.
 ++
 ++Another Git program has modified this repository since the last scan.  A rescan must be performed before another commit can be created.
 ++
 ++The rescan will be automatically started now.
 ++"]
 ++             unlock_index
 ++             rescan ui_ready
 ++             return
 ++     }
 ++
 ++     # -- At least one file should differ in the index.
 ++     #
 ++     set files_ready 0
 ++     foreach path [array names file_states] {
 ++             switch -glob -- [lindex $file_states($path) 0] {
 ++             _? {continue}
 ++             A? -
 ++             D? -
+++             T_ -
 ++             M? {set files_ready 1}
+++             _U -
 ++             U? {
 ++                     error_popup [mc "Unmerged files cannot be committed.
 ++
 ++File %s has merge conflicts.  You must resolve them and stage the file before committing.
 ++" [short_path $path]]
 ++                     unlock_index
 ++                     return
 ++             }
 ++             default {
 ++                     error_popup [mc "Unknown file state %s detected.
 ++
 ++File %s cannot be committed by this program.
 ++" [lindex $s 0] [short_path $path]]
 ++             }
 ++             }
 ++     }
 ++     if {!$files_ready && ![string match *merge $curType]} {
 ++             info_popup [mc "No changes to commit.
 ++
 ++You must stage at least 1 file before you can commit.
 ++"]
 ++             unlock_index
 ++             return
 ++     }
 ++
 ++     # -- A message is required.
 ++     #
 ++     set msg [string trim [$ui_comm get 1.0 end]]
 ++     regsub -all -line {[ \t\r]+$} $msg {} msg
 ++     if {$msg eq {}} {
 ++             error_popup [mc "Please supply a commit message.
 ++
 ++A good commit message has the following format:
 ++
 ++- First line: Describe in one sentence what you did.
 ++- Second line: Blank
 ++- Remaining lines: Describe why this change is good.
 ++"]
 ++             unlock_index
 ++             return
 ++     }
 ++
 ++     # -- Build the message file.
 ++     #
 ++     set msg_p [gitdir GITGUI_EDITMSG]
 ++     set msg_wt [open $msg_p w]
 ++     fconfigure $msg_wt -translation lf
 ++     if {[catch {set enc $repo_config(i18n.commitencoding)}]} {
 ++             set enc utf-8
 ++     }
 ++     set use_enc [tcl_encoding $enc]
 ++     if {$use_enc ne {}} {
 ++             fconfigure $msg_wt -encoding $use_enc
 ++     } else {
 ++             puts stderr [mc "warning: Tcl does not support encoding '%s'." $enc]
 ++             fconfigure $msg_wt -encoding utf-8
 ++     }
 ++     puts $msg_wt $msg
 ++     close $msg_wt
 ++
 ++     # -- Run the pre-commit hook.
 ++     #
 ++     set fd_ph [githook_read pre-commit]
 ++     if {$fd_ph eq {}} {
 ++             commit_commitmsg $curHEAD $msg_p
 ++             return
 ++     }
 ++
 ++     ui_status [mc "Calling pre-commit hook..."]
 ++     set pch_error {}
 ++     fconfigure $fd_ph -blocking 0 -translation binary -eofchar {}
 ++     fileevent $fd_ph readable \
 ++             [list commit_prehook_wait $fd_ph $curHEAD $msg_p]
 ++}
 ++
 ++proc commit_prehook_wait {fd_ph curHEAD msg_p} {
 ++     global pch_error
 ++
 ++     append pch_error [read $fd_ph]
 ++     fconfigure $fd_ph -blocking 1
 ++     if {[eof $fd_ph]} {
 ++             if {[catch {close $fd_ph}]} {
 ++                     catch {file delete $msg_p}
 ++                     ui_status [mc "Commit declined by pre-commit hook."]
 ++                     hook_failed_popup pre-commit $pch_error
 ++                     unlock_index
 ++             } else {
 ++                     commit_commitmsg $curHEAD $msg_p
 ++             }
 ++             set pch_error {}
 ++             return
 ++     }
 ++     fconfigure $fd_ph -blocking 0
 ++}
 ++
 ++proc commit_commitmsg {curHEAD msg_p} {
 ++     global pch_error
 ++
 ++     # -- Run the commit-msg hook.
 ++     #
 ++     set fd_ph [githook_read commit-msg $msg_p]
 ++     if {$fd_ph eq {}} {
 ++             commit_writetree $curHEAD $msg_p
 ++             return
 ++     }
 ++
 ++     ui_status [mc "Calling commit-msg hook..."]
 ++     set pch_error {}
 ++     fconfigure $fd_ph -blocking 0 -translation binary -eofchar {}
 ++     fileevent $fd_ph readable \
 ++             [list commit_commitmsg_wait $fd_ph $curHEAD $msg_p]
 ++}
 ++
 ++proc commit_commitmsg_wait {fd_ph curHEAD msg_p} {
 ++     global pch_error
 ++
 ++     append pch_error [read $fd_ph]
 ++     fconfigure $fd_ph -blocking 1
 ++     if {[eof $fd_ph]} {
 ++             if {[catch {close $fd_ph}]} {
 ++                     catch {file delete $msg_p}
 ++                     ui_status [mc "Commit declined by commit-msg hook."]
 ++                     hook_failed_popup commit-msg $pch_error
 ++                     unlock_index
 ++             } else {
 ++                     commit_writetree $curHEAD $msg_p
 ++             }
 ++             set pch_error {}
 ++             return
 ++     }
 ++     fconfigure $fd_ph -blocking 0
 ++}
 ++
 ++proc commit_writetree {curHEAD msg_p} {
 ++     ui_status [mc "Committing changes..."]
 ++     set fd_wt [git_read write-tree]
 ++     fileevent $fd_wt readable \
 ++             [list commit_committree $fd_wt $curHEAD $msg_p]
 ++}
 ++
 ++proc commit_committree {fd_wt curHEAD msg_p} {
 ++     global HEAD PARENT MERGE_HEAD commit_type
 ++     global current_branch
 ++     global ui_comm selected_commit_type
 ++     global file_states selected_paths rescan_active
 ++     global repo_config
 ++
 ++     gets $fd_wt tree_id
 ++     if {[catch {close $fd_wt} err]} {
 ++             catch {file delete $msg_p}
 ++             error_popup [strcat [mc "write-tree failed:"] "\n\n$err"]
 ++             ui_status [mc "Commit failed."]
 ++             unlock_index
 ++             return
 ++     }
 ++
 ++     # -- Verify this wasn't an empty change.
 ++     #
 ++     if {$commit_type eq {normal}} {
 ++             set fd_ot [git_read cat-file commit $PARENT]
 ++             fconfigure $fd_ot -encoding binary -translation lf
 ++             set old_tree [gets $fd_ot]
 ++             close $fd_ot
 ++
 ++             if {[string equal -length 5 {tree } $old_tree]
 ++                     && [string length $old_tree] == 45} {
 ++                     set old_tree [string range $old_tree 5 end]
 ++             } else {
 ++                     error [mc "Commit %s appears to be corrupt" $PARENT]
 ++             }
 ++
 ++             if {$tree_id eq $old_tree} {
 ++                     catch {file delete $msg_p}
 ++                     info_popup [mc "No changes to commit.
 ++
 ++No files were modified by this commit and it was not a merge commit.
 ++
 ++A rescan will be automatically started now.
 ++"]
 ++                     unlock_index
 ++                     rescan {ui_status [mc "No changes to commit."]}
 ++                     return
 ++             }
 ++     }
 ++
 ++     # -- Create the commit.
 ++     #
 ++     set cmd [list commit-tree $tree_id]
 ++     foreach p [concat $PARENT $MERGE_HEAD] {
 ++             lappend cmd -p $p
 ++     }
 ++     lappend cmd <$msg_p
 ++     if {[catch {set cmt_id [eval git $cmd]} err]} {
 ++             catch {file delete $msg_p}
 ++             error_popup [strcat [mc "commit-tree failed:"] "\n\n$err"]
 ++             ui_status [mc "Commit failed."]
 ++             unlock_index
 ++             return
 ++     }
 ++
 ++     # -- Update the HEAD ref.
 ++     #
 ++     set reflogm commit
 ++     if {$commit_type ne {normal}} {
 ++             append reflogm " ($commit_type)"
 ++     }
 ++     set msg_fd [open $msg_p r]
 ++     gets $msg_fd subject
 ++     close $msg_fd
 ++     append reflogm {: } $subject
 ++     if {[catch {
 ++                     git update-ref -m $reflogm HEAD $cmt_id $curHEAD
 ++             } err]} {
 ++             catch {file delete $msg_p}
 ++             error_popup [strcat [mc "update-ref failed:"] "\n\n$err"]
 ++             ui_status [mc "Commit failed."]
 ++             unlock_index
 ++             return
 ++     }
 ++
 ++     # -- Cleanup after ourselves.
 ++     #
 ++     catch {file delete $msg_p}
 ++     catch {file delete [gitdir MERGE_HEAD]}
 ++     catch {file delete [gitdir MERGE_MSG]}
 ++     catch {file delete [gitdir SQUASH_MSG]}
 ++     catch {file delete [gitdir GITGUI_MSG]}
 ++
 ++     # -- Let rerere do its thing.
 ++     #
 ++     if {[get_config rerere.enabled] eq {}} {
 ++             set rerere [file isdirectory [gitdir rr-cache]]
 ++     } else {
 ++             set rerere [is_config_true rerere.enabled]
 ++     }
 ++     if {$rerere} {
 ++             catch {git rerere}
 ++     }
 ++
 ++     # -- Run the post-commit hook.
 ++     #
 ++     set fd_ph [githook_read post-commit]
 ++     if {$fd_ph ne {}} {
 ++             upvar #0 pch_error$cmt_id pc_err
 ++             set pc_err {}
 ++             fconfigure $fd_ph -blocking 0 -translation binary -eofchar {}
 ++             fileevent $fd_ph readable \
 ++                     [list commit_postcommit_wait $fd_ph $cmt_id]
 ++     }
 ++
 ++     $ui_comm delete 0.0 end
 ++     $ui_comm edit reset
 ++     $ui_comm edit modified false
 ++     if {$::GITGUI_BCK_exists} {
 ++             catch {file delete [gitdir GITGUI_BCK]}
 ++             set ::GITGUI_BCK_exists 0
 ++     }
 ++
 ++     if {[is_enabled singlecommit]} do_quit
 ++
 ++     # -- Update in memory status
 ++     #
 ++     set selected_commit_type new
 ++     set commit_type normal
 ++     set HEAD $cmt_id
 ++     set PARENT $cmt_id
 ++     set MERGE_HEAD [list]
 ++
 ++     foreach path [array names file_states] {
 ++             set s $file_states($path)
 ++             set m [lindex $s 0]
 ++             switch -glob -- $m {
 ++             _O -
 ++             _M -
 ++             _D {continue}
 ++             __ -
 ++             A_ -
 ++             M_ -
+++             T_ -
 ++             D_ {
 ++                     unset file_states($path)
 ++                     catch {unset selected_paths($path)}
 ++             }
 ++             DO {
 ++                     set file_states($path) [list _O [lindex $s 1] {} {}]
 ++             }
 ++             AM -
 ++             AD -
 ++             MM -
 ++             MD {
 ++                     set file_states($path) [list \
 ++                             _[string index $m 1] \
 ++                             [lindex $s 1] \
 ++                             [lindex $s 3] \
 ++                             {}]
 ++             }
 ++             }
 ++     }
 ++
 ++     display_all_files
 ++     unlock_index
 ++     reshow_diff
 ++     ui_status [mc "Created commit %s: %s" [string range $cmt_id 0 7] $subject]
 ++}
 ++
 ++proc commit_postcommit_wait {fd_ph cmt_id} {
 ++     upvar #0 pch_error$cmt_id pch_error
 ++
 ++     append pch_error [read $fd_ph]
 ++     fconfigure $fd_ph -blocking 1
 ++     if {[eof $fd_ph]} {
 ++             if {[catch {close $fd_ph}]} {
 ++                     hook_failed_popup post-commit $pch_error 0
 ++             }
 ++             unset pch_error
 ++             return
 ++     }
 ++     fconfigure $fd_ph -blocking 0
 ++}
index 1970b601e1ce34af120cfc3ee16e1b2c26c6b19d,0000000000000000000000000000000000000000,0000000000000000000000000000000000000000..a30c80a935b58ce0c783dfffde196a85eadca9a2
mode 100644,000000,000000..100644
--- /dev/null
--- /dev/null
@@@@ -1,523 -1,0 -1,0 +1,613 @@@@
-       } elseif {$current_diff_side eq {}
-               || [catch {set s $file_states($p)}]
-               || [lsearch -sorted -exact $file_lists($current_diff_side) $p] == -1} {
 ++# git-gui diff viewer
 ++# Copyright (C) 2006, 2007 Shawn Pearce
 ++
 ++proc clear_diff {} {
 ++     global ui_diff current_diff_path current_diff_header
 ++     global ui_index ui_workdir
 ++
 ++     $ui_diff conf -state normal
 ++     $ui_diff delete 0.0 end
 ++     $ui_diff conf -state disabled
 ++
 ++     set current_diff_path {}
 ++     set current_diff_header {}
 ++
 ++     $ui_index tag remove in_diff 0.0 end
 ++     $ui_workdir tag remove in_diff 0.0 end
 ++}
 ++
 ++proc reshow_diff {} {
 ++     global file_states file_lists
 ++     global current_diff_path current_diff_side
 ++     global ui_diff
 ++
 ++     set p $current_diff_path
 ++     if {$p eq {}} {
 ++             # No diff is being shown.
-       set is_3way_diff 0
-       set diff_active 1
+++     } elseif {$current_diff_side eq {}} {
 ++             clear_diff
+++     } elseif {[catch {set s $file_states($p)}]
+++             || [lsearch -sorted -exact $file_lists($current_diff_side) $p] == -1} {
+++
+++             if {[find_next_diff $current_diff_side $p {} {[^O]}]} {
+++                     next_diff
+++             } else {
+++                     clear_diff
+++             }
 ++     } else {
 ++             set save_pos [lindex [$ui_diff yview] 0]
 ++             show_diff $p $current_diff_side {} $save_pos
 ++     }
 ++}
 ++
 ++proc handle_empty_diff {} {
 ++     global current_diff_path file_states file_lists
 ++
 ++     set path $current_diff_path
 ++     set s $file_states($path)
 ++     if {[lindex $s 0] ne {_M}} return
 ++
 ++     info_popup [mc "No differences detected.
 ++
 ++%s has no changes.
 ++
 ++The modification date of this file was updated by another application, but the content within the file was not changed.
 ++
 ++A rescan will be automatically started to find other files which may have the same state." [short_path $path]]
 ++
 ++     clear_diff
 ++     display_file $path __
 ++     rescan ui_ready 0
 ++}
 ++
 ++proc show_diff {path w {lno {}} {scroll_pos {}}} {
 ++     global file_states file_lists
 ++     global is_3way_diff diff_active repo_config
 ++     global ui_diff ui_index ui_workdir
 ++     global current_diff_path current_diff_side current_diff_header
+++     global current_diff_queue
 ++
 ++     if {$diff_active || ![lock_index read]} return
 ++
 ++     clear_diff
 ++     if {$lno == {}} {
 ++             set lno [lsearch -sorted -exact $file_lists($w) $path]
 ++             if {$lno >= 0} {
 ++                     incr lno
 ++             }
 ++     }
 ++     if {$lno >= 1} {
 ++             $w tag add in_diff $lno.0 [expr {$lno + 1}].0
+++             $w see $lno.0
 ++     }
 ++
 ++     set s $file_states($path)
 ++     set m [lindex $s 0]
-       set current_diff_header {}
 ++     set current_diff_path $path
 ++     set current_diff_side $w
-               if {[string index $m 0] eq {U}} {
+++     set current_diff_queue {}
 ++     ui_status [mc "Loading diff of %s..." [escape_path $path]]
 ++
+++     if {[string first {U} $m] >= 0} {
+++             merge_load_stages $path [list show_unmerged_diff $scroll_pos]
+++     } elseif {$m eq {_O}} {
+++             show_other_diff $path $w $m $scroll_pos
+++     } else {
+++             start_show_diff $scroll_pos
+++     }
+++}
+++
+++proc show_unmerged_diff {scroll_pos} {
+++     global current_diff_path current_diff_side
+++     global merge_stages ui_diff
+++     global current_diff_queue
+++
+++     if {$merge_stages(2) eq {}} {
+++             lappend current_diff_queue \
+++                     [list "LOCAL: deleted\nREMOTE:\n" d======= \
+++                         [list ":1:$current_diff_path" ":3:$current_diff_path"]]
+++     } elseif {$merge_stages(3) eq {}} {
+++             lappend current_diff_queue \
+++                     [list "REMOTE: deleted\nLOCAL:\n" d======= \
+++                         [list ":1:$current_diff_path" ":2:$current_diff_path"]]
+++     } elseif {[lindex $merge_stages(1) 0] eq {120000}
+++             || [lindex $merge_stages(2) 0] eq {120000}
+++             || [lindex $merge_stages(3) 0] eq {120000}} {
+++             lappend current_diff_queue \
+++                     [list "LOCAL:\n" d======= \
+++                         [list ":1:$current_diff_path" ":2:$current_diff_path"]]
+++             lappend current_diff_queue \
+++                     [list "REMOTE:\n" d======= \
+++                         [list ":1:$current_diff_path" ":3:$current_diff_path"]]
+++     } else {
+++             start_show_diff $scroll_pos
+++             return
+++     }
+++
+++     advance_diff_queue $scroll_pos
+++}
+++
+++proc advance_diff_queue {scroll_pos} {
+++     global current_diff_queue ui_diff
+++
+++     set item [lindex $current_diff_queue 0]
+++     set current_diff_queue [lrange $current_diff_queue 1 end]
+++
+++     $ui_diff conf -state normal
+++     $ui_diff insert end [lindex $item 0] [lindex $item 1]
+++     $ui_diff conf -state disabled
+++
+++     start_show_diff $scroll_pos [lindex $item 2]
+++}
+++
+++proc show_other_diff {path w m scroll_pos} {
+++     global file_states file_lists
+++     global is_3way_diff diff_active repo_config
+++     global ui_diff ui_index ui_workdir
+++     global current_diff_path current_diff_side current_diff_header
+++
 ++     # - Git won't give us the diff, there's nothing to compare to!
 ++     #
 ++     if {$m eq {_O}} {
 ++             set max_sz [expr {128 * 1024}]
 ++             set type unknown
 ++             if {[catch {
 ++                             set type [file type $path]
 ++                             switch -- $type {
 ++                             directory {
 ++                                     set type submodule
 ++                                     set content {}
 ++                                     set sz 0
 ++                             }
 ++                             link {
 ++                                     set content [file readlink $path]
 ++                                     set sz [string length $content]
 ++                             }
 ++                             file {
 ++                                     set fd [open $path r]
 ++                                     fconfigure $fd -eofchar {}
 ++                                     set content [read $fd $max_sz]
 ++                                     close $fd
 ++                                     set sz [file size $path]
 ++                             }
 ++                             default {
 ++                                     error "'$type' not supported"
 ++                             }
 ++                             }
 ++                     } err ]} {
 ++                     set diff_active 0
 ++                     unlock_index
 ++                     ui_status [mc "Unable to display %s" [escape_path $path]]
 ++                     error_popup [strcat [mc "Error loading file:"] "\n\n$err"]
 ++                     return
 ++             }
 ++             $ui_diff conf -state normal
 ++             if {$type eq {submodule}} {
 ++                     $ui_diff insert end [append \
 ++                             "* " \
 ++                             [mc "Git Repository (subproject)"] \
 ++                             "\n"] d_@
 ++             } elseif {![catch {set type [exec file $path]}]} {
 ++                     set n [string length $path]
 ++                     if {[string equal -length $n $path $type]} {
 ++                             set type [string range $type $n end]
 ++                             regsub {^:?\s*} $type {} type
 ++                     }
 ++                     $ui_diff insert end "* $type\n" d_@
 ++             }
 ++             if {[string first "\0" $content] != -1} {
 ++                     $ui_diff insert end \
 ++                             [mc "* Binary file (not showing content)."] \
 ++                             d_@
 ++             } else {
 ++                     if {$sz > $max_sz} {
 ++                             $ui_diff insert end \
 ++"* Untracked file is $sz bytes.
 ++* Showing only first $max_sz bytes.
 ++" d_@
 ++                     }
 ++                     $ui_diff insert end $content
 ++                     if {$sz > $max_sz} {
 ++                             $ui_diff insert end "
 ++* Untracked file clipped here by [appname].
 ++* To see the entire file, use an external editor.
 ++" d_@
 ++                     }
 ++             }
 ++             $ui_diff conf -state disabled
 ++             set diff_active 0
 ++             unlock_index
 ++             if {$scroll_pos ne {}} {
 ++                     update
 ++                     $ui_diff yview moveto $scroll_pos
 ++             }
 ++             ui_ready
 ++             return
 ++     }
+++}
+++
+++proc start_show_diff {scroll_pos {add_opts {}}} {
+++     global file_states file_lists
+++     global is_3way_diff diff_active repo_config
+++     global ui_diff ui_index ui_workdir
+++     global current_diff_path current_diff_side current_diff_header
+++
+++     set path $current_diff_path
+++     set w $current_diff_side
+++
+++     set s $file_states($path)
+++     set m [lindex $s 0]
+++     set is_3way_diff 0
+++     set diff_active 1
+++     set current_diff_header {}
 ++
 ++     set cmd [list]
 ++     if {$w eq $ui_index} {
 ++             lappend cmd diff-index
 ++             lappend cmd --cached
 ++     } elseif {$w eq $ui_workdir} {
-       lappend cmd --
-       lappend cmd $path
+++             if {[string first {U} $m] >= 0} {
 ++                     lappend cmd diff
 ++             } else {
 ++                     lappend cmd diff-files
 ++             }
 ++     }
 ++
 ++     lappend cmd -p
 ++     lappend cmd --no-color
 ++     if {$repo_config(gui.diffcontext) >= 1} {
 ++             lappend cmd "-U$repo_config(gui.diffcontext)"
 ++     }
 ++     if {$w eq $ui_index} {
 ++             lappend cmd [PARENT]
 ++     }
-               clear_diff
-       } else {
-               set current_diff_path $current_diff_path
+++     if {$add_opts ne {}} {
+++             eval lappend cmd $add_opts
+++     } else {
+++             lappend cmd --
+++             lappend cmd $path
+++     }
 ++
 ++     if {[catch {set fd [eval git_read --nice $cmd]} err]} {
 ++             set diff_active 0
 ++             unlock_index
 ++             ui_status [mc "Unable to display %s" [escape_path $path]]
 ++             error_popup [strcat [mc "Error loading diff:"] "\n\n$err"]
 ++             return
 ++     }
 ++
 ++     set ::current_diff_inheader 1
 ++     fconfigure $fd \
 ++             -blocking 0 \
 ++             -encoding binary \
 ++             -translation binary
 ++     fileevent $fd readable [list read_diff $fd $scroll_pos]
 ++}
 ++
 ++proc read_diff {fd scroll_pos} {
 ++     global ui_diff diff_active
 ++     global is_3way_diff current_diff_header
+++     global current_diff_queue
 ++
 ++     $ui_diff conf -state normal
 ++     while {[gets $fd line] >= 0} {
 ++             # -- Cleanup uninteresting diff header lines.
 ++             #
 ++             if {$::current_diff_inheader} {
 ++                     if {   [string match {diff --git *}      $line]
 ++                         || [string match {diff --cc *}       $line]
 ++                         || [string match {diff --combined *} $line]
 ++                         || [string match {--- *}             $line]
 ++                         || [string match {+++ *}             $line]} {
 ++                             append current_diff_header $line "\n"
 ++                             continue
 ++                     }
 ++             }
 ++             if {[string match {index *} $line]} continue
 ++             if {$line eq {deleted file mode 120000}} {
 ++                     set line "deleted symlink"
 ++             }
 ++             set ::current_diff_inheader 0
 ++
 ++             # -- Automatically detect if this is a 3 way diff.
 ++             #
 ++             if {[string match {@@@ *} $line]} {set is_3way_diff 1}
 ++
 ++             if {[string match {mode *} $line]
 ++                     || [string match {new file *} $line]
 ++                     || [regexp {^(old|new) mode *} $line]
 ++                     || [string match {deleted file *} $line]
 ++                     || [string match {deleted symlink} $line]
 ++                     || [string match {Binary files * and * differ} $line]
 ++                     || $line eq {\ No newline at end of file}
 ++                     || [regexp {^\* Unmerged path } $line]} {
 ++                     set tags {}
 ++             } elseif {$is_3way_diff} {
 ++                     set op [string range $line 0 1]
 ++                     switch -- $op {
 ++                     {  } {set tags {}}
 ++                     {@@} {set tags d_@}
 ++                     { +} {set tags d_s+}
 ++                     { -} {set tags d_s-}
 ++                     {+ } {set tags d_+s}
 ++                     {- } {set tags d_-s}
 ++                     {--} {set tags d_--}
 ++                     {++} {
 ++                             if {[regexp {^\+\+([<>]{7} |={7})} $line _g op]} {
 ++                                     set line [string replace $line 0 1 {  }]
 ++                                     set tags d$op
 ++                             } else {
 ++                                     set tags d_++
 ++                             }
 ++                     }
 ++                     default {
 ++                             puts "error: Unhandled 3 way diff marker: {$op}"
 ++                             set tags {}
 ++                     }
 ++                     }
 ++             } else {
 ++                     set op [string index $line 0]
 ++                     switch -- $op {
 ++                     { } {set tags {}}
 ++                     {@} {set tags d_@}
 ++                     {-} {set tags d_-}
 ++                     {+} {
 ++                             if {[regexp {^\+([<>]{7} |={7})} $line _g op]} {
 ++                                     set line [string replace $line 0 0 { }]
 ++                                     set tags d$op
 ++                             } else {
 ++                                     set tags d_+
 ++                             }
 ++                     }
 ++                     default {
 ++                             puts "error: Unhandled 2 way diff marker: {$op}"
 ++                             set tags {}
 ++                     }
 ++                     }
 ++             }
 ++             $ui_diff insert end $line $tags
 ++             if {[string index $line end] eq "\r"} {
 ++                     $ui_diff tag add d_cr {end - 2c}
 ++             }
 ++             $ui_diff insert end "\n" $tags
 ++     }
 ++     $ui_diff conf -state disabled
 ++
 ++     if {[eof $fd]} {
 ++             close $fd
+++
+++             if {$current_diff_queue ne {}} {
+++                     advance_diff_queue $scroll_pos
+++                     return
+++             }
+++
 ++             set diff_active 0
 ++             unlock_index
 ++             if {$scroll_pos ne {}} {
 ++                     update
 ++                     $ui_diff yview moveto $scroll_pos
 ++             }
 ++             ui_ready
 ++
 ++             if {[$ui_diff index end] eq {2.0}} {
 ++                     handle_empty_diff
 ++             }
 ++     }
 ++}
 ++
 ++proc apply_hunk {x y} {
 ++     global current_diff_path current_diff_header current_diff_side
 ++     global ui_diff ui_index file_states
 ++
 ++     if {$current_diff_path eq {} || $current_diff_header eq {}} return
 ++     if {![lock_index apply_hunk]} return
 ++
 ++     set apply_cmd {apply --cached --whitespace=nowarn}
 ++     set mi [lindex $file_states($current_diff_path) 0]
 ++     if {$current_diff_side eq $ui_index} {
 ++             set failed_msg [mc "Failed to unstage selected hunk."]
 ++             lappend apply_cmd --reverse
 ++             if {[string index $mi 0] ne {M}} {
 ++                     unlock_index
 ++                     return
 ++             }
 ++     } else {
 ++             set failed_msg [mc "Failed to stage selected hunk."]
 ++             if {[string index $mi 1] ne {M}} {
 ++                     unlock_index
 ++                     return
 ++             }
 ++     }
 ++
 ++     set s_lno [lindex [split [$ui_diff index @$x,$y] .] 0]
 ++     set s_lno [$ui_diff search -backwards -regexp ^@@ $s_lno.0 0.0]
 ++     if {$s_lno eq {}} {
 ++             unlock_index
 ++             return
 ++     }
 ++
 ++     set e_lno [$ui_diff search -forwards -regexp ^@@ "$s_lno + 1 lines" end]
 ++     if {$e_lno eq {}} {
 ++             set e_lno end
 ++     }
 ++
 ++     if {[catch {
 ++             set p [eval git_write $apply_cmd]
 ++             fconfigure $p -translation binary -encoding binary
 ++             puts -nonewline $p $current_diff_header
 ++             puts -nonewline $p [$ui_diff get $s_lno $e_lno]
 ++             close $p} err]} {
 ++             error_popup [append $failed_msg "\n\n$err"]
 ++             unlock_index
 ++             return
 ++     }
 ++
 ++     $ui_diff conf -state normal
 ++     $ui_diff delete $s_lno $e_lno
 ++     $ui_diff conf -state disabled
 ++
 ++     if {[$ui_diff get 1.0 end] eq "\n"} {
 ++             set o _
 ++     } else {
 ++             set o ?
 ++     }
 ++
 ++     if {$current_diff_side eq $ui_index} {
 ++             set mi ${o}M
 ++     } elseif {[string index $mi 0] eq {_}} {
 ++             set mi M$o
 ++     } else {
 ++             set mi ?$o
 ++     }
 ++     unlock_index
 ++     display_file $current_diff_path $mi
+++     # This should trigger shift to the next changed file
 ++     if {$o eq {_}} {
+++             reshow_diff
 ++     }
 ++}
 ++
 ++proc apply_line {x y} {
 ++     global current_diff_path current_diff_header current_diff_side
 ++     global ui_diff ui_index file_states
 ++
 ++     if {$current_diff_path eq {} || $current_diff_header eq {}} return
 ++     if {![lock_index apply_hunk]} return
 ++
 ++     set apply_cmd {apply --cached --whitespace=nowarn}
 ++     set mi [lindex $file_states($current_diff_path) 0]
 ++     if {$current_diff_side eq $ui_index} {
 ++             set failed_msg [mc "Failed to unstage selected line."]
 ++             set to_context {+}
 ++             lappend apply_cmd --reverse
 ++             if {[string index $mi 0] ne {M}} {
 ++                     unlock_index
 ++                     return
 ++             }
 ++     } else {
 ++             set failed_msg [mc "Failed to stage selected line."]
 ++             set to_context {-}
 ++             if {[string index $mi 1] ne {M}} {
 ++                     unlock_index
 ++                     return
 ++             }
 ++     }
 ++
 ++     set the_l [$ui_diff index @$x,$y]
 ++
 ++     # operate only on change lines
 ++     set c1 [$ui_diff get "$the_l linestart"]
 ++     if {$c1 ne {+} && $c1 ne {-}} {
 ++             unlock_index
 ++             return
 ++     }
 ++     set sign $c1
 ++
 ++     set i_l [$ui_diff search -backwards -regexp ^@@ $the_l 0.0]
 ++     if {$i_l eq {}} {
 ++             unlock_index
 ++             return
 ++     }
 ++     # $i_l is now at the beginning of a line
 ++
 ++     # pick start line number from hunk header
 ++     set hh [$ui_diff get $i_l "$i_l + 1 lines"]
 ++     set hh [lindex [split $hh ,] 0]
 ++     set hln [lindex [split $hh -] 1]
 ++
 ++     # There is a special situation to take care of. Consider this hunk:
 ++     #
 ++     #    @@ -10,4 +10,4 @@
 ++     #     context before
 ++     #    -old 1
 ++     #    -old 2
 ++     #    +new 1
 ++     #    +new 2
 ++     #     context after
 ++     #
 ++     # We used to keep the context lines in the order they appear in the
 ++     # hunk. But then it is not possible to correctly stage only
 ++     # "-old 1" and "+new 1" - it would result in this staged text:
 ++     #
 ++     #    context before
 ++     #    old 2
 ++     #    new 1
 ++     #    context after
 ++     #
 ++     # (By symmetry it is not possible to *un*stage "old 2" and "new 2".)
 ++     #
 ++     # We resolve the problem by introducing an asymmetry, namely, when
 ++     # a "+" line is *staged*, it is moved in front of the context lines
 ++     # that are generated from the "-" lines that are immediately before
 ++     # the "+" block. That is, we construct this patch:
 ++     #
 ++     #    @@ -10,4 +10,5 @@
 ++     #     context before
 ++     #    +new 1
 ++     #     old 1
 ++     #     old 2
 ++     #     context after
 ++     #
 ++     # But we do *not* treat "-" lines that are *un*staged in a special
 ++     # way.
 ++     #
 ++     # With this asymmetry it is possible to stage the change
 ++     # "old 1" -> "new 1" directly, and to stage the change
 ++     # "old 2" -> "new 2" by first staging the entire hunk and
 ++     # then unstaging the change "old 1" -> "new 1".
 ++
 ++     # This is non-empty if and only if we are _staging_ changes;
 ++     # then it accumulates the consecutive "-" lines (after converting
 ++     # them to context lines) in order to be moved after the "+" change
 ++     # line.
 ++     set pre_context {}
 ++
 ++     set n 0
 ++     set i_l [$ui_diff index "$i_l + 1 lines"]
 ++     set patch {}
 ++     while {[$ui_diff compare $i_l < "end - 1 chars"] &&
 ++            [$ui_diff get $i_l "$i_l + 2 chars"] ne {@@}} {
 ++             set next_l [$ui_diff index "$i_l + 1 lines"]
 ++             set c1 [$ui_diff get $i_l]
 ++             if {[$ui_diff compare $i_l <= $the_l] &&
 ++                 [$ui_diff compare $the_l < $next_l]} {
 ++                     # the line to stage/unstage
 ++                     set ln [$ui_diff get $i_l $next_l]
 ++                     if {$c1 eq {-}} {
 ++                             set n [expr $n+1]
 ++                             set patch "$patch$pre_context$ln"
 ++                     } else {
 ++                             set patch "$patch$ln$pre_context"
 ++                     }
 ++                     set pre_context {}
 ++             } elseif {$c1 ne {-} && $c1 ne {+}} {
 ++                     # context line
 ++                     set ln [$ui_diff get $i_l $next_l]
 ++                     set patch "$patch$pre_context$ln"
 ++                     set n [expr $n+1]
 ++                     set pre_context {}
 ++             } elseif {$c1 eq $to_context} {
 ++                     # turn change line into context line
 ++                     set ln [$ui_diff get "$i_l + 1 chars" $next_l]
 ++                     if {$c1 eq {-}} {
 ++                             set pre_context "$pre_context $ln"
 ++                     } else {
 ++                             set patch "$patch $ln"
 ++                     }
 ++                     set n [expr $n+1]
 ++             }
 ++             set i_l $next_l
 ++     }
 ++     set patch "@@ -$hln,$n +$hln,[eval expr $n $sign 1] @@\n$patch"
 ++
 ++     if {[catch {
 ++             set p [eval git_write $apply_cmd]
 ++             fconfigure $p -translation binary -encoding binary
 ++             puts -nonewline $p $current_diff_header
 ++             puts -nonewline $p $patch
 ++             close $p} err]} {
 ++             error_popup [append $failed_msg "\n\n$err"]
 ++     }
 ++
 ++     unlock_index
 ++}
index 3c1fce7475d362d1880d915ff4bdf168fda28593,0000000000000000000000000000000000000000,0000000000000000000000000000000000000000..b045219a1cca6bf46218bf211c0c88b8d699e144
mode 100644,000000,000000..100644
--- /dev/null
--- /dev/null
@@@@ -1,437 -1,0 -1,0 +1,445 @@@@
-                       {Reverting selected files} \
 ++# git-gui index (add/remove) support
 ++# Copyright (C) 2006, 2007 Shawn Pearce
 ++
 ++proc _delete_indexlock {} {
 ++     if {[catch {file delete -- [gitdir index.lock]} err]} {
 ++             error_popup [strcat [mc "Unable to unlock the index."] "\n\n$err"]
 ++     }
 ++}
 ++
 ++proc _close_updateindex {fd after} {
 ++     fconfigure $fd -blocking 1
 ++     if {[catch {close $fd} err]} {
 ++             set w .indexfried
 ++             toplevel $w
 ++             wm title $w [strcat "[appname] ([reponame]): " [mc "Index Error"]]
 ++             wm geometry $w "+[winfo rootx .]+[winfo rooty .]"
 ++             pack [label $w.msg \
 ++                     -justify left \
 ++                     -anchor w \
 ++                     -text [strcat \
 ++                             [mc "Updating the Git index failed.  A rescan will be automatically started to resynchronize git-gui."] \
 ++                             "\n\n$err"] \
 ++                     ] -anchor w
 ++
 ++             frame $w.buttons
 ++             button $w.buttons.continue \
 ++                     -text [mc "Continue"] \
 ++                     -command [list destroy $w]
 ++             pack $w.buttons.continue -side right -padx 5
 ++             button $w.buttons.unlock \
 ++                     -text [mc "Unlock Index"] \
 ++                     -command "destroy $w; _delete_indexlock"
 ++             pack $w.buttons.unlock -side right
 ++             pack $w.buttons -side bottom -fill x -pady 10 -padx 10
 ++
 ++             wm protocol $w WM_DELETE_WINDOW update
 ++             bind $w.buttons.continue <Visibility> "
 ++                     grab $w
 ++                     focus $w.buttons.continue
 ++             "
 ++             tkwait window $w
 ++
 ++             $::main_status stop
 ++             unlock_index
 ++             rescan $after 0
 ++             return
 ++     }
 ++
 ++     $::main_status stop
 ++     unlock_index
 ++     uplevel #0 $after
 ++}
 ++
 ++proc update_indexinfo {msg pathList after} {
 ++     global update_index_cp
 ++
 ++     if {![lock_index update]} return
 ++
 ++     set update_index_cp 0
 ++     set pathList [lsort $pathList]
 ++     set totalCnt [llength $pathList]
 ++     set batch [expr {int($totalCnt * .01) + 1}]
 ++     if {$batch > 25} {set batch 25}
 ++
 ++     $::main_status start $msg [mc "files"]
 ++     set fd [git_write update-index -z --index-info]
 ++     fconfigure $fd \
 ++             -blocking 0 \
 ++             -buffering full \
 ++             -buffersize 512 \
 ++             -encoding binary \
 ++             -translation binary
 ++     fileevent $fd writable [list \
 ++             write_update_indexinfo \
 ++             $fd \
 ++             $pathList \
 ++             $totalCnt \
 ++             $batch \
 ++             $after \
 ++             ]
 ++}
 ++
 ++proc write_update_indexinfo {fd pathList totalCnt batch after} {
 ++     global update_index_cp
 ++     global file_states current_diff_path
 ++
 ++     if {$update_index_cp >= $totalCnt} {
 ++             _close_updateindex $fd $after
 ++             return
 ++     }
 ++
 ++     for {set i $batch} \
 ++             {$update_index_cp < $totalCnt && $i > 0} \
 ++             {incr i -1} {
 ++             set path [lindex $pathList $update_index_cp]
 ++             incr update_index_cp
 ++
 ++             set s $file_states($path)
 ++             switch -glob -- [lindex $s 0] {
 ++             A? {set new _O}
 ++             M? {set new _M}
+++             T_ {set new _T}
 ++             D_ {set new _D}
 ++             D? {set new _?}
 ++             ?? {continue}
 ++             }
 ++             set info [lindex $s 2]
 ++             if {$info eq {}} continue
 ++
 ++             puts -nonewline $fd "$info\t[encoding convertto $path]\0"
 ++             display_file $path $new
 ++     }
 ++
 ++     $::main_status update $update_index_cp $totalCnt
 ++}
 ++
 ++proc update_index {msg pathList after} {
 ++     global update_index_cp
 ++
 ++     if {![lock_index update]} return
 ++
 ++     set update_index_cp 0
 ++     set pathList [lsort $pathList]
 ++     set totalCnt [llength $pathList]
 ++     set batch [expr {int($totalCnt * .01) + 1}]
 ++     if {$batch > 25} {set batch 25}
 ++
 ++     $::main_status start $msg [mc "files"]
 ++     set fd [git_write update-index --add --remove -z --stdin]
 ++     fconfigure $fd \
 ++             -blocking 0 \
 ++             -buffering full \
 ++             -buffersize 512 \
 ++             -encoding binary \
 ++             -translation binary
 ++     fileevent $fd writable [list \
 ++             write_update_index \
 ++             $fd \
 ++             $pathList \
 ++             $totalCnt \
 ++             $batch \
 ++             $after \
 ++             ]
 ++}
 ++
 ++proc write_update_index {fd pathList totalCnt batch after} {
 ++     global update_index_cp
 ++     global file_states current_diff_path
 ++
 ++     if {$update_index_cp >= $totalCnt} {
 ++             _close_updateindex $fd $after
 ++             return
 ++     }
 ++
 ++     for {set i $batch} \
 ++             {$update_index_cp < $totalCnt && $i > 0} \
 ++             {incr i -1} {
 ++             set path [lindex $pathList $update_index_cp]
 ++             incr update_index_cp
 ++
 ++             switch -glob -- [lindex $file_states($path) 0] {
 ++             AD {set new __}
 ++             ?D {set new D_}
 ++             _O -
 ++             AM {set new A_}
+++             _T {set new T_}
+++             _U -
 ++             U? {
 ++                     if {[file exists $path]} {
 ++                             set new M_
 ++                     } else {
 ++                             set new D_
 ++                     }
 ++             }
 ++             ?M {set new M_}
 ++             ?? {continue}
 ++             }
 ++             puts -nonewline $fd "[encoding convertto $path]\0"
 ++             display_file $path $new
 ++     }
 ++
 ++     $::main_status update $update_index_cp $totalCnt
 ++}
 ++
 ++proc checkout_index {msg pathList after} {
 ++     global update_index_cp
 ++
 ++     if {![lock_index update]} return
 ++
 ++     set update_index_cp 0
 ++     set pathList [lsort $pathList]
 ++     set totalCnt [llength $pathList]
 ++     set batch [expr {int($totalCnt * .01) + 1}]
 ++     if {$batch > 25} {set batch 25}
 ++
 ++     $::main_status start $msg [mc "files"]
 ++     set fd [git_write checkout-index \
 ++             --index \
 ++             --quiet \
 ++             --force \
 ++             -z \
 ++             --stdin \
 ++             ]
 ++     fconfigure $fd \
 ++             -blocking 0 \
 ++             -buffering full \
 ++             -buffersize 512 \
 ++             -encoding binary \
 ++             -translation binary
 ++     fileevent $fd writable [list \
 ++             write_checkout_index \
 ++             $fd \
 ++             $pathList \
 ++             $totalCnt \
 ++             $batch \
 ++             $after \
 ++             ]
 ++}
 ++
 ++proc write_checkout_index {fd pathList totalCnt batch after} {
 ++     global update_index_cp
 ++     global file_states current_diff_path
 ++
 ++     if {$update_index_cp >= $totalCnt} {
 ++             _close_updateindex $fd $after
 ++             return
 ++     }
 ++
 ++     for {set i $batch} \
 ++             {$update_index_cp < $totalCnt && $i > 0} \
 ++             {incr i -1} {
 ++             set path [lindex $pathList $update_index_cp]
 ++             incr update_index_cp
 ++             switch -glob -- [lindex $file_states($path) 0] {
 ++             U? {continue}
 ++             ?M -
+++             ?T -
 ++             ?D {
 ++                     puts -nonewline $fd "[encoding convertto $path]\0"
 ++                     display_file $path ?_
 ++             }
 ++             }
 ++     }
 ++
 ++     $::main_status update $update_index_cp $totalCnt
 ++}
 ++
 ++proc unstage_helper {txt paths} {
 ++     global file_states current_diff_path
 ++
 ++     if {![lock_index begin-update]} return
 ++
 ++     set pathList [list]
 ++     set after {}
 ++     foreach path $paths {
 ++             switch -glob -- [lindex $file_states($path) 0] {
 ++             A? -
 ++             M? -
+++             T_ -
 ++             D? {
 ++                     lappend pathList $path
 ++                     if {$path eq $current_diff_path} {
 ++                             set after {reshow_diff;}
 ++                     }
 ++             }
 ++             }
 ++     }
 ++     if {$pathList eq {}} {
 ++             unlock_index
 ++     } else {
 ++             update_indexinfo \
 ++                     $txt \
 ++                     $pathList \
 ++                     [concat $after [list ui_ready]]
 ++     }
 ++}
 ++
 ++proc do_unstage_selection {} {
 ++     global current_diff_path selected_paths
 ++
 ++     if {[array size selected_paths] > 0} {
 ++             unstage_helper \
 ++                     {Unstaging selected files from commit} \
 ++                     [array names selected_paths]
 ++     } elseif {$current_diff_path ne {}} {
 ++             unstage_helper \
 ++                     [mc "Unstaging %s from commit" [short_path $current_diff_path]] \
 ++                     [list $current_diff_path]
 ++     }
 ++}
 ++
 ++proc add_helper {txt paths} {
 ++     global file_states current_diff_path
 ++
 ++     if {![lock_index begin-update]} return
 ++
 ++     set pathList [list]
 ++     set after {}
 ++     foreach path $paths {
 ++             switch -glob -- [lindex $file_states($path) 0] {
 ++             _O -
 ++             ?M -
 ++             ?D -
+++             ?T -
 ++             U? {
 ++                     lappend pathList $path
 ++                     if {$path eq $current_diff_path} {
 ++                             set after {reshow_diff;}
 ++                     }
 ++             }
 ++             }
 ++     }
 ++     if {$pathList eq {}} {
 ++             unlock_index
 ++     } else {
 ++             update_index \
 ++                     $txt \
 ++                     $pathList \
 ++                     [concat $after {ui_status [mc "Ready to commit."]}]
 ++     }
 ++}
 ++
 ++proc do_add_selection {} {
 ++     global current_diff_path selected_paths
 ++
 ++     if {[array size selected_paths] > 0} {
 ++             add_helper \
 ++                     {Adding selected files} \
 ++                     [array names selected_paths]
 ++     } elseif {$current_diff_path ne {}} {
 ++             add_helper \
 ++                     [mc "Adding %s" [short_path $current_diff_path]] \
 ++                     [list $current_diff_path]
 ++     }
 ++}
 ++
 ++proc do_add_all {} {
 ++     global file_states
 ++
 ++     set paths [list]
 ++     foreach path [array names file_states] {
 ++             switch -glob -- [lindex $file_states($path) 0] {
 ++             U? {continue}
 ++             ?M -
+++             ?T -
 ++             ?D {lappend paths $path}
 ++             }
 ++     }
 ++     add_helper {Adding all changed files} $paths
 ++}
 ++
 ++proc revert_helper {txt paths} {
 ++     global file_states current_diff_path
 ++
 ++     if {![lock_index begin-update]} return
 ++
 ++     set pathList [list]
 ++     set after {}
 ++     foreach path $paths {
 ++             switch -glob -- [lindex $file_states($path) 0] {
 ++             U? {continue}
 ++             ?M -
+++             ?T -
 ++             ?D {
 ++                     lappend pathList $path
 ++                     if {$path eq $current_diff_path} {
 ++                             set after {reshow_diff;}
 ++                     }
 ++             }
 ++             }
 ++     }
 ++
 ++
 ++     # Split question between singular and plural cases, because
 ++     # such distinction is needed in some languages. Previously, the
 ++     # code used "Revert changes in" for both, but that can't work
 ++     # in languages where 'in' must be combined with word from
 ++     # rest of string (in diffrent way for both cases of course).
 ++     #
 ++     # FIXME: Unfortunately, even that isn't enough in some languages
 ++     # as they have quite complex plural-form rules. Unfortunately,
 ++     # msgcat doesn't seem to support that kind of string translation.
 ++     #
 ++     set n [llength $pathList]
 ++     if {$n == 0} {
 ++             unlock_index
 ++             return
 ++     } elseif {$n == 1} {
 ++             set query [mc "Revert changes in file %s?" [short_path [lindex $pathList]]]
 ++     } else {
 ++             set query [mc "Revert changes in these %i files?" $n]
 ++     }
 ++
 ++     set reply [tk_dialog \
 ++             .confirm_revert \
 ++             "[appname] ([reponame])" \
 ++             "$query
 ++
 ++[mc "Any unstaged changes will be permanently lost by the revert."]" \
 ++             question \
 ++             1 \
 ++             [mc "Do Nothing"] \
 ++             [mc "Revert Changes"] \
 ++             ]
 ++     if {$reply == 1} {
 ++             checkout_index \
 ++                     $txt \
 ++                     $pathList \
 ++                     [concat $after [list ui_ready]]
 ++     } else {
 ++             unlock_index
 ++     }
 ++}
 ++
 ++proc do_revert_selection {} {
 ++     global current_diff_path selected_paths
 ++
 ++     if {[array size selected_paths] > 0} {
 ++             revert_helper \
-                       "Reverting [short_path $current_diff_path]" \
+++                     [mc "Reverting selected files"] \
 ++                     [array names selected_paths]
 ++     } elseif {$current_diff_path ne {}} {
 ++             revert_helper \
+++                     [mc "Reverting %s" [short_path $current_diff_path]] \
 ++                     [list $current_diff_path]
 ++     }
 ++}
 ++
 ++proc do_select_commit_type {} {
 ++     global commit_type selected_commit_type
 ++
 ++     if {$selected_commit_type eq {new}
 ++             && [string match amend* $commit_type]} {
 ++             create_new_commit
 ++     } elseif {$selected_commit_type eq {amend}
 ++             && ![string match amend* $commit_type]} {
 ++             load_last_commit
 ++
 ++             # The amend request was rejected...
 ++             #
 ++             if {![string match amend* $commit_type]} {
 ++                     set selected_commit_type new
 ++             }
 ++     }
 ++}
index 0000000000000000000000000000000000000000,0000000000000000000000000000000000000000,0000000000000000000000000000000000000000..79c58bc7bc8c79a13acd1370b184a530d67eaa20
new file mode 100644 (file)
--- /dev/null
--- /dev/null
--- /dev/null
@@@@ -1,0 -1,0 -1,0 +1,373 @@@@
+++# git-gui merge conflict resolution
+++# parts based on git-mergetool (c) 2006 Theodore Y. Ts'o
+++
+++proc merge_resolve_one {stage} {
+++     global current_diff_path
+++
+++     switch -- $stage {
+++             1 { set target [mc "the base version"] }
+++             2 { set target [mc "this branch"] }
+++             3 { set target [mc "the other branch"] }
+++     }
+++
+++     set op_question [mc "Force resolution to %s?
+++Note that the diff shows only conflicting changes.
+++
+++%s will be overwritten.
+++
+++This operation can be undone only by restarting the merge." \
+++             $target [short_path $current_diff_path]]
+++
+++     if {[ask_popup $op_question] eq {yes}} {
+++             merge_load_stages $current_diff_path [list merge_force_stage $stage]
+++     }
+++}
+++
+++proc merge_add_resolution {path} {
+++     global current_diff_path ui_workdir
+++
+++     set after [next_diff_after_action $ui_workdir $path {} {^_?U}]
+++
+++     update_index \
+++             [mc "Adding resolution for %s" [short_path $path]] \
+++             [list $path] \
+++             [concat $after [list ui_ready]]
+++}
+++
+++proc merge_force_stage {stage} {
+++     global current_diff_path merge_stages
+++
+++     if {$merge_stages($stage) ne {}} {
+++             git checkout-index -f --stage=$stage -- $current_diff_path
+++     } else {
+++             file delete -- $current_diff_path
+++     }
+++
+++     merge_add_resolution $current_diff_path
+++}
+++
+++proc merge_load_stages {path cont} {
+++     global merge_stages_fd merge_stages merge_stages_buf
+++
+++     if {[info exists merge_stages_fd]} {
+++             catch { kill_file_process $merge_stages_fd }
+++             catch { close $merge_stages_fd }
+++     }
+++
+++     set merge_stages(0) {}
+++     set merge_stages(1) {}
+++     set merge_stages(2) {}
+++     set merge_stages(3) {}
+++     set merge_stages_buf {}
+++
+++     set merge_stages_fd [eval git_read ls-files -u -z -- $path]
+++
+++     fconfigure $merge_stages_fd -blocking 0 -translation binary -encoding binary
+++     fileevent $merge_stages_fd readable [list read_merge_stages $merge_stages_fd $cont]
+++}
+++
+++proc read_merge_stages {fd cont} {
+++     global merge_stages_buf merge_stages_fd merge_stages
+++
+++     append merge_stages_buf [read $fd]
+++     set pck [split $merge_stages_buf "\0"]
+++     set merge_stages_buf [lindex $pck end]
+++
+++     if {[eof $fd] && $merge_stages_buf ne {}} {
+++             lappend pck {}
+++             set merge_stages_buf {}
+++     }
+++
+++     foreach p [lrange $pck 0 end-1] {
+++             set fcols [split $p "\t"]
+++             set cols  [split [lindex $fcols 0] " "]
+++             set stage [lindex $cols 2]
+++             
+++             set merge_stages($stage) [lrange $cols 0 1]
+++     }
+++
+++     if {[eof $fd]} {
+++             close $fd
+++             unset merge_stages_fd
+++             eval $cont
+++     }
+++}
+++
+++proc merge_resolve_tool {} {
+++     global current_diff_path
+++
+++     merge_load_stages $current_diff_path [list merge_resolve_tool2]
+++}
+++
+++proc merge_resolve_tool2 {} {
+++     global current_diff_path merge_stages
+++
+++     # Validate the stages
+++     if {$merge_stages(2) eq {} ||
+++         [lindex $merge_stages(2) 0] eq {120000} ||
+++         [lindex $merge_stages(2) 0] eq {160000} ||
+++         $merge_stages(3) eq {} ||
+++         [lindex $merge_stages(3) 0] eq {120000} ||
+++         [lindex $merge_stages(3) 0] eq {160000}
+++     } {
+++             error_popup [mc "Cannot resolve deletion or link conflicts using a tool"]
+++             return
+++     }
+++
+++     if {![file exists $current_diff_path]} {
+++             error_popup [mc "Conflict file does not exist"]
+++             return
+++     }
+++
+++     # Determine the tool to use
+++     set tool [get_config merge.tool]
+++     if {$tool eq {}} { set tool meld }
+++
+++     set merge_tool_path [get_config "mergetool.$tool.path"]
+++     if {$merge_tool_path eq {}} {
+++             switch -- $tool {
+++             emerge { set merge_tool_path "emacs" }
+++             araxis { set merge_tool_path "compare" }
+++             default { set merge_tool_path $tool }
+++             }
+++     }
+++
+++     # Make file names
+++     set filebase [file rootname $current_diff_path]
+++     set fileext  [file extension $current_diff_path]
+++     set basename [lindex [file split $current_diff_path] end]
+++
+++     set MERGED   $current_diff_path
+++     set BASE     "./$MERGED.BASE$fileext"
+++     set LOCAL    "./$MERGED.LOCAL$fileext"
+++     set REMOTE   "./$MERGED.REMOTE$fileext"
+++     set BACKUP   "./$MERGED.BACKUP$fileext"
+++
+++     set base_stage $merge_stages(1)
+++
+++     # Build the command line
+++     switch -- $tool {
+++     kdiff3 {
+++             if {$base_stage ne {}} {
+++                     set cmdline [list "$merge_tool_path" --auto --L1 "$MERGED (Base)" \
+++                             --L2 "$MERGED (Local)" --L3 "$MERGED (Remote)" -o "$MERGED" "$BASE" "$LOCAL" "$REMOTE"]
+++             } else {
+++                     set cmdline [list "$merge_tool_path" --auto --L1 "$MERGED (Local)" \
+++                             --L2 "$MERGED (Remote)" -o "$MERGED" "$LOCAL" "$REMOTE"]
+++             }
+++     }
+++     tkdiff {
+++             if {$base_stage ne {}} {
+++                     set cmdline [list "$merge_tool_path" -a "$BASE" -o "$MERGED" "$LOCAL" "$REMOTE"]
+++             } else {
+++                     set cmdline [list "$merge_tool_path" -o "$MERGED" "$LOCAL" "$REMOTE"]
+++             }
+++     }
+++     meld {
+++             set cmdline [list "$merge_tool_path" "$LOCAL" "$MERGED" "$REMOTE"]
+++     }
+++     gvimdiff {
+++             set cmdline [list "$merge_tool_path" -f "$LOCAL" "$MERGED" "$REMOTE"]
+++     }
+++     xxdiff {
+++             if {$base_stage ne {}} {
+++                     set cmdline [list "$merge_tool_path" -X --show-merged-pane \
+++                                         -R {Accel.SaveAsMerged: "Ctrl-S"} \
+++                                         -R {Accel.Search: "Ctrl+F"} \
+++                                         -R {Accel.SearchForward: "Ctrl-G"} \
+++                                         --merged-file "$MERGED" "$LOCAL" "$BASE" "$REMOTE"]
+++             } else {
+++                     set cmdline [list "$merge_tool_path" -X --show-merged-pane \
+++                                         -R {Accel.SaveAsMerged: "Ctrl-S"} \
+++                                         -R {Accel.Search: "Ctrl+F"} \
+++                                         -R {Accel.SearchForward: "Ctrl-G"} \
+++                                         --merged-file "$MERGED" "$LOCAL" "$REMOTE"]
+++             }
+++     }
+++     opendiff {
+++             if {$base_stage ne {}} {
+++                     set cmdline [list "$merge_tool_path" "$LOCAL" "$REMOTE" -ancestor "$BASE" -merge "$MERGED"]
+++             } else {
+++                     set cmdline [list "$merge_tool_path" "$LOCAL" "$REMOTE" -merge "$MERGED"]
+++             }
+++     }
+++     ecmerge {
+++             if {$base_stage ne {}} {
+++                     set cmdline [list "$merge_tool_path" "$BASE" "$LOCAL" "$REMOTE" --default --mode=merge3 --to="$MERGED"]
+++             } else {
+++                     set cmdline [list "$merge_tool_path" "$LOCAL" "$REMOTE" --default --mode=merge2 --to="$MERGED"]
+++             }
+++     }
+++     emerge {
+++             if {$base_stage ne {}} {
+++                     set cmdline [list "$merge_tool_path" -f emerge-files-with-ancestor-command \
+++                                     "$LOCAL" "$REMOTE" "$BASE" "$basename"]
+++             } else {
+++                     set cmdline [list "$merge_tool_path" -f emerge-files-command \
+++                                     "$LOCAL" "$REMOTE" "$basename"]
+++             }
+++     }
+++     winmerge {
+++             if {$base_stage ne {}} {
+++                     # This tool does not support 3-way merges.
+++                     # Use the 'conflict file' resolution feature instead.
+++                     set cmdline [list "$merge_tool_path" -e -ub "$MERGED"]
+++             } else {
+++                     set cmdline [list "$merge_tool_path" -e -ub -wl \
+++                             -dl "Theirs File" -dr "Mine File" "$REMOTE" "$LOCAL" "$MERGED"]
+++             }
+++     }
+++     araxis {
+++             if {$base_stage ne {}} {
+++                     set cmdline [list "$merge_tool_path" -wait -merge -3 -a1 \
+++                             -title1:"'$MERGED (Base)'" -title2:"'$MERGED (Local)'" \
+++                             -title3:"'$MERGED (Remote)'" \
+++                             "$BASE" "$LOCAL" "$REMOTE" "$MERGED"]
+++             } else {
+++                     set cmdline [list "$merge_tool_path" -wait -2 \
+++                              -title1:"'$MERGED (Local)'" -title2:"'$MERGED (Remote)'" \
+++                              "$LOCAL" "$REMOTE" "$MERGED"]
+++             }
+++     }
+++     p4merge {
+++             set cmdline [list "$merge_tool_path" "$BASE" "$REMOTE" "$LOCAL" "$MERGED"]
+++     }
+++     vimdiff {
+++             error_popup [mc "Not a GUI merge tool: '%s'" $tool]
+++             return
+++     }
+++     default {
+++             error_popup [mc "Unsupported merge tool '%s'" $tool]
+++             return
+++     }
+++     }
+++
+++     merge_tool_start $cmdline $MERGED $BACKUP [list $BASE $LOCAL $REMOTE]
+++}
+++
+++proc delete_temp_files {files} {
+++     foreach fname $files {
+++             file delete $fname
+++     }
+++}
+++
+++proc merge_tool_get_stages {target stages} {
+++     global merge_stages
+++
+++     set i 1
+++     foreach fname $stages {
+++             if {$merge_stages($i) eq {}} {
+++                     file delete $fname
+++                     catch { close [open $fname w] }
+++             } else {
+++                     # A hack to support autocrlf properly
+++                     git checkout-index -f --stage=$i -- $target
+++                     file rename -force -- $target $fname
+++             }
+++             incr i
+++     }
+++}
+++
+++proc merge_tool_start {cmdline target backup stages} {
+++     global merge_stages mtool_target mtool_tmpfiles mtool_fd mtool_mtime
+++
+++     if {[info exists mtool_fd]} {
+++             if {[ask_popup [mc "Merge tool is already running, terminate it?"]] eq {yes}} {
+++                     catch { kill_file_process $mtool_fd }
+++                     catch { close $mtool_fd }
+++                     unset mtool_fd
+++
+++                     set old_backup [lindex $mtool_tmpfiles end]
+++                     file rename -force -- $old_backup $mtool_target
+++                     delete_temp_files $mtool_tmpfiles
+++             } else {
+++                     return
+++             }
+++     }
+++
+++     # Save the original file
+++     file rename -force -- $target $backup
+++
+++     # Get the blobs; it destroys $target
+++     if {[catch {merge_tool_get_stages $target $stages} err]} {
+++             file rename -force -- $backup $target
+++             delete_temp_files $stages
+++             error_popup [mc "Error retrieving versions:\n%s" $err]
+++             return
+++     }
+++
+++     # Restore the conflict file
+++     file copy -force -- $backup $target
+++
+++     # Initialize global state
+++     set mtool_target $target
+++     set mtool_mtime [file mtime $target]
+++     set mtool_tmpfiles $stages
+++
+++     lappend mtool_tmpfiles $backup
+++
+++     # Force redirection to avoid interpreting output on stderr
+++     # as an error, and launch the tool
+++     lappend cmdline {2>@1}
+++
+++     if {[catch { set mtool_fd [_open_stdout_stderr $cmdline] } err]} {
+++             delete_temp_files $mtool_tmpfiles
+++             error_popup [mc "Could not start the merge tool:\n\n%s" $err]
+++             return
+++     }
+++
+++     ui_status [mc "Running merge tool..."]
+++
+++     fconfigure $mtool_fd -blocking 0 -translation binary -encoding binary
+++     fileevent $mtool_fd readable [list read_mtool_output $mtool_fd]
+++}
+++
+++proc read_mtool_output {fd} {
+++     global mtool_fd mtool_tmpfiles
+++
+++     read $fd
+++     if {[eof $fd]} {
+++             unset mtool_fd
+++
+++             fconfigure $fd -blocking 1
+++             merge_tool_finish $fd
+++     }
+++}
+++
+++proc merge_tool_finish {fd} {
+++     global mtool_tmpfiles mtool_target mtool_mtime
+++
+++     set backup [lindex $mtool_tmpfiles end]
+++     set failed 0
+++
+++     # Check the return code
+++     if {[catch {close $fd} err]} {
+++             set failed 1
+++             if {$err ne {child process exited abnormally}} {
+++                     error_popup [strcat [mc "Merge tool failed."] "\n\n$err"]
+++             }
+++     }
+++
+++     # Check the modification time of the target file
+++     if {!$failed && [file mtime $mtool_target] eq $mtool_mtime} {
+++             if {[ask_popup [mc "File %s unchanged, still accept as resolved?" \
+++                             [short_path $mtool_target]]] ne {yes}} {
+++                     set failed 1
+++             }
+++     }
+++
+++     # Finish
+++     if {$failed} {
+++             file rename -force -- $backup $mtool_target
+++             delete_temp_files $mtool_tmpfiles
+++             ui_status [mc "Merge tool failed."]
+++     } else {
+++             if {[is_config_true merge.keepbackup]} {
+++                     file rename -force -- $backup "$mtool_target.orig"
+++             }
+++
+++             delete_temp_files $mtool_tmpfiles
+++
+++             merge_add_resolution $mtool_target
+++     }
+++}
index 5e1346e601faf90114e9c62f11144f812835e872,0000000000000000000000000000000000000000,0000000000000000000000000000000000000000..9b865f6a754aeb0933e1696ae1c9ffaee899e936
mode 100644,000000,000000..100644
--- /dev/null
--- /dev/null
@@@@ -1,281 -1,0 -1,0 +1,283 @@@@
 ++# git-gui options editor
 ++# Copyright (C) 2006, 2007 Shawn Pearce
 ++
 ++proc save_config {} {
 ++     global default_config font_descs
 ++     global repo_config global_config
 ++     global repo_config_new global_config_new
 ++     global ui_comm_spell
 ++
 ++     foreach option $font_descs {
 ++             set name [lindex $option 0]
 ++             set font [lindex $option 1]
 ++             font configure $font \
 ++                     -family $global_config_new(gui.$font^^family) \
 ++                     -size $global_config_new(gui.$font^^size)
 ++             font configure ${font}bold \
 ++                     -family $global_config_new(gui.$font^^family) \
 ++                     -size $global_config_new(gui.$font^^size)
 ++             font configure ${font}italic \
 ++                     -family $global_config_new(gui.$font^^family) \
 ++                     -size $global_config_new(gui.$font^^size)
 ++             set global_config_new(gui.$name) [font configure $font]
 ++             unset global_config_new(gui.$font^^family)
 ++             unset global_config_new(gui.$font^^size)
 ++     }
 ++
 ++     foreach name [array names default_config] {
 ++             set value $global_config_new($name)
 ++             if {$value ne $global_config($name)} {
 ++                     if {$value eq $default_config($name)} {
 ++                             catch {git config --global --unset $name}
 ++                     } else {
 ++                             regsub -all "\[{}\]" $value {"} value
 ++                             git config --global $name $value
 ++                     }
 ++                     set global_config($name) $value
 ++                     if {$value eq $repo_config($name)} {
 ++                             catch {git config --unset $name}
 ++                             set repo_config($name) $value
 ++                     }
 ++             }
 ++     }
 ++
 ++     foreach name [array names default_config] {
 ++             set value $repo_config_new($name)
 ++             if {$value ne $repo_config($name)} {
 ++                     if {$value eq $global_config($name)} {
 ++                             catch {git config --unset $name}
 ++                     } else {
 ++                             regsub -all "\[{}\]" $value {"} value
 ++                             git config $name $value
 ++                     }
 ++                     set repo_config($name) $value
 ++             }
 ++     }
 ++
 ++     if {[info exists repo_config(gui.spellingdictionary)]} {
 ++             set value $repo_config(gui.spellingdictionary)
 ++             if {$value eq {none}} {
 ++                     if {[info exists ui_comm_spell]} {
 ++                             $ui_comm_spell stop
 ++                     }
 ++             } elseif {[info exists ui_comm_spell]} {
 ++                     $ui_comm_spell lang $value
 ++             }
 ++     }
 ++}
 ++
 ++proc do_options {} {
 ++     global repo_config global_config font_descs
 ++     global repo_config_new global_config_new
 ++     global ui_comm_spell
 ++
 ++     array unset repo_config_new
 ++     array unset global_config_new
 ++     foreach name [array names repo_config] {
 ++             set repo_config_new($name) $repo_config($name)
 ++     }
 ++     load_config 1
 ++     foreach name [array names repo_config] {
 ++             switch -- $name {
 ++             gui.diffcontext {continue}
 ++             }
 ++             set repo_config_new($name) $repo_config($name)
 ++     }
 ++     foreach name [array names global_config] {
 ++             set global_config_new($name) $global_config($name)
 ++     }
 ++
 ++     set w .options_editor
 ++     toplevel $w
 ++     wm geometry $w "+[winfo rootx .]+[winfo rooty .]"
 ++
 ++     frame $w.buttons
 ++     button $w.buttons.restore -text [mc "Restore Defaults"] \
 ++             -default normal \
 ++             -command do_restore_defaults
 ++     pack $w.buttons.restore -side left
 ++     button $w.buttons.save -text [mc Save] \
 ++             -default active \
 ++             -command [list do_save_config $w]
 ++     pack $w.buttons.save -side right
 ++     button $w.buttons.cancel -text [mc "Cancel"] \
 ++             -default normal \
 ++             -command [list destroy $w]
 ++     pack $w.buttons.cancel -side right -padx 5
 ++     pack $w.buttons -side bottom -fill x -pady 10 -padx 10
 ++
 ++     labelframe $w.repo -text [mc "%s Repository" [reponame]]
 ++     labelframe $w.global -text [mc "Global (All Repositories)"]
 ++     pack $w.repo -side left -fill both -expand 1 -pady 5 -padx 5
 ++     pack $w.global -side right -fill both -expand 1 -pady 5 -padx 5
 ++
 ++     set optid 0
 ++     foreach option {
 ++             {t user.name {mc "User Name"}}
 ++             {t user.email {mc "Email Address"}}
 ++
 ++             {b merge.summary {mc "Summarize Merge Commits"}}
 ++             {i-1..5 merge.verbosity {mc "Merge Verbosity"}}
 ++             {b merge.diffstat {mc "Show Diffstat After Merge"}}
+++             {t merge.tool {mc "Use Merge Tool"}}
 ++
 ++             {b gui.trustmtime  {mc "Trust File Modification Timestamps"}}
 ++             {b gui.pruneduringfetch {mc "Prune Tracking Branches During Fetch"}}
 ++             {b gui.matchtrackingbranch {mc "Match Tracking Branches"}}
 ++             {b gui.fastcopyblame {mc "Blame Copy Only On Changed Files"}}
 ++             {i-20..200 gui.copyblamethreshold {mc "Minimum Letters To Blame Copy On"}}
+++             {i-0..300 gui.blamehistoryctx {mc "Blame History Context Radius (days)"}}
 ++             {i-1..99 gui.diffcontext {mc "Number of Diff Context Lines"}}
 ++             {i-0..99 gui.commitmsgwidth {mc "Commit Message Text Width"}}
 ++             {t gui.newbranchtemplate {mc "New Branch Name Template"}}
 ++             } {
 ++             set type [lindex $option 0]
 ++             set name [lindex $option 1]
 ++             set text [eval [lindex $option 2]]
 ++             incr optid
 ++             foreach f {repo global} {
 ++                     switch -glob -- $type {
 ++                     b {
 ++                             checkbutton $w.$f.$optid -text $text \
 ++                                     -variable ${f}_config_new($name) \
 ++                                     -onvalue true \
 ++                                     -offvalue false
 ++                             pack $w.$f.$optid -side top -anchor w
 ++                     }
 ++                     i-* {
 ++                             regexp -- {-(\d+)\.\.(\d+)$} $type _junk min max
 ++                             frame $w.$f.$optid
 ++                             label $w.$f.$optid.l -text "$text:"
 ++                             pack $w.$f.$optid.l -side left -anchor w -fill x
 ++                             spinbox $w.$f.$optid.v \
 ++                                     -textvariable ${f}_config_new($name) \
 ++                                     -from $min \
 ++                                     -to $max \
 ++                                     -increment 1 \
 ++                                     -width [expr {1 + [string length $max]}]
 ++                             bind $w.$f.$optid.v <FocusIn> {%W selection range 0 end}
 ++                             pack $w.$f.$optid.v -side right -anchor e -padx 5
 ++                             pack $w.$f.$optid -side top -anchor w -fill x
 ++                     }
 ++                     t {
 ++                             frame $w.$f.$optid
 ++                             label $w.$f.$optid.l -text "$text:"
 ++                             entry $w.$f.$optid.v \
 ++                                     -borderwidth 1 \
 ++                                     -relief sunken \
 ++                                     -width 20 \
 ++                                     -textvariable ${f}_config_new($name)
 ++                             pack $w.$f.$optid.l -side left -anchor w
 ++                             pack $w.$f.$optid.v -side left -anchor w \
 ++                                     -fill x -expand 1 \
 ++                                     -padx 5
 ++                             pack $w.$f.$optid -side top -anchor w -fill x
 ++                     }
 ++                     }
 ++             }
 ++     }
 ++
 ++     set all_dicts [linsert \
 ++             [spellcheck::available_langs] \
 ++             0 \
 ++             none]
 ++     incr optid
 ++     foreach f {repo global} {
 ++             if {![info exists ${f}_config_new(gui.spellingdictionary)]} {
 ++                     if {[info exists ui_comm_spell]} {
 ++                             set value [$ui_comm_spell lang]
 ++                     } else {
 ++                             set value none
 ++                     }
 ++                     set ${f}_config_new(gui.spellingdictionary) $value
 ++             }
 ++
 ++             frame $w.$f.$optid
 ++             label $w.$f.$optid.l -text [mc "Spelling Dictionary:"]
 ++             eval tk_optionMenu $w.$f.$optid.v \
 ++                     ${f}_config_new(gui.spellingdictionary) \
 ++                     $all_dicts
 ++             pack $w.$f.$optid.l -side left -anchor w -fill x
 ++             pack $w.$f.$optid.v -side right -anchor e -padx 5
 ++             pack $w.$f.$optid -side top -anchor w -fill x
 ++     }
 ++     unset all_dicts
 ++
 ++     set all_fonts [lsort [font families]]
 ++     foreach option $font_descs {
 ++             set name [lindex $option 0]
 ++             set font [lindex $option 1]
 ++             set text [eval [lindex $option 2]]
 ++
 ++             set global_config_new(gui.$font^^family) \
 ++                     [font configure $font -family]
 ++             set global_config_new(gui.$font^^size) \
 ++                     [font configure $font -size]
 ++
 ++             frame $w.global.$name
 ++             label $w.global.$name.l -text "$text:"
 ++             button $w.global.$name.b \
 ++                     -text [mc "Change Font"] \
 ++                     -command [list \
 ++                             choose_font::pick \
 ++                             $w \
 ++                             [mc "Choose %s" $text] \
 ++                             global_config_new(gui.$font^^family) \
 ++                             global_config_new(gui.$font^^size) \
 ++                             ]
 ++             label $w.global.$name.f -textvariable global_config_new(gui.$font^^family)
 ++             label $w.global.$name.s -textvariable global_config_new(gui.$font^^size)
 ++             label $w.global.$name.pt -text [mc "pt."]
 ++             pack $w.global.$name.l -side left -anchor w
 ++             pack $w.global.$name.b -side right -anchor e
 ++             pack $w.global.$name.pt -side right -anchor w
 ++             pack $w.global.$name.s -side right -anchor w
 ++             pack $w.global.$name.f -side right -anchor w
 ++             pack $w.global.$name -side top -anchor w -fill x
 ++     }
 ++
 ++     bind $w <Visibility> "grab $w; focus $w.buttons.save"
 ++     bind $w <Key-Escape> "destroy $w"
 ++     bind $w <Key-Return> [list do_save_config $w]
 ++
 ++     if {[is_MacOSX]} {
 ++             set t [mc "Preferences"]
 ++     } else {
 ++             set t [mc "Options"]
 ++     }
 ++     wm title $w "[appname] ([reponame]): $t"
 ++     tkwait window $w
 ++}
 ++
 ++proc do_restore_defaults {} {
 ++     global font_descs default_config repo_config
 ++     global repo_config_new global_config_new
 ++
 ++     foreach name [array names default_config] {
 ++             set repo_config_new($name) $default_config($name)
 ++             set global_config_new($name) $default_config($name)
 ++     }
 ++
 ++     foreach option $font_descs {
 ++             set name [lindex $option 0]
 ++             set repo_config(gui.$name) $default_config(gui.$name)
 ++     }
 ++     apply_config
 ++
 ++     foreach option $font_descs {
 ++             set name [lindex $option 0]
 ++             set font [lindex $option 1]
 ++             set global_config_new(gui.$font^^family) \
 ++                     [font configure $font -family]
 ++             set global_config_new(gui.$font^^size) \
 ++                     [font configure $font -size]
 ++     }
 ++}
 ++
 ++proc do_save_config {w} {
 ++     if {[catch {save_config} err]} {
 ++             error_popup [strcat [mc "Failed to completely save options:"] "\n\n$err"]
 ++     }
 ++     reshow_diff
 ++     destroy $w
 ++}
diff --cc gitk-git/gitk
index 087c4ac733be4b788751d0bae5b7aad22ce0dd99,0000000000000000000000000000000000000000,0000000000000000000000000000000000000000..2eaa2ae7d6f692f6063ebbd211eaab30212c2eae
mode 100644,000000,000000..100644
--- /dev/null
--- /dev/null
@@@@ -1,9996 -1,0 -1,0 +1,10038 @@@@
-      global pending_select mainheadid
 ++#!/bin/sh
 ++# Tcl ignores the next line -*- tcl -*- \
 ++exec wish "$0" -- "$@"
 ++
 ++# Copyright © 2005-2008 Paul Mackerras.  All rights reserved.
 ++# This program is free software; it may be used, copied, modified
 ++# and distributed under the terms of the GNU General Public Licence,
 ++# either version 2, or (at your option) any later version.
 ++
 ++proc gitdir {} {
 ++    global env
 ++    if {[info exists env(GIT_DIR)]} {
 ++     return $env(GIT_DIR)
 ++    } else {
 ++     return [exec git rev-parse --git-dir]
 ++    }
 ++}
 ++
 ++# A simple scheduler for compute-intensive stuff.
 ++# The aim is to make sure that event handlers for GUI actions can
 ++# run at least every 50-100 ms.  Unfortunately fileevent handlers are
 ++# run before X event handlers, so reading from a fast source can
 ++# make the GUI completely unresponsive.
 ++proc run args {
 ++    global isonrunq runq currunq
 ++
 ++    set script $args
 ++    if {[info exists isonrunq($script)]} return
 ++    if {$runq eq {} && ![info exists currunq]} {
 ++     after idle dorunq
 ++    }
 ++    lappend runq [list {} $script]
 ++    set isonrunq($script) 1
 ++}
 ++
 ++proc filerun {fd script} {
 ++    fileevent $fd readable [list filereadable $fd $script]
 ++}
 ++
 ++proc filereadable {fd script} {
 ++    global runq currunq
 ++
 ++    fileevent $fd readable {}
 ++    if {$runq eq {} && ![info exists currunq]} {
 ++     after idle dorunq
 ++    }
 ++    lappend runq [list $fd $script]
 ++}
 ++
 ++proc nukefile {fd} {
 ++    global runq
 ++
 ++    for {set i 0} {$i < [llength $runq]} {} {
 ++     if {[lindex $runq $i 0] eq $fd} {
 ++         set runq [lreplace $runq $i $i]
 ++     } else {
 ++         incr i
 ++     }
 ++    }
 ++}
 ++
 ++proc dorunq {} {
 ++    global isonrunq runq currunq
 ++
 ++    set tstart [clock clicks -milliseconds]
 ++    set t0 $tstart
 ++    while {[llength $runq] > 0} {
 ++     set fd [lindex $runq 0 0]
 ++     set script [lindex $runq 0 1]
 ++     set currunq [lindex $runq 0]
 ++     set runq [lrange $runq 1 end]
 ++     set repeat [eval $script]
 ++     unset currunq
 ++     set t1 [clock clicks -milliseconds]
 ++     set t [expr {$t1 - $t0}]
 ++     if {$repeat ne {} && $repeat} {
 ++         if {$fd eq {} || $repeat == 2} {
 ++             # script returns 1 if it wants to be readded
 ++             # file readers return 2 if they could do more straight away
 ++             lappend runq [list $fd $script]
 ++         } else {
 ++             fileevent $fd readable [list filereadable $fd $script]
 ++         }
 ++     } elseif {$fd eq {}} {
 ++         unset isonrunq($script)
 ++     }
 ++     set t0 $t1
 ++     if {$t1 - $tstart >= 80} break
 ++    }
 ++    if {$runq ne {}} {
 ++     after idle dorunq
 ++    }
 ++}
 ++
 ++proc reg_instance {fd} {
 ++    global commfd leftover loginstance
 ++
 ++    set i [incr loginstance]
 ++    set commfd($i) $fd
 ++    set leftover($i) {}
 ++    return $i
 ++}
 ++
 ++proc unmerged_files {files} {
 ++    global nr_unmerged
 ++
 ++    # find the list of unmerged files
 ++    set mlist {}
 ++    set nr_unmerged 0
 ++    if {[catch {
 ++     set fd [open "| git ls-files -u" r]
 ++    } err]} {
 ++     show_error {} . "[mc "Couldn't get list of unmerged files:"] $err"
 ++     exit 1
 ++    }
 ++    while {[gets $fd line] >= 0} {
 ++     set i [string first "\t" $line]
 ++     if {$i < 0} continue
 ++     set fname [string range $line [expr {$i+1}] end]
 ++     if {[lsearch -exact $mlist $fname] >= 0} continue
 ++     incr nr_unmerged
 ++     if {$files eq {} || [path_filter $files $fname]} {
 ++         lappend mlist $fname
 ++     }
 ++    }
 ++    catch {close $fd}
 ++    return $mlist
 ++}
 ++
 ++proc parseviewargs {n arglist} {
 ++    global vdatemode vmergeonly vflags vdflags vrevs vfiltered vorigargs
 ++
 ++    set vdatemode($n) 0
 ++    set vmergeonly($n) 0
 ++    set glflags {}
 ++    set diffargs {}
 ++    set nextisval 0
 ++    set revargs {}
 ++    set origargs $arglist
 ++    set allknown 1
 ++    set filtered 0
 ++    set i -1
 ++    foreach arg $arglist {
 ++     incr i
 ++     if {$nextisval} {
 ++         lappend glflags $arg
 ++         set nextisval 0
 ++         continue
 ++     }
 ++     switch -glob -- $arg {
 ++         "-d" -
 ++         "--date-order" {
 ++             set vdatemode($n) 1
 ++             # remove from origargs in case we hit an unknown option
 ++             set origargs [lreplace $origargs $i $i]
 ++             incr i -1
 ++         }
 ++         # These request or affect diff output, which we don't want.
 ++         # Some could be used to set our defaults for diff display.
 ++         "-[puabwcrRBMC]" -
 ++         "--no-renames" - "--full-index" - "--binary" - "--abbrev=*" -
 ++         "--find-copies-harder" - "-l*" - "--ext-diff" - "--no-ext-diff" -
 ++         "--src-prefix=*" - "--dst-prefix=*" - "--no-prefix" -
 ++         "-O*" - "--text" - "--full-diff" - "--ignore-space-at-eol" -
 ++         "--ignore-space-change" - "-U*" - "--unified=*" {
 ++             lappend diffargs $arg
 ++         }
 ++         # These cause our parsing of git log's output to fail, or else
 ++         # they're options we want to set ourselves, so ignore them.
 ++         "--raw" - "--patch-with-raw" - "--patch-with-stat" -
 ++         "--name-only" - "--name-status" - "--color" - "--color-words" -
 ++         "--log-size" - "--pretty=*" - "--decorate" - "--abbrev-commit" -
 ++         "--cc" - "-z" - "--header" - "--parents" - "--boundary" -
 ++         "--no-color" - "-g" - "--walk-reflogs" - "--no-walk" -
 ++         "--timestamp" - "relative-date" - "--date=*" - "--stdin" -
 ++         "--objects" - "--objects-edge" - "--reverse" {
 ++         }
 ++         # These are harmless, and some are even useful
 ++         "--stat=*" - "--numstat" - "--shortstat" - "--summary" -
 ++         "--check" - "--exit-code" - "--quiet" - "--topo-order" -
 ++         "--full-history" - "--dense" - "--sparse" -
 ++         "--follow" - "--left-right" - "--encoding=*" {
 ++             lappend glflags $arg
 ++         }
 ++         # These mean that we get a subset of the commits
 ++         "--diff-filter=*" - "--no-merges" - "--unpacked" -
 ++         "--max-count=*" - "--skip=*" - "--since=*" - "--after=*" -
 ++         "--until=*" - "--before=*" - "--max-age=*" - "--min-age=*" -
 ++         "--author=*" - "--committer=*" - "--grep=*" - "-[iE]" -
 ++         "--remove-empty" - "--first-parent" - "--cherry-pick" -
 ++         "-S*" - "--pickaxe-all" - "--pickaxe-regex" - {
 ++             set filtered 1
 ++             lappend glflags $arg
 ++         }
 ++         # This appears to be the only one that has a value as a
 ++         # separate word following it
 ++         "-n" {
 ++             set filtered 1
 ++             set nextisval 1
 ++             lappend glflags $arg
 ++         }
 ++         "--not" {
 ++             set notflag [expr {!$notflag}]
 ++             lappend revargs $arg
 ++         }
 ++         "--all" {
 ++             lappend revargs $arg
 ++         }
 ++         "--merge" {
 ++             set vmergeonly($n) 1
 ++             # git rev-parse doesn't understand --merge
 ++             lappend revargs --gitk-symmetric-diff-marker MERGE_HEAD...HEAD
 ++         }
 ++         # Other flag arguments including -<n>
 ++         "-*" {
 ++             if {[string is digit -strict [string range $arg 1 end]]} {
 ++                 set filtered 1
 ++             } else {
 ++                 # a flag argument that we don't recognize;
 ++                 # that means we can't optimize
 ++                 set allknown 0
 ++             }
 ++             lappend glflags $arg
 ++         }
 ++         # Non-flag arguments specify commits or ranges of commits
 ++         default {
 ++             if {[string match "*...*" $arg]} {
 ++                 lappend revargs --gitk-symmetric-diff-marker
 ++             }
 ++             lappend revargs $arg
 ++         }
 ++     }
 ++    }
 ++    set vdflags($n) $diffargs
 ++    set vflags($n) $glflags
 ++    set vrevs($n) $revargs
 ++    set vfiltered($n) $filtered
 ++    set vorigargs($n) $origargs
 ++    return $allknown
 ++}
 ++
 ++proc parseviewrevs {view revs} {
 ++    global vposids vnegids
 ++
 ++    if {$revs eq {}} {
 ++     set revs HEAD
 ++    }
 ++    if {[catch {set ids [eval exec git rev-parse $revs]} err]} {
 ++     # we get stdout followed by stderr in $err
 ++     # for an unknown rev, git rev-parse echoes it and then errors out
 ++     set errlines [split $err "\n"]
 ++     set badrev {}
 ++     for {set l 0} {$l < [llength $errlines]} {incr l} {
 ++         set line [lindex $errlines $l]
 ++         if {!([string length $line] == 40 && [string is xdigit $line])} {
 ++             if {[string match "fatal:*" $line]} {
 ++                 if {[string match "fatal: ambiguous argument*" $line]
 ++                     && $badrev ne {}} {
 ++                     if {[llength $badrev] == 1} {
 ++                         set err "unknown revision $badrev"
 ++                     } else {
 ++                         set err "unknown revisions: [join $badrev ", "]"
 ++                     }
 ++                 } else {
 ++                     set err [join [lrange $errlines $l end] "\n"]
 ++                 }
 ++                 break
 ++             }
 ++             lappend badrev $line
 ++         }
 ++     }                   
 ++     error_popup "Error parsing revisions: $err"
 ++     return {}
 ++    }
 ++    set ret {}
 ++    set pos {}
 ++    set neg {}
 ++    set sdm 0
 ++    foreach id [split $ids "\n"] {
 ++     if {$id eq "--gitk-symmetric-diff-marker"} {
 ++         set sdm 4
 ++     } elseif {[string match "^*" $id]} {
 ++         if {$sdm != 1} {
 ++             lappend ret $id
 ++             if {$sdm == 3} {
 ++                 set sdm 0
 ++             }
 ++         }
 ++         lappend neg [string range $id 1 end]
 ++     } else {
 ++         if {$sdm != 2} {
 ++             lappend ret $id
 ++         } else {
 ++             lset ret end [lindex $ret end]...$id
 ++         }
 ++         lappend pos $id
 ++     }
 ++     incr sdm -1
 ++    }
 ++    set vposids($view) $pos
 ++    set vnegids($view) $neg
 ++    return $ret
 ++}
 ++
 ++# Start off a git log process and arrange to read its output
 ++proc start_rev_list {view} {
 ++    global startmsecs commitidx viewcomplete curview
 ++    global tclencoding
 ++    global viewargs viewargscmd viewfiles vfilelimit
 ++    global showlocalchanges commitinterest
 ++    global viewactive viewinstances vmergeonly
 ++    global mainheadid
 ++    global vcanopt vflags vrevs vorigargs
 ++
 ++    set startmsecs [clock clicks -milliseconds]
 ++    set commitidx($view) 0
 ++    # these are set this way for the error exits
 ++    set viewcomplete($view) 1
 ++    set viewactive($view) 0
 ++    varcinit $view
 ++
 ++    set args $viewargs($view)
 ++    if {$viewargscmd($view) ne {}} {
 ++     if {[catch {
 ++         set str [exec sh -c $viewargscmd($view)]
 ++     } err]} {
 ++         error_popup "Error executing --argscmd command: $err"
 ++         return 0
 ++     }
 ++     set args [concat $args [split $str "\n"]]
 ++    }
 ++    set vcanopt($view) [parseviewargs $view $args]
 ++
 ++    set files $viewfiles($view)
 ++    if {$vmergeonly($view)} {
 ++     set files [unmerged_files $files]
 ++     if {$files eq {}} {
 ++         global nr_unmerged
 ++         if {$nr_unmerged == 0} {
 ++             error_popup [mc "No files selected: --merge specified but\
 ++                          no files are unmerged."]
 ++         } else {
 ++             error_popup [mc "No files selected: --merge specified but\
 ++                          no unmerged files are within file limit."]
 ++         }
 ++         return 0
 ++     }
 ++    }
 ++    set vfilelimit($view) $files
 ++
 ++    if {$vcanopt($view)} {
 ++     set revs [parseviewrevs $view $vrevs($view)]
 ++     if {$revs eq {}} {
 ++         return 0
 ++     }
 ++     set args [concat $vflags($view) $revs]
 ++    } else {
 ++     set args $vorigargs($view)
 ++    }
 ++
 ++    if {[catch {
 ++     set fd [open [concat | git log --no-color -z --pretty=raw --parents \
 ++                      --boundary $args "--" $files] r]
 ++    } err]} {
 ++     error_popup "[mc "Error executing git log:"] $err"
 ++     return 0
 ++    }
 ++    set i [reg_instance $fd]
 ++    set viewinstances($view) [list $i]
 ++    if {$showlocalchanges && $mainheadid ne {}} {
 ++     lappend commitinterest($mainheadid) {dodiffindex}
 ++    }
 ++    fconfigure $fd -blocking 0 -translation lf -eofchar {}
 ++    if {$tclencoding != {}} {
 ++     fconfigure $fd -encoding $tclencoding
 ++    }
 ++    filerun $fd [list getcommitlines $fd $i $view 0]
 ++    nowbusy $view [mc "Reading"]
 ++    set viewcomplete($view) 0
 ++    set viewactive($view) 1
 ++    return 1
 ++}
 ++
 ++proc stop_instance {inst} {
 ++    global commfd leftover
 ++
 ++    set fd $commfd($inst)
 ++    catch {
 ++     set pid [pid $fd]
 ++
 ++     if {$::tcl_platform(platform) eq {windows}} {
 ++         exec kill -f $pid
 ++     } else {
 ++         exec kill $pid
 ++     }
 ++    }
 ++    catch {close $fd}
 ++    nukefile $fd
 ++    unset commfd($inst)
 ++    unset leftover($inst)
 ++}
 ++
 ++proc stop_backends {} {
 ++    global commfd
 ++
 ++    foreach inst [array names commfd] {
 ++     stop_instance $inst
 ++    }
 ++}
 ++
 ++proc stop_rev_list {view} {
 ++    global viewinstances
 ++
 ++    foreach inst $viewinstances($view) {
 ++     stop_instance $inst
 ++    }
 ++    set viewinstances($view) {}
 ++}
 ++
 ++proc reset_pending_select {selid} {
+++    global pending_select mainheadid selectheadid
 ++
 ++    if {$selid ne {}} {
 ++     set pending_select $selid
+++    } elseif {$selectheadid ne {}} {
+++     set pending_select $selectheadid
 ++    } else {
 ++     set pending_select $mainheadid
 ++    }
 ++}
 ++
 ++proc getcommits {selid} {
 ++    global canv curview need_redisplay viewactive
 ++
 ++    initlayout
 ++    if {[start_rev_list $curview]} {
 ++     reset_pending_select $selid
 ++     show_status [mc "Reading commits..."]
 ++     set need_redisplay 1
 ++    } else {
 ++     show_status [mc "No commits selected"]
 ++    }
 ++}
 ++
 ++proc updatecommits {} {
 ++    global curview vcanopt vorigargs vfilelimit viewinstances
 ++    global viewactive viewcomplete tclencoding
 ++    global startmsecs showneartags showlocalchanges
 ++    global mainheadid pending_select
 ++    global isworktree
 ++    global varcid vposids vnegids vflags vrevs
 ++
 ++    set isworktree [expr {[exec git rev-parse --is-inside-work-tree] == "true"}]
 ++    set oldmainid $mainheadid
 ++    rereadrefs
 ++    if {$showlocalchanges} {
 ++     if {$mainheadid ne $oldmainid} {
 ++         dohidelocalchanges
 ++     }
 ++     if {[commitinview $mainheadid $curview]} {
 ++         dodiffindex
 ++     }
 ++    }
 ++    set view $curview
 ++    if {$vcanopt($view)} {
 ++     set oldpos $vposids($view)
 ++     set oldneg $vnegids($view)
 ++     set revs [parseviewrevs $view $vrevs($view)]
 ++     if {$revs eq {}} {
 ++         return
 ++     }
 ++     # note: getting the delta when negative refs change is hard,
 ++     # and could require multiple git log invocations, so in that
 ++     # case we ask git log for all the commits (not just the delta)
 ++     if {$oldneg eq $vnegids($view)} {
 ++         set newrevs {}
 ++         set npos 0
 ++         # take out positive refs that we asked for before or
 ++         # that we have already seen
 ++         foreach rev $revs {
 ++             if {[string length $rev] == 40} {
 ++                 if {[lsearch -exact $oldpos $rev] < 0
 ++                     && ![info exists varcid($view,$rev)]} {
 ++                     lappend newrevs $rev
 ++                     incr npos
 ++                 }
 ++             } else {
 ++                 lappend $newrevs $rev
 ++             }
 ++         }
 ++         if {$npos == 0} return
 ++         set revs $newrevs
 ++         set vposids($view) [lsort -unique [concat $oldpos $vposids($view)]]
 ++     }
 ++     set args [concat $vflags($view) $revs --not $oldpos]
 ++    } else {
 ++     set args $vorigargs($view)
 ++    }
 ++    if {[catch {
 ++     set fd [open [concat | git log --no-color -z --pretty=raw --parents \
 ++                       --boundary $args "--" $vfilelimit($view)] r]
 ++    } err]} {
 ++     error_popup "Error executing git log: $err"
 ++     return
 ++    }
 ++    if {$viewactive($view) == 0} {
 ++     set startmsecs [clock clicks -milliseconds]
 ++    }
 ++    set i [reg_instance $fd]
 ++    lappend viewinstances($view) $i
 ++    fconfigure $fd -blocking 0 -translation lf -eofchar {}
 ++    if {$tclencoding != {}} {
 ++     fconfigure $fd -encoding $tclencoding
 ++    }
 ++    filerun $fd [list getcommitlines $fd $i $view 1]
 ++    incr viewactive($view)
 ++    set viewcomplete($view) 0
 ++    reset_pending_select {}
 ++    nowbusy $view "Reading"
 ++    if {$showneartags} {
 ++     getallcommits
 ++    }
 ++}
 ++
 ++proc reloadcommits {} {
 ++    global curview viewcomplete selectedline currentid thickerline
 ++    global showneartags treediffs commitinterest cached_commitrow
 ++    global targetid
 ++
 ++    set selid {}
 ++    if {$selectedline ne {}} {
 ++     set selid $currentid
 ++    }
 ++
 ++    if {!$viewcomplete($curview)} {
 ++     stop_rev_list $curview
 ++    }
 ++    resetvarcs $curview
 ++    set selectedline {}
 ++    catch {unset currentid}
 ++    catch {unset thickerline}
 ++    catch {unset treediffs}
 ++    readrefs
 ++    changedrefs
 ++    if {$showneartags} {
 ++     getallcommits
 ++    }
 ++    clear_display
 ++    catch {unset commitinterest}
 ++    catch {unset cached_commitrow}
 ++    catch {unset targetid}
 ++    setcanvscroll
 ++    getcommits $selid
 ++    return 0
 ++}
 ++
 ++# This makes a string representation of a positive integer which
 ++# sorts as a string in numerical order
 ++proc strrep {n} {
 ++    if {$n < 16} {
 ++     return [format "%x" $n]
 ++    } elseif {$n < 256} {
 ++     return [format "x%.2x" $n]
 ++    } elseif {$n < 65536} {
 ++     return [format "y%.4x" $n]
 ++    }
 ++    return [format "z%.8x" $n]
 ++}
 ++
 ++# Procedures used in reordering commits from git log (without
 ++# --topo-order) into the order for display.
 ++
 ++proc varcinit {view} {
 ++    global varcstart vupptr vdownptr vleftptr vbackptr varctok varcrow
 ++    global vtokmod varcmod vrowmod varcix vlastins
 ++
 ++    set varcstart($view) {{}}
 ++    set vupptr($view) {0}
 ++    set vdownptr($view) {0}
 ++    set vleftptr($view) {0}
 ++    set vbackptr($view) {0}
 ++    set varctok($view) {{}}
 ++    set varcrow($view) {{}}
 ++    set vtokmod($view) {}
 ++    set varcmod($view) 0
 ++    set vrowmod($view) 0
 ++    set varcix($view) {{}}
 ++    set vlastins($view) {0}
 ++}
 ++
 ++proc resetvarcs {view} {
 ++    global varcid varccommits parents children vseedcount ordertok
 ++
 ++    foreach vid [array names varcid $view,*] {
 ++     unset varcid($vid)
 ++     unset children($vid)
 ++     unset parents($vid)
 ++    }
 ++    # some commits might have children but haven't been seen yet
 ++    foreach vid [array names children $view,*] {
 ++     unset children($vid)
 ++    }
 ++    foreach va [array names varccommits $view,*] {
 ++     unset varccommits($va)
 ++    }
 ++    foreach vd [array names vseedcount $view,*] {
 ++     unset vseedcount($vd)
 ++    }
 ++    catch {unset ordertok}
 ++}
 ++
 ++# returns a list of the commits with no children
 ++proc seeds {v} {
 ++    global vdownptr vleftptr varcstart
 ++
 ++    set ret {}
 ++    set a [lindex $vdownptr($v) 0]
 ++    while {$a != 0} {
 ++     lappend ret [lindex $varcstart($v) $a]
 ++     set a [lindex $vleftptr($v) $a]
 ++    }
 ++    return $ret
 ++}
 ++
 ++proc newvarc {view id} {
 ++    global varcid varctok parents children vdatemode
 ++    global vupptr vdownptr vleftptr vbackptr varcrow varcix varcstart
 ++    global commitdata commitinfo vseedcount varccommits vlastins
 ++
 ++    set a [llength $varctok($view)]
 ++    set vid $view,$id
 ++    if {[llength $children($vid)] == 0 || $vdatemode($view)} {
 ++     if {![info exists commitinfo($id)]} {
 ++         parsecommit $id $commitdata($id) 1
 ++     }
 ++     set cdate [lindex $commitinfo($id) 4]
 ++     if {![string is integer -strict $cdate]} {
 ++         set cdate 0
 ++     }
 ++     if {![info exists vseedcount($view,$cdate)]} {
 ++         set vseedcount($view,$cdate) -1
 ++     }
 ++     set c [incr vseedcount($view,$cdate)]
 ++     set cdate [expr {$cdate ^ 0xffffffff}]
 ++     set tok "s[strrep $cdate][strrep $c]"
 ++    } else {
 ++     set tok {}
 ++    }
 ++    set ka 0
 ++    if {[llength $children($vid)] > 0} {
 ++     set kid [lindex $children($vid) end]
 ++     set k $varcid($view,$kid)
 ++     if {[string compare [lindex $varctok($view) $k] $tok] > 0} {
 ++         set ki $kid
 ++         set ka $k
 ++         set tok [lindex $varctok($view) $k]
 ++     }
 ++    }
 ++    if {$ka != 0} {
 ++     set i [lsearch -exact $parents($view,$ki) $id]
 ++     set j [expr {[llength $parents($view,$ki)] - 1 - $i}]
 ++     append tok [strrep $j]
 ++    }
 ++    set c [lindex $vlastins($view) $ka]
 ++    if {$c == 0 || [string compare $tok [lindex $varctok($view) $c]] < 0} {
 ++     set c $ka
 ++     set b [lindex $vdownptr($view) $ka]
 ++    } else {
 ++     set b [lindex $vleftptr($view) $c]
 ++    }
 ++    while {$b != 0 && [string compare $tok [lindex $varctok($view) $b]] >= 0} {
 ++     set c $b
 ++     set b [lindex $vleftptr($view) $c]
 ++    }
 ++    if {$c == $ka} {
 ++     lset vdownptr($view) $ka $a
 ++     lappend vbackptr($view) 0
 ++    } else {
 ++     lset vleftptr($view) $c $a
 ++     lappend vbackptr($view) $c
 ++    }
 ++    lset vlastins($view) $ka $a
 ++    lappend vupptr($view) $ka
 ++    lappend vleftptr($view) $b
 ++    if {$b != 0} {
 ++     lset vbackptr($view) $b $a
 ++    }
 ++    lappend varctok($view) $tok
 ++    lappend varcstart($view) $id
 ++    lappend vdownptr($view) 0
 ++    lappend varcrow($view) {}
 ++    lappend varcix($view) {}
 ++    set varccommits($view,$a) {}
 ++    lappend vlastins($view) 0
 ++    return $a
 ++}
 ++
 ++proc splitvarc {p v} {
 ++    global varcid varcstart varccommits varctok
 ++    global vupptr vdownptr vleftptr vbackptr varcix varcrow vlastins
 ++
 ++    set oa $varcid($v,$p)
 ++    set ac $varccommits($v,$oa)
 ++    set i [lsearch -exact $varccommits($v,$oa) $p]
 ++    if {$i <= 0} return
 ++    set na [llength $varctok($v)]
 ++    # "%" sorts before "0"...
 ++    set tok "[lindex $varctok($v) $oa]%[strrep $i]"
 ++    lappend varctok($v) $tok
 ++    lappend varcrow($v) {}
 ++    lappend varcix($v) {}
 ++    set varccommits($v,$oa) [lrange $ac 0 [expr {$i - 1}]]
 ++    set varccommits($v,$na) [lrange $ac $i end]
 ++    lappend varcstart($v) $p
 ++    foreach id $varccommits($v,$na) {
 ++     set varcid($v,$id) $na
 ++    }
 ++    lappend vdownptr($v) [lindex $vdownptr($v) $oa]
 ++    lappend vlastins($v) [lindex $vlastins($v) $oa]
 ++    lset vdownptr($v) $oa $na
 ++    lset vlastins($v) $oa 0
 ++    lappend vupptr($v) $oa
 ++    lappend vleftptr($v) 0
 ++    lappend vbackptr($v) 0
 ++    for {set b [lindex $vdownptr($v) $na]} {$b != 0} {set b [lindex $vleftptr($v) $b]} {
 ++     lset vupptr($v) $b $na
 ++    }
 ++}
 ++
 ++proc renumbervarc {a v} {
 ++    global parents children varctok varcstart varccommits
 ++    global vupptr vdownptr vleftptr vbackptr vlastins varcid vtokmod vdatemode
 ++
 ++    set t1 [clock clicks -milliseconds]
 ++    set todo {}
 ++    set isrelated($a) 1
 ++    set kidchanged($a) 1
 ++    set ntot 0
 ++    while {$a != 0} {
 ++     if {[info exists isrelated($a)]} {
 ++         lappend todo $a
 ++         set id [lindex $varccommits($v,$a) end]
 ++         foreach p $parents($v,$id) {
 ++             if {[info exists varcid($v,$p)]} {
 ++                 set isrelated($varcid($v,$p)) 1
 ++             }
 ++         }
 ++     }
 ++     incr ntot
 ++     set b [lindex $vdownptr($v) $a]
 ++     if {$b == 0} {
 ++         while {$a != 0} {
 ++             set b [lindex $vleftptr($v) $a]
 ++             if {$b != 0} break
 ++             set a [lindex $vupptr($v) $a]
 ++         }
 ++     }
 ++     set a $b
 ++    }
 ++    foreach a $todo {
 ++     if {![info exists kidchanged($a)]} continue
 ++     set id [lindex $varcstart($v) $a]
 ++     if {[llength $children($v,$id)] > 1} {
 ++         set children($v,$id) [lsort -command [list vtokcmp $v] \
 ++                                   $children($v,$id)]
 ++     }
 ++     set oldtok [lindex $varctok($v) $a]
 ++     if {!$vdatemode($v)} {
 ++         set tok {}
 ++     } else {
 ++         set tok $oldtok
 ++     }
 ++     set ka 0
 ++     set kid [last_real_child $v,$id]
 ++     if {$kid ne {}} {
 ++         set k $varcid($v,$kid)
 ++         if {[string compare [lindex $varctok($v) $k] $tok] > 0} {
 ++             set ki $kid
 ++             set ka $k
 ++             set tok [lindex $varctok($v) $k]
 ++         }
 ++     }
 ++     if {$ka != 0} {
 ++         set i [lsearch -exact $parents($v,$ki) $id]
 ++         set j [expr {[llength $parents($v,$ki)] - 1 - $i}]
 ++         append tok [strrep $j]
 ++     }
 ++     if {$tok eq $oldtok} {
 ++         continue
 ++     }
 ++     set id [lindex $varccommits($v,$a) end]
 ++     foreach p $parents($v,$id) {
 ++         if {[info exists varcid($v,$p)]} {
 ++             set kidchanged($varcid($v,$p)) 1
 ++         } else {
 ++             set sortkids($p) 1
 ++         }
 ++     }
 ++     lset varctok($v) $a $tok
 ++     set b [lindex $vupptr($v) $a]
 ++     if {$b != $ka} {
 ++         if {[string compare [lindex $varctok($v) $ka] $vtokmod($v)] < 0} {
 ++             modify_arc $v $ka
 ++         }
 ++         if {[string compare [lindex $varctok($v) $b] $vtokmod($v)] < 0} {
 ++             modify_arc $v $b
 ++         }
 ++         set c [lindex $vbackptr($v) $a]
 ++         set d [lindex $vleftptr($v) $a]
 ++         if {$c == 0} {
 ++             lset vdownptr($v) $b $d
 ++         } else {
 ++             lset vleftptr($v) $c $d
 ++         }
 ++         if {$d != 0} {
 ++             lset vbackptr($v) $d $c
 ++         }
 ++         if {[lindex $vlastins($v) $b] == $a} {
 ++             lset vlastins($v) $b $c
 ++         }
 ++         lset vupptr($v) $a $ka
 ++         set c [lindex $vlastins($v) $ka]
 ++         if {$c == 0 || \
 ++                 [string compare $tok [lindex $varctok($v) $c]] < 0} {
 ++             set c $ka
 ++             set b [lindex $vdownptr($v) $ka]
 ++         } else {
 ++             set b [lindex $vleftptr($v) $c]
 ++         }
 ++         while {$b != 0 && \
 ++                   [string compare $tok [lindex $varctok($v) $b]] >= 0} {
 ++             set c $b
 ++             set b [lindex $vleftptr($v) $c]
 ++         }
 ++         if {$c == $ka} {
 ++             lset vdownptr($v) $ka $a
 ++             lset vbackptr($v) $a 0
 ++         } else {
 ++             lset vleftptr($v) $c $a
 ++             lset vbackptr($v) $a $c
 ++         }
 ++         lset vleftptr($v) $a $b
 ++         if {$b != 0} {
 ++             lset vbackptr($v) $b $a
 ++         }
 ++         lset vlastins($v) $ka $a
 ++     }
 ++    }
 ++    foreach id [array names sortkids] {
 ++     if {[llength $children($v,$id)] > 1} {
 ++         set children($v,$id) [lsort -command [list vtokcmp $v] \
 ++                                   $children($v,$id)]
 ++     }
 ++    }
 ++    set t2 [clock clicks -milliseconds]
 ++    #puts "renumbervarc did [llength $todo] of $ntot arcs in [expr {$t2-$t1}]ms"
 ++}
 ++
 ++# Fix up the graph after we have found out that in view $v,
 ++# $p (a commit that we have already seen) is actually the parent
 ++# of the last commit in arc $a.
 ++proc fix_reversal {p a v} {
 ++    global varcid varcstart varctok vupptr
 ++
 ++    set pa $varcid($v,$p)
 ++    if {$p ne [lindex $varcstart($v) $pa]} {
 ++     splitvarc $p $v
 ++     set pa $varcid($v,$p)
 ++    }
 ++    # seeds always need to be renumbered
 ++    if {[lindex $vupptr($v) $pa] == 0 ||
 ++     [string compare [lindex $varctok($v) $a] \
 ++          [lindex $varctok($v) $pa]] > 0} {
 ++     renumbervarc $pa $v
 ++    }
 ++}
 ++
 ++proc insertrow {id p v} {
 ++    global cmitlisted children parents varcid varctok vtokmod
 ++    global varccommits ordertok commitidx numcommits curview
 ++    global targetid targetrow
 ++
 ++    readcommit $id
 ++    set vid $v,$id
 ++    set cmitlisted($vid) 1
 ++    set children($vid) {}
 ++    set parents($vid) [list $p]
 ++    set a [newvarc $v $id]
 ++    set varcid($vid) $a
 ++    if {[string compare [lindex $varctok($v) $a] $vtokmod($v)] < 0} {
 ++     modify_arc $v $a
 ++    }
 ++    lappend varccommits($v,$a) $id
 ++    set vp $v,$p
 ++    if {[llength [lappend children($vp) $id]] > 1} {
 ++     set children($vp) [lsort -command [list vtokcmp $v] $children($vp)]
 ++     catch {unset ordertok}
 ++    }
 ++    fix_reversal $p $a $v
 ++    incr commitidx($v)
 ++    if {$v == $curview} {
 ++     set numcommits $commitidx($v)
 ++     setcanvscroll
 ++     if {[info exists targetid]} {
 ++         if {![comes_before $targetid $p]} {
 ++             incr targetrow
 ++         }
 ++     }
 ++    }
 ++}
 ++
 ++proc insertfakerow {id p} {
 ++    global varcid varccommits parents children cmitlisted
 ++    global commitidx varctok vtokmod targetid targetrow curview numcommits
 ++
 ++    set v $curview
 ++    set a $varcid($v,$p)
 ++    set i [lsearch -exact $varccommits($v,$a) $p]
 ++    if {$i < 0} {
 ++     puts "oops: insertfakerow can't find [shortids $p] on arc $a"
 ++     return
 ++    }
 ++    set children($v,$id) {}
 ++    set parents($v,$id) [list $p]
 ++    set varcid($v,$id) $a
 ++    lappend children($v,$p) $id
 ++    set cmitlisted($v,$id) 1
 ++    set numcommits [incr commitidx($v)]
 ++    # note we deliberately don't update varcstart($v) even if $i == 0
 ++    set varccommits($v,$a) [linsert $varccommits($v,$a) $i $id]
 ++    modify_arc $v $a $i
 ++    if {[info exists targetid]} {
 ++     if {![comes_before $targetid $p]} {
 ++         incr targetrow
 ++     }
 ++    }
 ++    setcanvscroll
 ++    drawvisible
 ++}
 ++
 ++proc removefakerow {id} {
 ++    global varcid varccommits parents children commitidx
 ++    global varctok vtokmod cmitlisted currentid selectedline
 ++    global targetid curview numcommits
 ++
 ++    set v $curview
 ++    if {[llength $parents($v,$id)] != 1} {
 ++     puts "oops: removefakerow [shortids $id] has [llength $parents($v,$id)] parents"
 ++     return
 ++    }
 ++    set p [lindex $parents($v,$id) 0]
 ++    set a $varcid($v,$id)
 ++    set i [lsearch -exact $varccommits($v,$a) $id]
 ++    if {$i < 0} {
 ++     puts "oops: removefakerow can't find [shortids $id] on arc $a"
 ++     return
 ++    }
 ++    unset varcid($v,$id)
 ++    set varccommits($v,$a) [lreplace $varccommits($v,$a) $i $i]
 ++    unset parents($v,$id)
 ++    unset children($v,$id)
 ++    unset cmitlisted($v,$id)
 ++    set numcommits [incr commitidx($v) -1]
 ++    set j [lsearch -exact $children($v,$p) $id]
 ++    if {$j >= 0} {
 ++     set children($v,$p) [lreplace $children($v,$p) $j $j]
 ++    }
 ++    modify_arc $v $a $i
 ++    if {[info exist currentid] && $id eq $currentid} {
 ++     unset currentid
 ++     set selectedline {}
 ++    }
 ++    if {[info exists targetid] && $targetid eq $id} {
 ++     set targetid $p
 ++    }
 ++    setcanvscroll
 ++    drawvisible
 ++}
 ++
 ++proc first_real_child {vp} {
 ++    global children nullid nullid2
 ++
 ++    foreach id $children($vp) {
 ++     if {$id ne $nullid && $id ne $nullid2} {
 ++         return $id
 ++     }
 ++    }
 ++    return {}
 ++}
 ++
 ++proc last_real_child {vp} {
 ++    global children nullid nullid2
 ++
 ++    set kids $children($vp)
 ++    for {set i [llength $kids]} {[incr i -1] >= 0} {} {
 ++     set id [lindex $kids $i]
 ++     if {$id ne $nullid && $id ne $nullid2} {
 ++         return $id
 ++     }
 ++    }
 ++    return {}
 ++}
 ++
 ++proc vtokcmp {v a b} {
 ++    global varctok varcid
 ++
 ++    return [string compare [lindex $varctok($v) $varcid($v,$a)] \
 ++             [lindex $varctok($v) $varcid($v,$b)]]
 ++}
 ++
 ++# This assumes that if lim is not given, the caller has checked that
 ++# arc a's token is less than $vtokmod($v)
 ++proc modify_arc {v a {lim {}}} {
 ++    global varctok vtokmod varcmod varcrow vupptr curview vrowmod varccommits
 ++
 ++    if {$lim ne {}} {
 ++     set c [string compare [lindex $varctok($v) $a] $vtokmod($v)]
 ++     if {$c > 0} return
 ++     if {$c == 0} {
 ++         set r [lindex $varcrow($v) $a]
 ++         if {$r ne {} && $vrowmod($v) <= $r + $lim} return
 ++     }
 ++    }
 ++    set vtokmod($v) [lindex $varctok($v) $a]
 ++    set varcmod($v) $a
 ++    if {$v == $curview} {
 ++     while {$a != 0 && [lindex $varcrow($v) $a] eq {}} {
 ++         set a [lindex $vupptr($v) $a]
 ++         set lim {}
 ++     }
 ++     set r 0
 ++     if {$a != 0} {
 ++         if {$lim eq {}} {
 ++             set lim [llength $varccommits($v,$a)]
 ++         }
 ++         set r [expr {[lindex $varcrow($v) $a] + $lim}]
 ++     }
 ++     set vrowmod($v) $r
 ++     undolayout $r
 ++    }
 ++}
 ++
 ++proc update_arcrows {v} {
 ++    global vtokmod varcmod vrowmod varcrow commitidx currentid selectedline
 ++    global varcid vrownum varcorder varcix varccommits
 ++    global vupptr vdownptr vleftptr varctok
 ++    global displayorder parentlist curview cached_commitrow
 ++
 ++    if {$vrowmod($v) == $commitidx($v)} return
 ++    if {$v == $curview} {
 ++     if {[llength $displayorder] > $vrowmod($v)} {
 ++         set displayorder [lrange $displayorder 0 [expr {$vrowmod($v) - 1}]]
 ++         set parentlist [lrange $parentlist 0 [expr {$vrowmod($v) - 1}]]
 ++     }
 ++     catch {unset cached_commitrow}
 ++    }
 ++    set narctot [expr {[llength $varctok($v)] - 1}]
 ++    set a $varcmod($v)
 ++    while {$a != 0 && [lindex $varcix($v) $a] eq {}} {
 ++     # go up the tree until we find something that has a row number,
 ++     # or we get to a seed
 ++     set a [lindex $vupptr($v) $a]
 ++    }
 ++    if {$a == 0} {
 ++     set a [lindex $vdownptr($v) 0]
 ++     if {$a == 0} return
 ++     set vrownum($v) {0}
 ++     set varcorder($v) [list $a]
 ++     lset varcix($v) $a 0
 ++     lset varcrow($v) $a 0
 ++     set arcn 0
 ++     set row 0
 ++    } else {
 ++     set arcn [lindex $varcix($v) $a]
 ++     if {[llength $vrownum($v)] > $arcn + 1} {
 ++         set vrownum($v) [lrange $vrownum($v) 0 $arcn]
 ++         set varcorder($v) [lrange $varcorder($v) 0 $arcn]
 ++     }
 ++     set row [lindex $varcrow($v) $a]
 ++    }
 ++    while {1} {
 ++     set p $a
 ++     incr row [llength $varccommits($v,$a)]
 ++     # go down if possible
 ++     set b [lindex $vdownptr($v) $a]
 ++     if {$b == 0} {
 ++         # if not, go left, or go up until we can go left
 ++         while {$a != 0} {
 ++             set b [lindex $vleftptr($v) $a]
 ++             if {$b != 0} break
 ++             set a [lindex $vupptr($v) $a]
 ++         }
 ++         if {$a == 0} break
 ++     }
 ++     set a $b
 ++     incr arcn
 ++     lappend vrownum($v) $row
 ++     lappend varcorder($v) $a
 ++     lset varcix($v) $a $arcn
 ++     lset varcrow($v) $a $row
 ++    }
 ++    set vtokmod($v) [lindex $varctok($v) $p]
 ++    set varcmod($v) $p
 ++    set vrowmod($v) $row
 ++    if {[info exists currentid]} {
 ++     set selectedline [rowofcommit $currentid]
 ++    }
 ++}
 ++
 ++# Test whether view $v contains commit $id
 ++proc commitinview {id v} {
 ++    global varcid
 ++
 ++    return [info exists varcid($v,$id)]
 ++}
 ++
 ++# Return the row number for commit $id in the current view
 ++proc rowofcommit {id} {
 ++    global varcid varccommits varcrow curview cached_commitrow
 ++    global varctok vtokmod
 ++
 ++    set v $curview
 ++    if {![info exists varcid($v,$id)]} {
 ++     puts "oops rowofcommit no arc for [shortids $id]"
 ++     return {}
 ++    }
 ++    set a $varcid($v,$id)
 ++    if {[string compare [lindex $varctok($v) $a] $vtokmod($v)] >= 0} {
 ++     update_arcrows $v
 ++    }
 ++    if {[info exists cached_commitrow($id)]} {
 ++     return $cached_commitrow($id)
 ++    }
 ++    set i [lsearch -exact $varccommits($v,$a) $id]
 ++    if {$i < 0} {
 ++     puts "oops didn't find commit [shortids $id] in arc $a"
 ++     return {}
 ++    }
 ++    incr i [lindex $varcrow($v) $a]
 ++    set cached_commitrow($id) $i
 ++    return $i
 ++}
 ++
 ++# Returns 1 if a is on an earlier row than b, otherwise 0
 ++proc comes_before {a b} {
 ++    global varcid varctok curview
 ++
 ++    set v $curview
 ++    if {$a eq $b || ![info exists varcid($v,$a)] || \
 ++         ![info exists varcid($v,$b)]} {
 ++     return 0
 ++    }
 ++    if {$varcid($v,$a) != $varcid($v,$b)} {
 ++     return [expr {[string compare [lindex $varctok($v) $varcid($v,$a)] \
 ++                        [lindex $varctok($v) $varcid($v,$b)]] < 0}]
 ++    }
 ++    return [expr {[rowofcommit $a] < [rowofcommit $b]}]
 ++}
 ++
 ++proc bsearch {l elt} {
 ++    if {[llength $l] == 0 || $elt <= [lindex $l 0]} {
 ++     return 0
 ++    }
 ++    set lo 0
 ++    set hi [llength $l]
 ++    while {$hi - $lo > 1} {
 ++     set mid [expr {int(($lo + $hi) / 2)}]
 ++     set t [lindex $l $mid]
 ++     if {$elt < $t} {
 ++         set hi $mid
 ++     } elseif {$elt > $t} {
 ++         set lo $mid
 ++     } else {
 ++         return $mid
 ++     }
 ++    }
 ++    return $lo
 ++}
 ++
 ++# Make sure rows $start..$end-1 are valid in displayorder and parentlist
 ++proc make_disporder {start end} {
 ++    global vrownum curview commitidx displayorder parentlist
 ++    global varccommits varcorder parents vrowmod varcrow
 ++    global d_valid_start d_valid_end
 ++
 ++    if {$end > $vrowmod($curview)} {
 ++     update_arcrows $curview
 ++    }
 ++    set ai [bsearch $vrownum($curview) $start]
 ++    set start [lindex $vrownum($curview) $ai]
 ++    set narc [llength $vrownum($curview)]
 ++    for {set r $start} {$ai < $narc && $r < $end} {incr ai} {
 ++     set a [lindex $varcorder($curview) $ai]
 ++     set l [llength $displayorder]
 ++     set al [llength $varccommits($curview,$a)]
 ++     if {$l < $r + $al} {
 ++         if {$l < $r} {
 ++             set pad [ntimes [expr {$r - $l}] {}]
 ++             set displayorder [concat $displayorder $pad]
 ++             set parentlist [concat $parentlist $pad]
 ++         } elseif {$l > $r} {
 ++             set displayorder [lrange $displayorder 0 [expr {$r - 1}]]
 ++             set parentlist [lrange $parentlist 0 [expr {$r - 1}]]
 ++         }
 ++         foreach id $varccommits($curview,$a) {
 ++             lappend displayorder $id
 ++             lappend parentlist $parents($curview,$id)
 ++         }
 ++     } elseif {[lindex $displayorder [expr {$r + $al - 1}]] eq {}} {
 ++         set i $r
 ++         foreach id $varccommits($curview,$a) {
 ++             lset displayorder $i $id
 ++             lset parentlist $i $parents($curview,$id)
 ++             incr i
 ++         }
 ++     }
 ++     incr r $al
 ++    }
 ++}
 ++
 ++proc commitonrow {row} {
 ++    global displayorder
 ++
 ++    set id [lindex $displayorder $row]
 ++    if {$id eq {}} {
 ++     make_disporder $row [expr {$row + 1}]
 ++     set id [lindex $displayorder $row]
 ++    }
 ++    return $id
 ++}
 ++
 ++proc closevarcs {v} {
 ++    global varctok varccommits varcid parents children
 ++    global cmitlisted commitidx commitinterest vtokmod
 ++
 ++    set missing_parents 0
 ++    set scripts {}
 ++    set narcs [llength $varctok($v)]
 ++    for {set a 1} {$a < $narcs} {incr a} {
 ++     set id [lindex $varccommits($v,$a) end]
 ++     foreach p $parents($v,$id) {
 ++         if {[info exists varcid($v,$p)]} continue
 ++         # add p as a new commit
 ++         incr missing_parents
 ++         set cmitlisted($v,$p) 0
 ++         set parents($v,$p) {}
 ++         if {[llength $children($v,$p)] == 1 &&
 ++             [llength $parents($v,$id)] == 1} {
 ++             set b $a
 ++         } else {
 ++             set b [newvarc $v $p]
 ++         }
 ++         set varcid($v,$p) $b
 ++         if {[string compare [lindex $varctok($v) $b] $vtokmod($v)] < 0} {
 ++             modify_arc $v $b
 ++         }
 ++         lappend varccommits($v,$b) $p
 ++         incr commitidx($v)
 ++         if {[info exists commitinterest($p)]} {
 ++             foreach script $commitinterest($p) {
 ++                 lappend scripts [string map [list "%I" $p] $script]
 ++             }
 ++             unset commitinterest($id)
 ++         }
 ++     }
 ++    }
 ++    if {$missing_parents > 0} {
 ++     foreach s $scripts {
 ++         eval $s
 ++     }
 ++    }
 ++}
 ++
 ++# Use $rwid as a substitute for $id, i.e. reparent $id's children to $rwid
 ++# Assumes we already have an arc for $rwid.
 ++proc rewrite_commit {v id rwid} {
 ++    global children parents varcid varctok vtokmod varccommits
 ++
 ++    foreach ch $children($v,$id) {
 ++     # make $rwid be $ch's parent in place of $id
 ++     set i [lsearch -exact $parents($v,$ch) $id]
 ++     if {$i < 0} {
 ++         puts "oops rewrite_commit didn't find $id in parent list for $ch"
 ++     }
 ++     set parents($v,$ch) [lreplace $parents($v,$ch) $i $i $rwid]
 ++     # add $ch to $rwid's children and sort the list if necessary
 ++     if {[llength [lappend children($v,$rwid) $ch]] > 1} {
 ++         set children($v,$rwid) [lsort -command [list vtokcmp $v] \
 ++                                     $children($v,$rwid)]
 ++     }
 ++     # fix the graph after joining $id to $rwid
 ++     set a $varcid($v,$ch)
 ++     fix_reversal $rwid $a $v
 ++     # parentlist is wrong for the last element of arc $a
 ++     # even if displayorder is right, hence the 3rd arg here
 ++     modify_arc $v $a [expr {[llength $varccommits($v,$a)] - 1}]
 ++    }
 ++}
 ++
 ++proc getcommitlines {fd inst view updating}  {
 ++    global cmitlisted commitinterest leftover
 ++    global commitidx commitdata vdatemode
 ++    global parents children curview hlview
 ++    global idpending ordertok
 ++    global varccommits varcid varctok vtokmod vfilelimit
 ++
 ++    set stuff [read $fd 500000]
 ++    # git log doesn't terminate the last commit with a null...
 ++    if {$stuff == {} && $leftover($inst) ne {} && [eof $fd]} {
 ++     set stuff "\0"
 ++    }
 ++    if {$stuff == {}} {
 ++     if {![eof $fd]} {
 ++         return 1
 ++     }
 ++     global commfd viewcomplete viewactive viewname
 ++     global viewinstances
 ++     unset commfd($inst)
 ++     set i [lsearch -exact $viewinstances($view) $inst]
 ++     if {$i >= 0} {
 ++         set viewinstances($view) [lreplace $viewinstances($view) $i $i]
 ++     }
 ++     # set it blocking so we wait for the process to terminate
 ++     fconfigure $fd -blocking 1
 ++     if {[catch {close $fd} err]} {
 ++         set fv {}
 ++         if {$view != $curview} {
 ++             set fv " for the \"$viewname($view)\" view"
 ++         }
 ++         if {[string range $err 0 4] == "usage"} {
 ++             set err "Gitk: error reading commits$fv:\
 ++                     bad arguments to git log."
 ++             if {$viewname($view) eq "Command line"} {
 ++                 append err \
 ++                     "  (Note: arguments to gitk are passed to git log\
 ++                      to allow selection of commits to be displayed.)"
 ++             }
 ++         } else {
 ++             set err "Error reading commits$fv: $err"
 ++         }
 ++         error_popup $err
 ++     }
 ++     if {[incr viewactive($view) -1] <= 0} {
 ++         set viewcomplete($view) 1
 ++         # Check if we have seen any ids listed as parents that haven't
 ++         # appeared in the list
 ++         closevarcs $view
 ++         notbusy $view
 ++     }
 ++     if {$view == $curview} {
 ++         run chewcommits
 ++     }
 ++     return 0
 ++    }
 ++    set start 0
 ++    set gotsome 0
 ++    set scripts {}
 ++    while 1 {
 ++     set i [string first "\0" $stuff $start]
 ++     if {$i < 0} {
 ++         append leftover($inst) [string range $stuff $start end]
 ++         break
 ++     }
 ++     if {$start == 0} {
 ++         set cmit $leftover($inst)
 ++         append cmit [string range $stuff 0 [expr {$i - 1}]]
 ++         set leftover($inst) {}
 ++     } else {
 ++         set cmit [string range $stuff $start [expr {$i - 1}]]
 ++     }
 ++     set start [expr {$i + 1}]
 ++     set j [string first "\n" $cmit]
 ++     set ok 0
 ++     set listed 1
 ++     if {$j >= 0 && [string match "commit *" $cmit]} {
 ++         set ids [string range $cmit 7 [expr {$j - 1}]]
 ++         if {[string match {[-^<>]*} $ids]} {
 ++             switch -- [string index $ids 0] {
 ++                 "-" {set listed 0}
 ++                 "^" {set listed 2}
 ++                 "<" {set listed 3}
 ++                 ">" {set listed 4}
 ++             }
 ++             set ids [string range $ids 1 end]
 ++         }
 ++         set ok 1
 ++         foreach id $ids {
 ++             if {[string length $id] != 40} {
 ++                 set ok 0
 ++                 break
 ++             }
 ++         }
 ++     }
 ++     if {!$ok} {
 ++         set shortcmit $cmit
 ++         if {[string length $shortcmit] > 80} {
 ++             set shortcmit "[string range $shortcmit 0 80]..."
 ++         }
 ++         error_popup "[mc "Can't parse git log output:"] {$shortcmit}"
 ++         exit 1
 ++     }
 ++     set id [lindex $ids 0]
 ++     set vid $view,$id
 ++
 ++     if {!$listed && $updating && ![info exists varcid($vid)] &&
 ++         $vfilelimit($view) ne {}} {
 ++         # git log doesn't rewrite parents for unlisted commits
 ++         # when doing path limiting, so work around that here
 ++         # by working out the rewritten parent with git rev-list
 ++         # and if we already know about it, using the rewritten
 ++         # parent as a substitute parent for $id's children.
 ++         if {![catch {
 ++             set rwid [exec git rev-list --first-parent --max-count=1 \
 ++                           $id -- $vfilelimit($view)]
 ++         }]} {
 ++             if {$rwid ne {} && [info exists varcid($view,$rwid)]} {
 ++                 # use $rwid in place of $id
 ++                 rewrite_commit $view $id $rwid
 ++                 continue
 ++             }
 ++         }
 ++     }
 ++
 ++     set a 0
 ++     if {[info exists varcid($vid)]} {
 ++         if {$cmitlisted($vid) || !$listed} continue
 ++         set a $varcid($vid)
 ++     }
 ++     if {$listed} {
 ++         set olds [lrange $ids 1 end]
 ++     } else {
 ++         set olds {}
 ++     }
 ++     set commitdata($id) [string range $cmit [expr {$j + 1}] end]
 ++     set cmitlisted($vid) $listed
 ++     set parents($vid) $olds
 ++     if {![info exists children($vid)]} {
 ++         set children($vid) {}
 ++     } elseif {$a == 0 && [llength $children($vid)] == 1} {
 ++         set k [lindex $children($vid) 0]
 ++         if {[llength $parents($view,$k)] == 1 &&
 ++             (!$vdatemode($view) ||
 ++              $varcid($view,$k) == [llength $varctok($view)] - 1)} {
 ++             set a $varcid($view,$k)
 ++         }
 ++     }
 ++     if {$a == 0} {
 ++         # new arc
 ++         set a [newvarc $view $id]
 ++     }
 ++     if {[string compare [lindex $varctok($view) $a] $vtokmod($view)] < 0} {
 ++         modify_arc $view $a
 ++     }
 ++     if {![info exists varcid($vid)]} {
 ++         set varcid($vid) $a
 ++         lappend varccommits($view,$a) $id
 ++         incr commitidx($view)
 ++     }
 ++
 ++     set i 0
 ++     foreach p $olds {
 ++         if {$i == 0 || [lsearch -exact $olds $p] >= $i} {
 ++             set vp $view,$p
 ++             if {[llength [lappend children($vp) $id]] > 1 &&
 ++                 [vtokcmp $view [lindex $children($vp) end-1] $id] > 0} {
 ++                 set children($vp) [lsort -command [list vtokcmp $view] \
 ++                                        $children($vp)]
 ++                 catch {unset ordertok}
 ++             }
 ++             if {[info exists varcid($view,$p)]} {
 ++                 fix_reversal $p $a $view
 ++             }
 ++         }
 ++         incr i
 ++     }
 ++
 ++     if {[info exists commitinterest($id)]} {
 ++         foreach script $commitinterest($id) {
 ++             lappend scripts [string map [list "%I" $id] $script]
 ++         }
 ++         unset commitinterest($id)
 ++     }
 ++     set gotsome 1
 ++    }
 ++    if {$gotsome} {
 ++     global numcommits hlview
 ++
 ++     if {$view == $curview} {
 ++         set numcommits $commitidx($view)
 ++         run chewcommits
 ++     }
 ++     if {[info exists hlview] && $view == $hlview} {
 ++         # we never actually get here...
 ++         run vhighlightmore
 ++     }
 ++     foreach s $scripts {
 ++         eval $s
 ++     }
 ++    }
 ++    return 2
 ++}
 ++
 ++proc chewcommits {} {
 ++    global curview hlview viewcomplete
 ++    global pending_select
 ++
 ++    layoutmore
 ++    if {$viewcomplete($curview)} {
 ++     global commitidx varctok
 ++     global numcommits startmsecs
 ++
 ++     if {[info exists pending_select]} {
 ++         update
 ++         reset_pending_select {}
 ++
 ++         if {[commitinview $pending_select $curview]} {
 ++             selectline [rowofcommit $pending_select] 1
 ++         } else {
 ++             set row [first_real_row]
 ++             selectline $row 1
 ++         }
 ++     }
 ++     if {$commitidx($curview) > 0} {
 ++         #set ms [expr {[clock clicks -milliseconds] - $startmsecs}]
 ++         #puts "overall $ms ms for $numcommits commits"
 ++         #puts "[llength $varctok($view)] arcs, $commitidx($view) commits"
 ++     } else {
 ++         show_status [mc "No commits selected"]
 ++     }
 ++     notbusy layout
 ++    }
 ++    return 0
 ++}
 ++
 ++proc readcommit {id} {
 ++    if {[catch {set contents [exec git cat-file commit $id]}]} return
 ++    parsecommit $id $contents 0
 ++}
 ++
 ++proc parsecommit {id contents listed} {
 ++    global commitinfo cdate
 ++
 ++    set inhdr 1
 ++    set comment {}
 ++    set headline {}
 ++    set auname {}
 ++    set audate {}
 ++    set comname {}
 ++    set comdate {}
 ++    set hdrend [string first "\n\n" $contents]
 ++    if {$hdrend < 0} {
 ++     # should never happen...
 ++     set hdrend [string length $contents]
 ++    }
 ++    set header [string range $contents 0 [expr {$hdrend - 1}]]
 ++    set comment [string range $contents [expr {$hdrend + 2}] end]
 ++    foreach line [split $header "\n"] {
 ++     set tag [lindex $line 0]
 ++     if {$tag == "author"} {
 ++         set audate [lindex $line end-1]
 ++         set auname [lrange $line 1 end-2]
 ++     } elseif {$tag == "committer"} {
 ++         set comdate [lindex $line end-1]
 ++         set comname [lrange $line 1 end-2]
 ++     }
 ++    }
 ++    set headline {}
 ++    # take the first non-blank line of the comment as the headline
 ++    set headline [string trimleft $comment]
 ++    set i [string first "\n" $headline]
 ++    if {$i >= 0} {
 ++     set headline [string range $headline 0 $i]
 ++    }
 ++    set headline [string trimright $headline]
 ++    set i [string first "\r" $headline]
 ++    if {$i >= 0} {
 ++     set headline [string trimright [string range $headline 0 $i]]
 ++    }
 ++    if {!$listed} {
 ++     # git log indents the comment by 4 spaces;
 ++     # if we got this via git cat-file, add the indentation
 ++     set newcomment {}
 ++     foreach line [split $comment "\n"] {
 ++         append newcomment "    "
 ++         append newcomment $line
 ++         append newcomment "\n"
 ++     }
 ++     set comment $newcomment
 ++    }
 ++    if {$comdate != {}} {
 ++     set cdate($id) $comdate
 ++    }
 ++    set commitinfo($id) [list $headline $auname $audate \
 ++                          $comname $comdate $comment]
 ++}
 ++
 ++proc getcommit {id} {
 ++    global commitdata commitinfo
 ++
 ++    if {[info exists commitdata($id)]} {
 ++     parsecommit $id $commitdata($id) 1
 ++    } else {
 ++     readcommit $id
 ++     if {![info exists commitinfo($id)]} {
 ++         set commitinfo($id) [list [mc "No commit information available"]]
 ++     }
 ++    }
 ++    return 1
 ++}
 ++
 ++proc readrefs {} {
 ++    global tagids idtags headids idheads tagobjid
 ++    global otherrefids idotherrefs mainhead mainheadid
+++    global selecthead selectheadid
 ++
 ++    foreach v {tagids idtags headids idheads otherrefids idotherrefs} {
 ++     catch {unset $v}
 ++    }
 ++    set refd [open [list | git show-ref -d] r]
 ++    while {[gets $refd line] >= 0} {
 ++     if {[string index $line 40] ne " "} continue
 ++     set id [string range $line 0 39]
 ++     set ref [string range $line 41 end]
 ++     if {![string match "refs/*" $ref]} continue
 ++     set name [string range $ref 5 end]
 ++     if {[string match "remotes/*" $name]} {
 ++         if {![string match "*/HEAD" $name]} {
 ++             set headids($name) $id
 ++             lappend idheads($id) $name
 ++         }
 ++     } elseif {[string match "heads/*" $name]} {
 ++         set name [string range $name 6 end]
 ++         set headids($name) $id
 ++         lappend idheads($id) $name
 ++     } elseif {[string match "tags/*" $name]} {
 ++         # this lets refs/tags/foo^{} overwrite refs/tags/foo,
 ++         # which is what we want since the former is the commit ID
 ++         set name [string range $name 5 end]
 ++         if {[string match "*^{}" $name]} {
 ++             set name [string range $name 0 end-3]
 ++         } else {
 ++             set tagobjid($name) $id
 ++         }
 ++         set tagids($name) $id
 ++         lappend idtags($id) $name
 ++     } else {
 ++         set otherrefids($name) $id
 ++         lappend idotherrefs($id) $name
 ++     }
 ++    }
 ++    catch {close $refd}
 ++    set mainhead {}
 ++    set mainheadid {}
 ++    catch {
 ++     set mainheadid [exec git rev-parse HEAD]
 ++     set thehead [exec git symbolic-ref HEAD]
 ++     if {[string match "refs/heads/*" $thehead]} {
 ++         set mainhead [string range $thehead 11 end]
 ++     }
 ++    }
+++    set selectheadid {}
+++    if {$selecthead ne {}} {
+++     catch {
+++         set selectheadid [exec git rev-parse --verify $selecthead]
+++     }
+++    }
 ++}
 ++
 ++# skip over fake commits
 ++proc first_real_row {} {
 ++    global nullid nullid2 numcommits
 ++
 ++    for {set row 0} {$row < $numcommits} {incr row} {
 ++     set id [commitonrow $row]
 ++     if {$id ne $nullid && $id ne $nullid2} {
 ++         break
 ++     }
 ++    }
 ++    return $row
 ++}
 ++
 ++# update things for a head moved to a child of its previous location
 ++proc movehead {id name} {
 ++    global headids idheads
 ++
 ++    removehead $headids($name) $name
 ++    set headids($name) $id
 ++    lappend idheads($id) $name
 ++}
 ++
 ++# update things when a head has been removed
 ++proc removehead {id name} {
 ++    global headids idheads
 ++
 ++    if {$idheads($id) eq $name} {
 ++     unset idheads($id)
 ++    } else {
 ++     set i [lsearch -exact $idheads($id) $name]
 ++     if {$i >= 0} {
 ++         set idheads($id) [lreplace $idheads($id) $i $i]
 ++     }
 ++    }
 ++    unset headids($name)
 ++}
 ++
 ++proc show_error {w top msg} {
 ++    message $w.m -text $msg -justify center -aspect 400
 ++    pack $w.m -side top -fill x -padx 20 -pady 20
 ++    button $w.ok -text [mc OK] -command "destroy $top"
 ++    pack $w.ok -side bottom -fill x
 ++    bind $top <Visibility> "grab $top; focus $top"
 ++    bind $top <Key-Return> "destroy $top"
 ++    tkwait window $top
 ++}
 ++
 ++proc error_popup msg {
 ++    set w .error
 ++    toplevel $w
 ++    wm transient $w .
 ++    show_error $w $w $msg
 ++}
 ++
 ++proc confirm_popup msg {
 ++    global confirm_ok
 ++    set confirm_ok 0
 ++    set w .confirm
 ++    toplevel $w
 ++    wm transient $w .
 ++    message $w.m -text $msg -justify center -aspect 400
 ++    pack $w.m -side top -fill x -padx 20 -pady 20
 ++    button $w.ok -text [mc OK] -command "set confirm_ok 1; destroy $w"
 ++    pack $w.ok -side left -fill x
 ++    button $w.cancel -text [mc Cancel] -command "destroy $w"
 ++    pack $w.cancel -side right -fill x
 ++    bind $w <Visibility> "grab $w; focus $w"
 ++    tkwait window $w
 ++    return $confirm_ok
 ++}
 ++
 ++proc setoptions {} {
 ++    option add *Panedwindow.showHandle 1 startupFile
 ++    option add *Panedwindow.sashRelief raised startupFile
 ++    option add *Button.font uifont startupFile
 ++    option add *Checkbutton.font uifont startupFile
 ++    option add *Radiobutton.font uifont startupFile
 ++    option add *Menu.font uifont startupFile
 ++    option add *Menubutton.font uifont startupFile
 ++    option add *Label.font uifont startupFile
 ++    option add *Message.font uifont startupFile
 ++    option add *Entry.font uifont startupFile
 ++}
 ++
 ++proc makewindow {} {
 ++    global canv canv2 canv3 linespc charspc ctext cflist cscroll
 ++    global tabstop
 ++    global findtype findtypemenu findloc findstring fstring geometry
 ++    global entries sha1entry sha1string sha1but
 ++    global diffcontextstring diffcontext
 ++    global ignorespace
 ++    global maincursor textcursor curtextcursor
 ++    global rowctxmenu fakerowmenu mergemax wrapcomment
 ++    global highlight_files gdttype
 ++    global searchstring sstring
 ++    global bgcolor fgcolor bglist fglist diffcolors selectbgcolor
 ++    global headctxmenu progresscanv progressitem progresscoords statusw
 ++    global fprogitem fprogcoord lastprogupdate progupdatepending
 ++    global rprogitem rprogcoord rownumsel numcommits
 ++    global have_tk85
 ++
 ++    menu .bar
 ++    .bar add cascade -label [mc "File"] -menu .bar.file
 ++    menu .bar.file
 ++    .bar.file add command -label [mc "Update"] -command updatecommits
 ++    .bar.file add command -label [mc "Reload"] -command reloadcommits
 ++    .bar.file add command -label [mc "Reread references"] -command rereadrefs
 ++    .bar.file add command -label [mc "List references"] -command showrefs
 ++    .bar.file add command -label [mc "Quit"] -command doquit
 ++    menu .bar.edit
 ++    .bar add cascade -label [mc "Edit"] -menu .bar.edit
 ++    .bar.edit add command -label [mc "Preferences"] -command doprefs
 ++
 ++    menu .bar.view
 ++    .bar add cascade -label [mc "View"] -menu .bar.view
 ++    .bar.view add command -label [mc "New view..."] -command {newview 0}
 ++    .bar.view add command -label [mc "Edit view..."] -command editview \
 ++     -state disabled
 ++    .bar.view add command -label [mc "Delete view"] -command delview -state disabled
 ++    .bar.view add separator
 ++    .bar.view add radiobutton -label [mc "All files"] -command {showview 0} \
 ++     -variable selectedview -value 0
 ++
 ++    menu .bar.help
 ++    .bar add cascade -label [mc "Help"] -menu .bar.help
 ++    .bar.help add command -label [mc "About gitk"] -command about
 ++    .bar.help add command -label [mc "Key bindings"] -command keys
 ++    .bar.help configure
 ++    . configure -menu .bar
 ++
 ++    # the gui has upper and lower half, parts of a paned window.
 ++    panedwindow .ctop -orient vertical
 ++
 ++    # possibly use assumed geometry
 ++    if {![info exists geometry(pwsash0)]} {
 ++        set geometry(topheight) [expr {15 * $linespc}]
 ++        set geometry(topwidth) [expr {80 * $charspc}]
 ++        set geometry(botheight) [expr {15 * $linespc}]
 ++        set geometry(botwidth) [expr {50 * $charspc}]
 ++        set geometry(pwsash0) "[expr {40 * $charspc}] 2"
 ++        set geometry(pwsash1) "[expr {60 * $charspc}] 2"
 ++    }
 ++
 ++    # the upper half will have a paned window, a scroll bar to the right, and some stuff below
 ++    frame .tf -height $geometry(topheight) -width $geometry(topwidth)
 ++    frame .tf.histframe
 ++    panedwindow .tf.histframe.pwclist -orient horizontal -sashpad 0 -handlesize 4
 ++
 ++    # create three canvases
 ++    set cscroll .tf.histframe.csb
 ++    set canv .tf.histframe.pwclist.canv
 ++    canvas $canv \
 ++     -selectbackground $selectbgcolor \
 ++     -background $bgcolor -bd 0 \
 ++     -yscrollincr $linespc -yscrollcommand "scrollcanv $cscroll"
 ++    .tf.histframe.pwclist add $canv
 ++    set canv2 .tf.histframe.pwclist.canv2
 ++    canvas $canv2 \
 ++     -selectbackground $selectbgcolor \
 ++     -background $bgcolor -bd 0 -yscrollincr $linespc
 ++    .tf.histframe.pwclist add $canv2
 ++    set canv3 .tf.histframe.pwclist.canv3
 ++    canvas $canv3 \
 ++     -selectbackground $selectbgcolor \
 ++     -background $bgcolor -bd 0 -yscrollincr $linespc
 ++    .tf.histframe.pwclist add $canv3
 ++    eval .tf.histframe.pwclist sash place 0 $geometry(pwsash0)
 ++    eval .tf.histframe.pwclist sash place 1 $geometry(pwsash1)
 ++
 ++    # a scroll bar to rule them
 ++    scrollbar $cscroll -command {allcanvs yview} -highlightthickness 0
 ++    pack $cscroll -side right -fill y
 ++    bind .tf.histframe.pwclist <Configure> {resizeclistpanes %W %w}
 ++    lappend bglist $canv $canv2 $canv3
 ++    pack .tf.histframe.pwclist -fill both -expand 1 -side left
 ++
 ++    # we have two button bars at bottom of top frame. Bar 1
 ++    frame .tf.bar
 ++    frame .tf.lbar -height 15
 ++
 ++    set sha1entry .tf.bar.sha1
 ++    set entries $sha1entry
 ++    set sha1but .tf.bar.sha1label
 ++    button $sha1but -text [mc "SHA1 ID: "] -state disabled -relief flat \
 ++     -command gotocommit -width 8
 ++    $sha1but conf -disabledforeground [$sha1but cget -foreground]
 ++    pack .tf.bar.sha1label -side left
 ++    entry $sha1entry -width 40 -font textfont -textvariable sha1string
 ++    trace add variable sha1string write sha1change
 ++    pack $sha1entry -side left -pady 2
 ++
 ++    image create bitmap bm-left -data {
 ++     #define left_width 16
 ++     #define left_height 16
 ++     static unsigned char left_bits[] = {
 ++     0x00, 0x00, 0xc0, 0x01, 0xe0, 0x00, 0x70, 0x00, 0x38, 0x00, 0x1c, 0x00,
 ++     0x0e, 0x00, 0xff, 0x7f, 0xff, 0x7f, 0xff, 0x7f, 0x0e, 0x00, 0x1c, 0x00,
 ++     0x38, 0x00, 0x70, 0x00, 0xe0, 0x00, 0xc0, 0x01};
 ++    }
 ++    image create bitmap bm-right -data {
 ++     #define right_width 16
 ++     #define right_height 16
 ++     static unsigned char right_bits[] = {
 ++     0x00, 0x00, 0xc0, 0x01, 0x80, 0x03, 0x00, 0x07, 0x00, 0x0e, 0x00, 0x1c,
 ++     0x00, 0x38, 0xff, 0x7f, 0xff, 0x7f, 0xff, 0x7f, 0x00, 0x38, 0x00, 0x1c,
 ++     0x00, 0x0e, 0x00, 0x07, 0x80, 0x03, 0xc0, 0x01};
 ++    }
 ++    button .tf.bar.leftbut -image bm-left -command goback \
 ++     -state disabled -width 26
 ++    pack .tf.bar.leftbut -side left -fill y
 ++    button .tf.bar.rightbut -image bm-right -command goforw \
 ++     -state disabled -width 26
 ++    pack .tf.bar.rightbut -side left -fill y
 ++
 ++    label .tf.bar.rowlabel -text [mc "Row"]
 ++    set rownumsel {}
 ++    label .tf.bar.rownum -width 7 -font textfont -textvariable rownumsel \
 ++     -relief sunken -anchor e
 ++    label .tf.bar.rowlabel2 -text "/"
 ++    label .tf.bar.numcommits -width 7 -font textfont -textvariable numcommits \
 ++     -relief sunken -anchor e
 ++    pack .tf.bar.rowlabel .tf.bar.rownum .tf.bar.rowlabel2 .tf.bar.numcommits \
 ++     -side left
 ++    global selectedline
 ++    trace add variable selectedline write selectedline_change
 ++
 ++    # Status label and progress bar
 ++    set statusw .tf.bar.status
 ++    label $statusw -width 15 -relief sunken
 ++    pack $statusw -side left -padx 5
 ++    set h [expr {[font metrics uifont -linespace] + 2}]
 ++    set progresscanv .tf.bar.progress
 ++    canvas $progresscanv -relief sunken -height $h -borderwidth 2
 ++    set progressitem [$progresscanv create rect -1 0 0 $h -fill green]
 ++    set fprogitem [$progresscanv create rect -1 0 0 $h -fill yellow]
 ++    set rprogitem [$progresscanv create rect -1 0 0 $h -fill red]
 ++    pack $progresscanv -side right -expand 1 -fill x
 ++    set progresscoords {0 0}
 ++    set fprogcoord 0
 ++    set rprogcoord 0
 ++    bind $progresscanv <Configure> adjustprogress
 ++    set lastprogupdate [clock clicks -milliseconds]
 ++    set progupdatepending 0
 ++
 ++    # build up the bottom bar of upper window
 ++    label .tf.lbar.flabel -text "[mc "Find"] "
 ++    button .tf.lbar.fnext -text [mc "next"] -command {dofind 1 1}
 ++    button .tf.lbar.fprev -text [mc "prev"] -command {dofind -1 1}
 ++    label .tf.lbar.flab2 -text " [mc "commit"] "
 ++    pack .tf.lbar.flabel .tf.lbar.fnext .tf.lbar.fprev .tf.lbar.flab2 \
 ++     -side left -fill y
 ++    set gdttype [mc "containing:"]
 ++    set gm [tk_optionMenu .tf.lbar.gdttype gdttype \
 ++             [mc "containing:"] \
 ++             [mc "touching paths:"] \
 ++             [mc "adding/removing string:"]]
 ++    trace add variable gdttype write gdttype_change
 ++    pack .tf.lbar.gdttype -side left -fill y
 ++
 ++    set findstring {}
 ++    set fstring .tf.lbar.findstring
 ++    lappend entries $fstring
 ++    entry $fstring -width 30 -font textfont -textvariable findstring
 ++    trace add variable findstring write find_change
 ++    set findtype [mc "Exact"]
 ++    set findtypemenu [tk_optionMenu .tf.lbar.findtype \
 ++                   findtype [mc "Exact"] [mc "IgnCase"] [mc "Regexp"]]
 ++    trace add variable findtype write findcom_change
 ++    set findloc [mc "All fields"]
 ++    tk_optionMenu .tf.lbar.findloc findloc [mc "All fields"] [mc "Headline"] \
 ++     [mc "Comments"] [mc "Author"] [mc "Committer"]
 ++    trace add variable findloc write find_change
 ++    pack .tf.lbar.findloc -side right
 ++    pack .tf.lbar.findtype -side right
 ++    pack $fstring -side left -expand 1 -fill x
 ++
 ++    # Finish putting the upper half of the viewer together
 ++    pack .tf.lbar -in .tf -side bottom -fill x
 ++    pack .tf.bar -in .tf -side bottom -fill x
 ++    pack .tf.histframe -fill both -side top -expand 1
 ++    .ctop add .tf
 ++    .ctop paneconfigure .tf -height $geometry(topheight)
 ++    .ctop paneconfigure .tf -width $geometry(topwidth)
 ++
 ++    # now build up the bottom
 ++    panedwindow .pwbottom -orient horizontal
 ++
 ++    # lower left, a text box over search bar, scroll bar to the right
 ++    # if we know window height, then that will set the lower text height, otherwise
 ++    # we set lower text height which will drive window height
 ++    if {[info exists geometry(main)]} {
 ++        frame .bleft -width $geometry(botwidth)
 ++    } else {
 ++        frame .bleft -width $geometry(botwidth) -height $geometry(botheight)
 ++    }
 ++    frame .bleft.top
 ++    frame .bleft.mid
 ++    frame .bleft.bottom
 ++
 ++    button .bleft.top.search -text [mc "Search"] -command dosearch
 ++    pack .bleft.top.search -side left -padx 5
 ++    set sstring .bleft.top.sstring
 ++    entry $sstring -width 20 -font textfont -textvariable searchstring
 ++    lappend entries $sstring
 ++    trace add variable searchstring write incrsearch
 ++    pack $sstring -side left -expand 1 -fill x
 ++    radiobutton .bleft.mid.diff -text [mc "Diff"] \
 ++     -command changediffdisp -variable diffelide -value {0 0}
 ++    radiobutton .bleft.mid.old -text [mc "Old version"] \
 ++     -command changediffdisp -variable diffelide -value {0 1}
 ++    radiobutton .bleft.mid.new -text [mc "New version"] \
 ++     -command changediffdisp -variable diffelide -value {1 0}
 ++    label .bleft.mid.labeldiffcontext -text "      [mc "Lines of context"]: "
 ++    pack .bleft.mid.diff .bleft.mid.old .bleft.mid.new -side left
 ++    spinbox .bleft.mid.diffcontext -width 5 -font textfont \
 ++     -from 1 -increment 1 -to 10000000 \
 ++     -validate all -validatecommand "diffcontextvalidate %P" \
 ++     -textvariable diffcontextstring
 ++    .bleft.mid.diffcontext set $diffcontext
 ++    trace add variable diffcontextstring write diffcontextchange
 ++    lappend entries .bleft.mid.diffcontext
 ++    pack .bleft.mid.labeldiffcontext .bleft.mid.diffcontext -side left
 ++    checkbutton .bleft.mid.ignspace -text [mc "Ignore space change"] \
 ++     -command changeignorespace -variable ignorespace
 ++    pack .bleft.mid.ignspace -side left -padx 5
 ++    set ctext .bleft.bottom.ctext
 ++    text $ctext -background $bgcolor -foreground $fgcolor \
 ++     -state disabled -font textfont \
 ++     -yscrollcommand scrolltext -wrap none \
 ++     -xscrollcommand ".bleft.bottom.sbhorizontal set"
 ++    if {$have_tk85} {
 ++     $ctext conf -tabstyle wordprocessor
 ++    }
 ++    scrollbar .bleft.bottom.sb -command "$ctext yview"
 ++    scrollbar .bleft.bottom.sbhorizontal -command "$ctext xview" -orient h \
 ++     -width 10
 ++    pack .bleft.top -side top -fill x
 ++    pack .bleft.mid -side top -fill x
 ++    grid $ctext .bleft.bottom.sb -sticky nsew
 ++    grid .bleft.bottom.sbhorizontal -sticky ew
 ++    grid columnconfigure .bleft.bottom 0 -weight 1
 ++    grid rowconfigure .bleft.bottom 0 -weight 1
 ++    grid rowconfigure .bleft.bottom 1 -weight 0
 ++    pack .bleft.bottom -side top -fill both -expand 1
 ++    lappend bglist $ctext
 ++    lappend fglist $ctext
 ++
 ++    $ctext tag conf comment -wrap $wrapcomment
 ++    $ctext tag conf filesep -font textfontbold -back "#aaaaaa"
 ++    $ctext tag conf hunksep -fore [lindex $diffcolors 2]
 ++    $ctext tag conf d0 -fore [lindex $diffcolors 0]
 ++    $ctext tag conf d1 -fore [lindex $diffcolors 1]
 ++    $ctext tag conf m0 -fore red
 ++    $ctext tag conf m1 -fore blue
 ++    $ctext tag conf m2 -fore green
 ++    $ctext tag conf m3 -fore purple
 ++    $ctext tag conf m4 -fore brown
 ++    $ctext tag conf m5 -fore "#009090"
 ++    $ctext tag conf m6 -fore magenta
 ++    $ctext tag conf m7 -fore "#808000"
 ++    $ctext tag conf m8 -fore "#009000"
 ++    $ctext tag conf m9 -fore "#ff0080"
 ++    $ctext tag conf m10 -fore cyan
 ++    $ctext tag conf m11 -fore "#b07070"
 ++    $ctext tag conf m12 -fore "#70b0f0"
 ++    $ctext tag conf m13 -fore "#70f0b0"
 ++    $ctext tag conf m14 -fore "#f0b070"
 ++    $ctext tag conf m15 -fore "#ff70b0"
 ++    $ctext tag conf mmax -fore darkgrey
 ++    set mergemax 16
 ++    $ctext tag conf mresult -font textfontbold
 ++    $ctext tag conf msep -font textfontbold
 ++    $ctext tag conf found -back yellow
 ++
 ++    .pwbottom add .bleft
 ++    .pwbottom paneconfigure .bleft -width $geometry(botwidth)
 ++
 ++    # lower right
 ++    frame .bright
 ++    frame .bright.mode
 ++    radiobutton .bright.mode.patch -text [mc "Patch"] \
 ++     -command reselectline -variable cmitmode -value "patch"
 ++    radiobutton .bright.mode.tree -text [mc "Tree"] \
 ++     -command reselectline -variable cmitmode -value "tree"
 ++    grid .bright.mode.patch .bright.mode.tree -sticky ew
 ++    pack .bright.mode -side top -fill x
 ++    set cflist .bright.cfiles
 ++    set indent [font measure mainfont "nn"]
 ++    text $cflist \
 ++     -selectbackground $selectbgcolor \
 ++     -background $bgcolor -foreground $fgcolor \
 ++     -font mainfont \
 ++     -tabs [list $indent [expr {2 * $indent}]] \
 ++     -yscrollcommand ".bright.sb set" \
 ++     -cursor [. cget -cursor] \
 ++     -spacing1 1 -spacing3 1
 ++    lappend bglist $cflist
 ++    lappend fglist $cflist
 ++    scrollbar .bright.sb -command "$cflist yview"
 ++    pack .bright.sb -side right -fill y
 ++    pack $cflist -side left -fill both -expand 1
 ++    $cflist tag configure highlight \
 ++     -background [$cflist cget -selectbackground]
 ++    $cflist tag configure bold -font mainfontbold
 ++
 ++    .pwbottom add .bright
 ++    .ctop add .pwbottom
 ++
 ++    # restore window width & height if known
 ++    if {[info exists geometry(main)]} {
 ++     if {[scan $geometry(main) "%dx%d" w h] >= 2} {
 ++         if {$w > [winfo screenwidth .]} {
 ++             set w [winfo screenwidth .]
 ++         }
 ++         if {$h > [winfo screenheight .]} {
 ++             set h [winfo screenheight .]
 ++         }
 ++         wm geometry . "${w}x$h"
 ++     }
 ++    }
 ++
 ++    if {[tk windowingsystem] eq {aqua}} {
 ++        set M1B M1
 ++    } else {
 ++        set M1B Control
 ++    }
 ++
 ++    bind .pwbottom <Configure> {resizecdetpanes %W %w}
 ++    pack .ctop -fill both -expand 1
 ++    bindall <1> {selcanvline %W %x %y}
 ++    #bindall <B1-Motion> {selcanvline %W %x %y}
 ++    if {[tk windowingsystem] == "win32"} {
 ++     bind . <MouseWheel> { windows_mousewheel_redirector %W %X %Y %D }
 ++     bind $ctext <MouseWheel> { windows_mousewheel_redirector %W %X %Y %D ; break }
 ++    } else {
 ++     bindall <ButtonRelease-4> "allcanvs yview scroll -5 units"
 ++     bindall <ButtonRelease-5> "allcanvs yview scroll 5 units"
 ++        if {[tk windowingsystem] eq "aqua"} {
 ++            bindall <MouseWheel> {
 ++                set delta [expr {- (%D)}]
 ++                allcanvs yview scroll $delta units
 ++            }
 ++        }
 ++    }
 ++    bindall <2> "canvscan mark %W %x %y"
 ++    bindall <B2-Motion> "canvscan dragto %W %x %y"
 ++    bindkey <Home> selfirstline
 ++    bindkey <End> sellastline
 ++    bind . <Key-Up> "selnextline -1"
 ++    bind . <Key-Down> "selnextline 1"
 ++    bind . <Shift-Key-Up> "dofind -1 0"
 ++    bind . <Shift-Key-Down> "dofind 1 0"
 ++    bindkey <Key-Right> "goforw"
 ++    bindkey <Key-Left> "goback"
 ++    bind . <Key-Prior> "selnextpage -1"
 ++    bind . <Key-Next> "selnextpage 1"
 ++    bind . <$M1B-Home> "allcanvs yview moveto 0.0"
 ++    bind . <$M1B-End> "allcanvs yview moveto 1.0"
 ++    bind . <$M1B-Key-Up> "allcanvs yview scroll -1 units"
 ++    bind . <$M1B-Key-Down> "allcanvs yview scroll 1 units"
 ++    bind . <$M1B-Key-Prior> "allcanvs yview scroll -1 pages"
 ++    bind . <$M1B-Key-Next> "allcanvs yview scroll 1 pages"
 ++    bindkey <Key-Delete> "$ctext yview scroll -1 pages"
 ++    bindkey <Key-BackSpace> "$ctext yview scroll -1 pages"
 ++    bindkey <Key-space> "$ctext yview scroll 1 pages"
 ++    bindkey p "selnextline -1"
 ++    bindkey n "selnextline 1"
 ++    bindkey z "goback"
 ++    bindkey x "goforw"
 ++    bindkey i "selnextline -1"
 ++    bindkey k "selnextline 1"
 ++    bindkey j "goback"
 ++    bindkey l "goforw"
 ++    bindkey b prevfile
 ++    bindkey d "$ctext yview scroll 18 units"
 ++    bindkey u "$ctext yview scroll -18 units"
 ++    bindkey / {dofind 1 1}
 ++    bindkey <Key-Return> {dofind 1 1}
 ++    bindkey ? {dofind -1 1}
 ++    bindkey f nextfile
 ++    bindkey <F5> updatecommits
 ++    bind . <$M1B-q> doquit
 ++    bind . <$M1B-f> {dofind 1 1}
 ++    bind . <$M1B-g> {dofind 1 0}
 ++    bind . <$M1B-r> dosearchback
 ++    bind . <$M1B-s> dosearch
 ++    bind . <$M1B-equal> {incrfont 1}
 ++    bind . <$M1B-plus> {incrfont 1}
 ++    bind . <$M1B-KP_Add> {incrfont 1}
 ++    bind . <$M1B-minus> {incrfont -1}
 ++    bind . <$M1B-KP_Subtract> {incrfont -1}
 ++    wm protocol . WM_DELETE_WINDOW doquit
 ++    bind . <Destroy> {stop_backends}
 ++    bind . <Button-1> "click %W"
 ++    bind $fstring <Key-Return> {dofind 1 1}
 ++    bind $sha1entry <Key-Return> gotocommit
 ++    bind $sha1entry <<PasteSelection>> clearsha1
 ++    bind $cflist <1> {sel_flist %W %x %y; break}
 ++    bind $cflist <B1-Motion> {sel_flist %W %x %y; break}
 ++    bind $cflist <ButtonRelease-1> {treeclick %W %x %y}
 ++    bind $cflist <Button-3> {pop_flist_menu %W %X %Y %x %y}
 ++
 ++    set maincursor [. cget -cursor]
 ++    set textcursor [$ctext cget -cursor]
 ++    set curtextcursor $textcursor
 ++
 ++    set rowctxmenu .rowctxmenu
 ++    menu $rowctxmenu -tearoff 0
 ++    $rowctxmenu add command -label [mc "Diff this -> selected"] \
 ++     -command {diffvssel 0}
 ++    $rowctxmenu add command -label [mc "Diff selected -> this"] \
 ++     -command {diffvssel 1}
 ++    $rowctxmenu add command -label [mc "Make patch"] -command mkpatch
 ++    $rowctxmenu add command -label [mc "Create tag"] -command mktag
 ++    $rowctxmenu add command -label [mc "Write commit to file"] -command writecommit
 ++    $rowctxmenu add command -label [mc "Create new branch"] -command mkbranch
 ++    $rowctxmenu add command -label [mc "Cherry-pick this commit"] \
 ++     -command cherrypick
 ++    $rowctxmenu add command -label [mc "Reset HEAD branch to here"] \
 ++     -command resethead
 ++
 ++    set fakerowmenu .fakerowmenu
 ++    menu $fakerowmenu -tearoff 0
 ++    $fakerowmenu add command -label [mc "Diff this -> selected"] \
 ++     -command {diffvssel 0}
 ++    $fakerowmenu add command -label [mc "Diff selected -> this"] \
 ++     -command {diffvssel 1}
 ++    $fakerowmenu add command -label [mc "Make patch"] -command mkpatch
 ++#    $fakerowmenu add command -label [mc "Commit"] -command {mkcommit 0}
 ++#    $fakerowmenu add command -label [mc "Commit all"] -command {mkcommit 1}
 ++#    $fakerowmenu add command -label [mc "Revert local changes"] -command revertlocal
 ++
 ++    set headctxmenu .headctxmenu
 ++    menu $headctxmenu -tearoff 0
 ++    $headctxmenu add command -label [mc "Check out this branch"] \
 ++     -command cobranch
 ++    $headctxmenu add command -label [mc "Remove this branch"] \
 ++     -command rmbranch
 ++
 ++    global flist_menu
 ++    set flist_menu .flistctxmenu
 ++    menu $flist_menu -tearoff 0
 ++    $flist_menu add command -label [mc "Highlight this too"] \
 ++     -command {flist_hl 0}
 ++    $flist_menu add command -label [mc "Highlight this only"] \
 ++     -command {flist_hl 1}
 ++    $flist_menu add command -label [mc "External diff"] \
 ++        -command {external_diff}
+++    $flist_menu add command -label [mc "Blame parent commit"] \
+++        -command {external_blame 1}
 ++}
 ++
 ++# Windows sends all mouse wheel events to the current focused window, not
 ++# the one where the mouse hovers, so bind those events here and redirect
 ++# to the correct window
 ++proc windows_mousewheel_redirector {W X Y D} {
 ++    global canv canv2 canv3
 ++    set w [winfo containing -displayof $W $X $Y]
 ++    if {$w ne ""} {
 ++     set u [expr {$D < 0 ? 5 : -5}]
 ++     if {$w == $canv || $w == $canv2 || $w == $canv3} {
 ++         allcanvs yview scroll $u units
 ++     } else {
 ++         catch {
 ++             $w yview scroll $u units
 ++         }
 ++     }
 ++    }
 ++}
 ++
 ++# Update row number label when selectedline changes
 ++proc selectedline_change {n1 n2 op} {
 ++    global selectedline rownumsel
 ++
 ++    if {$selectedline eq {}} {
 ++     set rownumsel {}
 ++    } else {
 ++     set rownumsel [expr {$selectedline + 1}]
 ++    }
 ++}
 ++
 ++# mouse-2 makes all windows scan vertically, but only the one
 ++# the cursor is in scans horizontally
 ++proc canvscan {op w x y} {
 ++    global canv canv2 canv3
 ++    foreach c [list $canv $canv2 $canv3] {
 ++     if {$c == $w} {
 ++         $c scan $op $x $y
 ++     } else {
 ++         $c scan $op 0 $y
 ++     }
 ++    }
 ++}
 ++
 ++proc scrollcanv {cscroll f0 f1} {
 ++    $cscroll set $f0 $f1
 ++    drawvisible
 ++    flushhighlights
 ++}
 ++
 ++# when we make a key binding for the toplevel, make sure
 ++# it doesn't get triggered when that key is pressed in the
 ++# find string entry widget.
 ++proc bindkey {ev script} {
 ++    global entries
 ++    bind . $ev $script
 ++    set escript [bind Entry $ev]
 ++    if {$escript == {}} {
 ++     set escript [bind Entry <Key>]
 ++    }
 ++    foreach e $entries {
 ++     bind $e $ev "$escript; break"
 ++    }
 ++}
 ++
 ++# set the focus back to the toplevel for any click outside
 ++# the entry widgets
 ++proc click {w} {
 ++    global ctext entries
 ++    foreach e [concat $entries $ctext] {
 ++     if {$w == $e} return
 ++    }
 ++    focus .
 ++}
 ++
 ++# Adjust the progress bar for a change in requested extent or canvas size
 ++proc adjustprogress {} {
 ++    global progresscanv progressitem progresscoords
 ++    global fprogitem fprogcoord lastprogupdate progupdatepending
 ++    global rprogitem rprogcoord
 ++
 ++    set w [expr {[winfo width $progresscanv] - 4}]
 ++    set x0 [expr {$w * [lindex $progresscoords 0]}]
 ++    set x1 [expr {$w * [lindex $progresscoords 1]}]
 ++    set h [winfo height $progresscanv]
 ++    $progresscanv coords $progressitem $x0 0 $x1 $h
 ++    $progresscanv coords $fprogitem 0 0 [expr {$w * $fprogcoord}] $h
 ++    $progresscanv coords $rprogitem 0 0 [expr {$w * $rprogcoord}] $h
 ++    set now [clock clicks -milliseconds]
 ++    if {$now >= $lastprogupdate + 100} {
 ++     set progupdatepending 0
 ++     update
 ++    } elseif {!$progupdatepending} {
 ++     set progupdatepending 1
 ++     after [expr {$lastprogupdate + 100 - $now}] doprogupdate
 ++    }
 ++}
 ++
 ++proc doprogupdate {} {
 ++    global lastprogupdate progupdatepending
 ++
 ++    if {$progupdatepending} {
 ++     set progupdatepending 0
 ++     set lastprogupdate [clock clicks -milliseconds]
 ++     update
 ++    }
 ++}
 ++
 ++proc savestuff {w} {
 ++    global canv canv2 canv3 mainfont textfont uifont tabstop
 ++    global stuffsaved findmergefiles maxgraphpct
 ++    global maxwidth showneartags showlocalchanges
 ++    global viewname viewfiles viewargs viewargscmd viewperm nextviewnum
 ++    global cmitmode wrapcomment datetimeformat limitdiffs
 ++    global colors bgcolor fgcolor diffcolors diffcontext selectbgcolor
 ++    global autoselect extdifftool
 ++
 ++    if {$stuffsaved} return
 ++    if {![winfo viewable .]} return
 ++    catch {
 ++     set f [open "~/.gitk-new" w]
 ++     puts $f [list set mainfont $mainfont]
 ++     puts $f [list set textfont $textfont]
 ++     puts $f [list set uifont $uifont]
 ++     puts $f [list set tabstop $tabstop]
 ++     puts $f [list set findmergefiles $findmergefiles]
 ++     puts $f [list set maxgraphpct $maxgraphpct]
 ++     puts $f [list set maxwidth $maxwidth]
 ++     puts $f [list set cmitmode $cmitmode]
 ++     puts $f [list set wrapcomment $wrapcomment]
 ++     puts $f [list set autoselect $autoselect]
 ++     puts $f [list set showneartags $showneartags]
 ++     puts $f [list set showlocalchanges $showlocalchanges]
 ++     puts $f [list set datetimeformat $datetimeformat]
 ++     puts $f [list set limitdiffs $limitdiffs]
 ++     puts $f [list set bgcolor $bgcolor]
 ++     puts $f [list set fgcolor $fgcolor]
 ++     puts $f [list set colors $colors]
 ++     puts $f [list set diffcolors $diffcolors]
 ++     puts $f [list set diffcontext $diffcontext]
 ++     puts $f [list set selectbgcolor $selectbgcolor]
 ++     puts $f [list set extdifftool $extdifftool]
 ++
 ++     puts $f "set geometry(main) [wm geometry .]"
 ++     puts $f "set geometry(topwidth) [winfo width .tf]"
 ++     puts $f "set geometry(topheight) [winfo height .tf]"
 ++        puts $f "set geometry(pwsash0) \"[.tf.histframe.pwclist sash coord 0]\""
 ++        puts $f "set geometry(pwsash1) \"[.tf.histframe.pwclist sash coord 1]\""
 ++     puts $f "set geometry(botwidth) [winfo width .bleft]"
 ++     puts $f "set geometry(botheight) [winfo height .bleft]"
 ++
 ++     puts -nonewline $f "set permviews {"
 ++     for {set v 0} {$v < $nextviewnum} {incr v} {
 ++         if {$viewperm($v)} {
 ++             puts $f "{[list $viewname($v) $viewfiles($v) $viewargs($v) $viewargscmd($v)]}"
 ++         }
 ++     }
 ++     puts $f "}"
 ++     close $f
 ++     file rename -force "~/.gitk-new" "~/.gitk"
 ++    }
 ++    set stuffsaved 1
 ++}
 ++
 ++proc resizeclistpanes {win w} {
 ++    global oldwidth
 ++    if {[info exists oldwidth($win)]} {
 ++     set s0 [$win sash coord 0]
 ++     set s1 [$win sash coord 1]
 ++     if {$w < 60} {
 ++         set sash0 [expr {int($w/2 - 2)}]
 ++         set sash1 [expr {int($w*5/6 - 2)}]
 ++     } else {
 ++         set factor [expr {1.0 * $w / $oldwidth($win)}]
 ++         set sash0 [expr {int($factor * [lindex $s0 0])}]
 ++         set sash1 [expr {int($factor * [lindex $s1 0])}]
 ++         if {$sash0 < 30} {
 ++             set sash0 30
 ++         }
 ++         if {$sash1 < $sash0 + 20} {
 ++             set sash1 [expr {$sash0 + 20}]
 ++         }
 ++         if {$sash1 > $w - 10} {
 ++             set sash1 [expr {$w - 10}]
 ++             if {$sash0 > $sash1 - 20} {
 ++                 set sash0 [expr {$sash1 - 20}]
 ++             }
 ++         }
 ++     }
 ++     $win sash place 0 $sash0 [lindex $s0 1]
 ++     $win sash place 1 $sash1 [lindex $s1 1]
 ++    }
 ++    set oldwidth($win) $w
 ++}
 ++
 ++proc resizecdetpanes {win w} {
 ++    global oldwidth
 ++    if {[info exists oldwidth($win)]} {
 ++     set s0 [$win sash coord 0]
 ++     if {$w < 60} {
 ++         set sash0 [expr {int($w*3/4 - 2)}]
 ++     } else {
 ++         set factor [expr {1.0 * $w / $oldwidth($win)}]
 ++         set sash0 [expr {int($factor * [lindex $s0 0])}]
 ++         if {$sash0 < 45} {
 ++             set sash0 45
 ++         }
 ++         if {$sash0 > $w - 15} {
 ++             set sash0 [expr {$w - 15}]
 ++         }
 ++     }
 ++     $win sash place 0 $sash0 [lindex $s0 1]
 ++    }
 ++    set oldwidth($win) $w
 ++}
 ++
 ++proc allcanvs args {
 ++    global canv canv2 canv3
 ++    eval $canv $args
 ++    eval $canv2 $args
 ++    eval $canv3 $args
 ++}
 ++
 ++proc bindall {event action} {
 ++    global canv canv2 canv3
 ++    bind $canv $event $action
 ++    bind $canv2 $event $action
 ++    bind $canv3 $event $action
 ++}
 ++
 ++proc about {} {
 ++    global uifont
 ++    set w .about
 ++    if {[winfo exists $w]} {
 ++     raise $w
 ++     return
 ++    }
 ++    toplevel $w
 ++    wm title $w [mc "About gitk"]
 ++    message $w.m -text [mc "
 ++Gitk - a commit viewer for git
 ++
 ++Copyright © 2005-2008 Paul Mackerras
 ++
 ++Use and redistribute under the terms of the GNU General Public License"] \
 ++         -justify center -aspect 400 -border 2 -bg white -relief groove
 ++    pack $w.m -side top -fill x -padx 2 -pady 2
 ++    button $w.ok -text [mc "Close"] -command "destroy $w" -default active
 ++    pack $w.ok -side bottom
 ++    bind $w <Visibility> "focus $w.ok"
 ++    bind $w <Key-Escape> "destroy $w"
 ++    bind $w <Key-Return> "destroy $w"
 ++}
 ++
 ++proc keys {} {
 ++    set w .keys
 ++    if {[winfo exists $w]} {
 ++     raise $w
 ++     return
 ++    }
 ++    if {[tk windowingsystem] eq {aqua}} {
 ++     set M1T Cmd
 ++    } else {
 ++     set M1T Ctrl
 ++    }
 ++    toplevel $w
 ++    wm title $w [mc "Gitk key bindings"]
 ++    message $w.m -text "
 ++[mc "Gitk key bindings:"]
 ++
 ++[mc "<%s-Q>          Quit" $M1T]
 ++[mc "<Home>          Move to first commit"]
 ++[mc "<End>           Move to last commit"]
 ++[mc "<Up>, p, i      Move up one commit"]
 ++[mc "<Down>, n, k    Move down one commit"]
 ++[mc "<Left>, z, j    Go back in history list"]
 ++[mc "<Right>, x, l   Go forward in history list"]
 ++[mc "<PageUp>        Move up one page in commit list"]
 ++[mc "<PageDown>      Move down one page in commit list"]
 ++[mc "<%s-Home>       Scroll to top of commit list" $M1T]
 ++[mc "<%s-End>        Scroll to bottom of commit list" $M1T]
 ++[mc "<%s-Up> Scroll commit list up one line" $M1T]
 ++[mc "<%s-Down>       Scroll commit list down one line" $M1T]
 ++[mc "<%s-PageUp>     Scroll commit list up one page" $M1T]
 ++[mc "<%s-PageDown>   Scroll commit list down one page" $M1T]
 ++[mc "<Shift-Up>      Find backwards (upwards, later commits)"]
 ++[mc "<Shift-Down>    Find forwards (downwards, earlier commits)"]
 ++[mc "<Delete>, b     Scroll diff view up one page"]
 ++[mc "<Backspace>     Scroll diff view up one page"]
 ++[mc "<Space>         Scroll diff view down one page"]
 ++[mc "u               Scroll diff view up 18 lines"]
 ++[mc "d               Scroll diff view down 18 lines"]
 ++[mc "<%s-F>          Find" $M1T]
 ++[mc "<%s-G>          Move to next find hit" $M1T]
 ++[mc "<Return>        Move to next find hit"]
 ++[mc "/               Move to next find hit, or redo find"]
 ++[mc "?               Move to previous find hit"]
 ++[mc "f               Scroll diff view to next file"]
 ++[mc "<%s-S>          Search for next hit in diff view" $M1T]
 ++[mc "<%s-R>          Search for previous hit in diff view" $M1T]
 ++[mc "<%s-KP+>        Increase font size" $M1T]
 ++[mc "<%s-plus>       Increase font size" $M1T]
 ++[mc "<%s-KP->        Decrease font size" $M1T]
 ++[mc "<%s-minus>      Decrease font size" $M1T]
 ++[mc "<F5>            Update"]
 ++" \
 ++         -justify left -bg white -border 2 -relief groove
 ++    pack $w.m -side top -fill both -padx 2 -pady 2
 ++    button $w.ok -text [mc "Close"] -command "destroy $w" -default active
 ++    pack $w.ok -side bottom
 ++    bind $w <Visibility> "focus $w.ok"
 ++    bind $w <Key-Escape> "destroy $w"
 ++    bind $w <Key-Return> "destroy $w"
 ++}
 ++
 ++# Procedures for manipulating the file list window at the
 ++# bottom right of the overall window.
 ++
 ++proc treeview {w l openlevs} {
 ++    global treecontents treediropen treeheight treeparent treeindex
 ++
 ++    set ix 0
 ++    set treeindex() 0
 ++    set lev 0
 ++    set prefix {}
 ++    set prefixend -1
 ++    set prefendstack {}
 ++    set htstack {}
 ++    set ht 0
 ++    set treecontents() {}
 ++    $w conf -state normal
 ++    foreach f $l {
 ++     while {[string range $f 0 $prefixend] ne $prefix} {
 ++         if {$lev <= $openlevs} {
 ++             $w mark set e:$treeindex($prefix) "end -1c"
 ++             $w mark gravity e:$treeindex($prefix) left
 ++         }
 ++         set treeheight($prefix) $ht
 ++         incr ht [lindex $htstack end]
 ++         set htstack [lreplace $htstack end end]
 ++         set prefixend [lindex $prefendstack end]
 ++         set prefendstack [lreplace $prefendstack end end]
 ++         set prefix [string range $prefix 0 $prefixend]
 ++         incr lev -1
 ++     }
 ++     set tail [string range $f [expr {$prefixend+1}] end]
 ++     while {[set slash [string first "/" $tail]] >= 0} {
 ++         lappend htstack $ht
 ++         set ht 0
 ++         lappend prefendstack $prefixend
 ++         incr prefixend [expr {$slash + 1}]
 ++         set d [string range $tail 0 $slash]
 ++         lappend treecontents($prefix) $d
 ++         set oldprefix $prefix
 ++         append prefix $d
 ++         set treecontents($prefix) {}
 ++         set treeindex($prefix) [incr ix]
 ++         set treeparent($prefix) $oldprefix
 ++         set tail [string range $tail [expr {$slash+1}] end]
 ++         if {$lev <= $openlevs} {
 ++             set ht 1
 ++             set treediropen($prefix) [expr {$lev < $openlevs}]
 ++             set bm [expr {$lev == $openlevs? "tri-rt": "tri-dn"}]
 ++             $w mark set d:$ix "end -1c"
 ++             $w mark gravity d:$ix left
 ++             set str "\n"
 ++             for {set i 0} {$i < $lev} {incr i} {append str "\t"}
 ++             $w insert end $str
 ++             $w image create end -align center -image $bm -padx 1 \
 ++                 -name a:$ix
 ++             $w insert end $d [highlight_tag $prefix]
 ++             $w mark set s:$ix "end -1c"
 ++             $w mark gravity s:$ix left
 ++         }
 ++         incr lev
 ++     }
 ++     if {$tail ne {}} {
 ++         if {$lev <= $openlevs} {
 ++             incr ht
 ++             set str "\n"
 ++             for {set i 0} {$i < $lev} {incr i} {append str "\t"}
 ++             $w insert end $str
 ++             $w insert end $tail [highlight_tag $f]
 ++         }
 ++         lappend treecontents($prefix) $tail
 ++     }
 ++    }
 ++    while {$htstack ne {}} {
 ++     set treeheight($prefix) $ht
 ++     incr ht [lindex $htstack end]
 ++     set htstack [lreplace $htstack end end]
 ++     set prefixend [lindex $prefendstack end]
 ++     set prefendstack [lreplace $prefendstack end end]
 ++     set prefix [string range $prefix 0 $prefixend]
 ++    }
 ++    $w conf -state disabled
 ++}
 ++
 ++proc linetoelt {l} {
 ++    global treeheight treecontents
 ++
 ++    set y 2
 ++    set prefix {}
 ++    while {1} {
 ++     foreach e $treecontents($prefix) {
 ++         if {$y == $l} {
 ++             return "$prefix$e"
 ++         }
 ++         set n 1
 ++         if {[string index $e end] eq "/"} {
 ++             set n $treeheight($prefix$e)
 ++             if {$y + $n > $l} {
 ++                 append prefix $e
 ++                 incr y
 ++                 break
 ++             }
 ++         }
 ++         incr y $n
 ++     }
 ++    }
 ++}
 ++
 ++proc highlight_tree {y prefix} {
 ++    global treeheight treecontents cflist
 ++
 ++    foreach e $treecontents($prefix) {
 ++     set path $prefix$e
 ++     if {[highlight_tag $path] ne {}} {
 ++         $cflist tag add bold $y.0 "$y.0 lineend"
 ++     }
 ++     incr y
 ++     if {[string index $e end] eq "/" && $treeheight($path) > 1} {
 ++         set y [highlight_tree $y $path]
 ++     }
 ++    }
 ++    return $y
 ++}
 ++
 ++proc treeclosedir {w dir} {
 ++    global treediropen treeheight treeparent treeindex
 ++
 ++    set ix $treeindex($dir)
 ++    $w conf -state normal
 ++    $w delete s:$ix e:$ix
 ++    set treediropen($dir) 0
 ++    $w image configure a:$ix -image tri-rt
 ++    $w conf -state disabled
 ++    set n [expr {1 - $treeheight($dir)}]
 ++    while {$dir ne {}} {
 ++     incr treeheight($dir) $n
 ++     set dir $treeparent($dir)
 ++    }
 ++}
 ++
 ++proc treeopendir {w dir} {
 ++    global treediropen treeheight treeparent treecontents treeindex
 ++
 ++    set ix $treeindex($dir)
 ++    $w conf -state normal
 ++    $w image configure a:$ix -image tri-dn
 ++    $w mark set e:$ix s:$ix
 ++    $w mark gravity e:$ix right
 ++    set lev 0
 ++    set str "\n"
 ++    set n [llength $treecontents($dir)]
 ++    for {set x $dir} {$x ne {}} {set x $treeparent($x)} {
 ++     incr lev
 ++     append str "\t"
 ++     incr treeheight($x) $n
 ++    }
 ++    foreach e $treecontents($dir) {
 ++     set de $dir$e
 ++     if {[string index $e end] eq "/"} {
 ++         set iy $treeindex($de)
 ++         $w mark set d:$iy e:$ix
 ++         $w mark gravity d:$iy left
 ++         $w insert e:$ix $str
 ++         set treediropen($de) 0
 ++         $w image create e:$ix -align center -image tri-rt -padx 1 \
 ++             -name a:$iy
 ++         $w insert e:$ix $e [highlight_tag $de]
 ++         $w mark set s:$iy e:$ix
 ++         $w mark gravity s:$iy left
 ++         set treeheight($de) 1
 ++     } else {
 ++         $w insert e:$ix $str
 ++         $w insert e:$ix $e [highlight_tag $de]
 ++     }
 ++    }
 ++    $w mark gravity e:$ix left
 ++    $w conf -state disabled
 ++    set treediropen($dir) 1
 ++    set top [lindex [split [$w index @0,0] .] 0]
 ++    set ht [$w cget -height]
 ++    set l [lindex [split [$w index s:$ix] .] 0]
 ++    if {$l < $top} {
 ++     $w yview $l.0
 ++    } elseif {$l + $n + 1 > $top + $ht} {
 ++     set top [expr {$l + $n + 2 - $ht}]
 ++     if {$l < $top} {
 ++         set top $l
 ++     }
 ++     $w yview $top.0
 ++    }
 ++}
 ++
 ++proc treeclick {w x y} {
 ++    global treediropen cmitmode ctext cflist cflist_top
 ++
 ++    if {$cmitmode ne "tree"} return
 ++    if {![info exists cflist_top]} return
 ++    set l [lindex [split [$w index "@$x,$y"] "."] 0]
 ++    $cflist tag remove highlight $cflist_top.0 "$cflist_top.0 lineend"
 ++    $cflist tag add highlight $l.0 "$l.0 lineend"
 ++    set cflist_top $l
 ++    if {$l == 1} {
 ++     $ctext yview 1.0
 ++     return
 ++    }
 ++    set e [linetoelt $l]
 ++    if {[string index $e end] ne "/"} {
 ++     showfile $e
 ++    } elseif {$treediropen($e)} {
 ++     treeclosedir $w $e
 ++    } else {
 ++     treeopendir $w $e
 ++    }
 ++}
 ++
 ++proc setfilelist {id} {
 ++    global treefilelist cflist
 ++
 ++    treeview $cflist $treefilelist($id) 0
 ++}
 ++
 ++image create bitmap tri-rt -background black -foreground blue -data {
 ++    #define tri-rt_width 13
 ++    #define tri-rt_height 13
 ++    static unsigned char tri-rt_bits[] = {
 ++       0x00, 0x00, 0x00, 0x00, 0x10, 0x00, 0x30, 0x00, 0x70, 0x00, 0xf0, 0x00,
 ++       0xf0, 0x01, 0xf0, 0x00, 0x70, 0x00, 0x30, 0x00, 0x10, 0x00, 0x00, 0x00,
 ++       0x00, 0x00};
 ++} -maskdata {
 ++    #define tri-rt-mask_width 13
 ++    #define tri-rt-mask_height 13
 ++    static unsigned char tri-rt-mask_bits[] = {
 ++       0x08, 0x00, 0x18, 0x00, 0x38, 0x00, 0x78, 0x00, 0xf8, 0x00, 0xf8, 0x01,
 ++       0xf8, 0x03, 0xf8, 0x01, 0xf8, 0x00, 0x78, 0x00, 0x38, 0x00, 0x18, 0x00,
 ++       0x08, 0x00};
 ++}
 ++image create bitmap tri-dn -background black -foreground blue -data {
 ++    #define tri-dn_width 13
 ++    #define tri-dn_height 13
 ++    static unsigned char tri-dn_bits[] = {
 ++       0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xfc, 0x07, 0xf8, 0x03,
 ++       0xf0, 0x01, 0xe0, 0x00, 0x40, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
 ++       0x00, 0x00};
 ++} -maskdata {
 ++    #define tri-dn-mask_width 13
 ++    #define tri-dn-mask_height 13
 ++    static unsigned char tri-dn-mask_bits[] = {
 ++       0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0x1f, 0xfe, 0x0f, 0xfc, 0x07,
 ++       0xf8, 0x03, 0xf0, 0x01, 0xe0, 0x00, 0x40, 0x00, 0x00, 0x00, 0x00, 0x00,
 ++       0x00, 0x00};
 ++}
 ++
 ++image create bitmap reficon-T -background black -foreground yellow -data {
 ++    #define tagicon_width 13
 ++    #define tagicon_height 9
 ++    static unsigned char tagicon_bits[] = {
 ++       0x00, 0x00, 0x00, 0x00, 0xf0, 0x07, 0xf8, 0x07,
 ++       0xfc, 0x07, 0xf8, 0x07, 0xf0, 0x07, 0x00, 0x00, 0x00, 0x00};
 ++} -maskdata {
 ++    #define tagicon-mask_width 13
 ++    #define tagicon-mask_height 9
 ++    static unsigned char tagicon-mask_bits[] = {
 ++       0x00, 0x00, 0xf0, 0x0f, 0xf8, 0x0f, 0xfc, 0x0f,
 ++       0xfe, 0x0f, 0xfc, 0x0f, 0xf8, 0x0f, 0xf0, 0x0f, 0x00, 0x00};
 ++}
 ++set rectdata {
 ++    #define headicon_width 13
 ++    #define headicon_height 9
 ++    static unsigned char headicon_bits[] = {
 ++       0x00, 0x00, 0x00, 0x00, 0xf8, 0x07, 0xf8, 0x07,
 ++       0xf8, 0x07, 0xf8, 0x07, 0xf8, 0x07, 0x00, 0x00, 0x00, 0x00};
 ++}
 ++set rectmask {
 ++    #define headicon-mask_width 13
 ++    #define headicon-mask_height 9
 ++    static unsigned char headicon-mask_bits[] = {
 ++       0x00, 0x00, 0xfc, 0x0f, 0xfc, 0x0f, 0xfc, 0x0f,
 ++       0xfc, 0x0f, 0xfc, 0x0f, 0xfc, 0x0f, 0xfc, 0x0f, 0x00, 0x00};
 ++}
 ++image create bitmap reficon-H -background black -foreground green \
 ++    -data $rectdata -maskdata $rectmask
 ++image create bitmap reficon-o -background black -foreground "#ddddff" \
 ++    -data $rectdata -maskdata $rectmask
 ++
 ++proc init_flist {first} {
 ++    global cflist cflist_top difffilestart
 ++
 ++    $cflist conf -state normal
 ++    $cflist delete 0.0 end
 ++    if {$first ne {}} {
 ++     $cflist insert end $first
 ++     set cflist_top 1
 ++     $cflist tag add highlight 1.0 "1.0 lineend"
 ++    } else {
 ++     catch {unset cflist_top}
 ++    }
 ++    $cflist conf -state disabled
 ++    set difffilestart {}
 ++}
 ++
 ++proc highlight_tag {f} {
 ++    global highlight_paths
 ++
 ++    foreach p $highlight_paths {
 ++     if {[string match $p $f]} {
 ++         return "bold"
 ++     }
 ++    }
 ++    return {}
 ++}
 ++
 ++proc highlight_filelist {} {
 ++    global cmitmode cflist
 ++
 ++    $cflist conf -state normal
 ++    if {$cmitmode ne "tree"} {
 ++     set end [lindex [split [$cflist index end] .] 0]
 ++     for {set l 2} {$l < $end} {incr l} {
 ++         set line [$cflist get $l.0 "$l.0 lineend"]
 ++         if {[highlight_tag $line] ne {}} {
 ++             $cflist tag add bold $l.0 "$l.0 lineend"
 ++         }
 ++     }
 ++    } else {
 ++     highlight_tree 2 {}
 ++    }
 ++    $cflist conf -state disabled
 ++}
 ++
 ++proc unhighlight_filelist {} {
 ++    global cflist
 ++
 ++    $cflist conf -state normal
 ++    $cflist tag remove bold 1.0 end
 ++    $cflist conf -state disabled
 ++}
 ++
 ++proc add_flist {fl} {
 ++    global cflist
 ++
 ++    $cflist conf -state normal
 ++    foreach f $fl {
 ++     $cflist insert end "\n"
 ++     $cflist insert end $f [highlight_tag $f]
 ++    }
 ++    $cflist conf -state disabled
 ++}
 ++
 ++proc sel_flist {w x y} {
 ++    global ctext difffilestart cflist cflist_top cmitmode
 ++
 ++    if {$cmitmode eq "tree"} return
 ++    if {![info exists cflist_top]} return
 ++    set l [lindex [split [$w index "@$x,$y"] "."] 0]
 ++    $cflist tag remove highlight $cflist_top.0 "$cflist_top.0 lineend"
 ++    $cflist tag add highlight $l.0 "$l.0 lineend"
 ++    set cflist_top $l
 ++    if {$l == 1} {
 ++     $ctext yview 1.0
 ++    } else {
 ++     catch {$ctext yview [lindex $difffilestart [expr {$l - 2}]]}
 ++    }
 ++}
 ++
 ++proc pop_flist_menu {w X Y x y} {
 ++    global ctext cflist cmitmode flist_menu flist_menu_file
 ++    global treediffs diffids
 ++
 ++    stopfinding
 ++    set l [lindex [split [$w index "@$x,$y"] "."] 0]
 ++    if {$l <= 1} return
 ++    if {$cmitmode eq "tree"} {
 ++     set e [linetoelt $l]
 ++     if {[string index $e end] eq "/"} return
 ++    } else {
 ++     set e [lindex $treediffs($diffids) [expr {$l-2}]]
 ++    }
 ++    set flist_menu_file $e
 ++    set xdiffstate "normal"
 ++    if {$cmitmode eq "tree"} {
 ++     set xdiffstate "disabled"
 ++    }
 ++    # Disable "External diff" item in tree mode
 ++    $flist_menu entryconf 2 -state $xdiffstate
 ++    tk_popup $flist_menu $X $Y
 ++}
 ++
 ++proc flist_hl {only} {
 ++    global flist_menu_file findstring gdttype
 ++
 ++    set x [shellquote $flist_menu_file]
 ++    if {$only || $findstring eq {} || $gdttype ne [mc "touching paths:"]} {
 ++     set findstring $x
 ++    } else {
 ++     append findstring " " $x
 ++    }
 ++    set gdttype [mc "touching paths:"]
 ++}
 ++
 ++proc save_file_from_commit {filename output what} {
 ++    global nullfile
 ++
 ++    if {[catch {exec git show $filename -- > $output} err]} {
 ++     if {[string match "fatal: bad revision *" $err]} {
 ++         return $nullfile
 ++     }
 ++     error_popup "Error getting \"$filename\" from $what: $err"
 ++     return {}
 ++    }
 ++    return $output
 ++}
 ++
 ++proc external_diff_get_one_file {diffid filename diffdir} {
 ++    global nullid nullid2 nullfile
 ++    global gitdir
 ++
 ++    if {$diffid == $nullid} {
 ++        set difffile [file join [file dirname $gitdir] $filename]
 ++     if {[file exists $difffile]} {
 ++         return $difffile
 ++     }
 ++     return $nullfile
 ++    }
 ++    if {$diffid == $nullid2} {
 ++        set difffile [file join $diffdir "\[index\] [file tail $filename]"]
 ++        return [save_file_from_commit :$filename $difffile index]
 ++    }
 ++    set difffile [file join $diffdir "\[$diffid\] [file tail $filename]"]
 ++    return [save_file_from_commit $diffid:$filename $difffile \
 ++            "revision $diffid"]
 ++}
 ++
 ++proc external_diff {} {
 ++    global gitktmpdir nullid nullid2
 ++    global flist_menu_file
 ++    global diffids
 ++    global diffnum
 ++    global gitdir extdifftool
 ++
 ++    if {[llength $diffids] == 1} {
 ++        # no reference commit given
 ++        set diffidto [lindex $diffids 0]
 ++        if {$diffidto eq $nullid} {
 ++            # diffing working copy with index
 ++            set diffidfrom $nullid2
 ++        } elseif {$diffidto eq $nullid2} {
 ++            # diffing index with HEAD
 ++            set diffidfrom "HEAD"
 ++        } else {
 ++            # use first parent commit
 ++            global parentlist selectedline
 ++            set diffidfrom [lindex $parentlist $selectedline 0]
 ++        }
 ++    } else {
 ++        set diffidfrom [lindex $diffids 0]
 ++        set diffidto [lindex $diffids 1]
 ++    }
 ++
 ++    # make sure that several diffs wont collide
 ++    if {![info exists gitktmpdir]} {
 ++     set gitktmpdir [file join [file dirname $gitdir] \
 ++                         [format ".gitk-tmp.%s" [pid]]]
 ++     if {[catch {file mkdir $gitktmpdir} err]} {
 ++         error_popup "Error creating temporary directory $gitktmpdir: $err"
 ++         unset gitktmpdir
 ++         return
 ++     }
 ++     set diffnum 0
 ++    }
 ++    incr diffnum
 ++    set diffdir [file join $gitktmpdir $diffnum]
 ++    if {[catch {file mkdir $diffdir} err]} {
 ++     error_popup "Error creating temporary directory $diffdir: $err"
 ++     return
 ++    }
 ++
 ++    # gather files to diff
 ++    set difffromfile [external_diff_get_one_file $diffidfrom $flist_menu_file $diffdir]
 ++    set difftofile [external_diff_get_one_file $diffidto $flist_menu_file $diffdir]
 ++
 ++    if {$difffromfile ne {} && $difftofile ne {}} {
 ++        set cmd [concat | [shellsplit $extdifftool] \
 ++                  [list $difffromfile $difftofile]]
 ++        if {[catch {set fl [open $cmd r]} err]} {
 ++            file delete -force $diffdir
 ++            error_popup [mc "$extdifftool: command failed: $err"]
 ++        } else {
 ++            fconfigure $fl -blocking 0
 ++            filerun $fl [list delete_at_eof $fl $diffdir]
 ++        }
 ++    }
 ++}
 ++
+++proc external_blame {parent_idx} {
+++    global flist_menu_file
+++    global nullid nullid2
+++    global parentlist selectedline currentid
+++
+++    if {$parent_idx > 0} {
+++     set base_commit [lindex $parentlist $selectedline [expr {$parent_idx-1}]]
+++    } else {
+++     set base_commit $currentid
+++    }
+++
+++    if {$base_commit eq {} || $base_commit eq $nullid || $base_commit eq $nullid2} {
+++     error_popup [mc "No such commit"]
+++     return
+++    }
+++
+++    if {[catch {exec git gui blame $base_commit $flist_menu_file &} err]} {
+++     error_popup [mc "git gui blame: command failed: $err"]
+++    }
+++}
+++
 ++# delete $dir when we see eof on $f (presumably because the child has exited)
 ++proc delete_at_eof {f dir} {
 ++    while {[gets $f line] >= 0} {}
 ++    if {[eof $f]} {
 ++     if {[catch {close $f} err]} {
 ++         error_popup "External diff viewer failed: $err"
 ++     }
 ++     file delete -force $dir
 ++     return 0
 ++    }
 ++    return 1
 ++}
 ++
 ++# Functions for adding and removing shell-type quoting
 ++
 ++proc shellquote {str} {
 ++    if {![string match "*\['\"\\ \t]*" $str]} {
 ++     return $str
 ++    }
 ++    if {![string match "*\['\"\\]*" $str]} {
 ++     return "\"$str\""
 ++    }
 ++    if {![string match "*'*" $str]} {
 ++     return "'$str'"
 ++    }
 ++    return "\"[string map {\" \\\" \\ \\\\} $str]\""
 ++}
 ++
 ++proc shellarglist {l} {
 ++    set str {}
 ++    foreach a $l {
 ++     if {$str ne {}} {
 ++         append str " "
 ++     }
 ++     append str [shellquote $a]
 ++    }
 ++    return $str
 ++}
 ++
 ++proc shelldequote {str} {
 ++    set ret {}
 ++    set used -1
 ++    while {1} {
 ++     incr used
 ++     if {![regexp -start $used -indices "\['\"\\\\ \t]" $str first]} {
 ++         append ret [string range $str $used end]
 ++         set used [string length $str]
 ++         break
 ++     }
 ++     set first [lindex $first 0]
 ++     set ch [string index $str $first]
 ++     if {$first > $used} {
 ++         append ret [string range $str $used [expr {$first - 1}]]
 ++         set used $first
 ++     }
 ++     if {$ch eq " " || $ch eq "\t"} break
 ++     incr used
 ++     if {$ch eq "'"} {
 ++         set first [string first "'" $str $used]
 ++         if {$first < 0} {
 ++             error "unmatched single-quote"
 ++         }
 ++         append ret [string range $str $used [expr {$first - 1}]]
 ++         set used $first
 ++         continue
 ++     }
 ++     if {$ch eq "\\"} {
 ++         if {$used >= [string length $str]} {
 ++             error "trailing backslash"
 ++         }
 ++         append ret [string index $str $used]
 ++         continue
 ++     }
 ++     # here ch == "\""
 ++     while {1} {
 ++         if {![regexp -start $used -indices "\[\"\\\\]" $str first]} {
 ++             error "unmatched double-quote"
 ++         }
 ++         set first [lindex $first 0]
 ++         set ch [string index $str $first]
 ++         if {$first > $used} {
 ++             append ret [string range $str $used [expr {$first - 1}]]
 ++             set used $first
 ++         }
 ++         if {$ch eq "\""} break
 ++         incr used
 ++         append ret [string index $str $used]
 ++         incr used
 ++     }
 ++    }
 ++    return [list $used $ret]
 ++}
 ++
 ++proc shellsplit {str} {
 ++    set l {}
 ++    while {1} {
 ++     set str [string trimleft $str]
 ++     if {$str eq {}} break
 ++     set dq [shelldequote $str]
 ++     set n [lindex $dq 0]
 ++     set word [lindex $dq 1]
 ++     set str [string range $str $n end]
 ++     lappend l $word
 ++    }
 ++    return $l
 ++}
 ++
 ++# Code to implement multiple views
 ++
 ++proc newview {ishighlight} {
 ++    global nextviewnum newviewname newviewperm newishighlight
 ++    global newviewargs revtreeargs viewargscmd newviewargscmd curview
 ++
 ++    set newishighlight $ishighlight
 ++    set top .gitkview
 ++    if {[winfo exists $top]} {
 ++     raise $top
 ++     return
 ++    }
 ++    set newviewname($nextviewnum) "[mc "View"] $nextviewnum"
 ++    set newviewperm($nextviewnum) 0
 ++    set newviewargs($nextviewnum) [shellarglist $revtreeargs]
 ++    set newviewargscmd($nextviewnum) $viewargscmd($curview)
 ++    vieweditor $top $nextviewnum [mc "Gitk view definition"]
 ++}
 ++
 ++proc editview {} {
 ++    global curview
 ++    global viewname viewperm newviewname newviewperm
 ++    global viewargs newviewargs viewargscmd newviewargscmd
 ++
 ++    set top .gitkvedit-$curview
 ++    if {[winfo exists $top]} {
 ++     raise $top
 ++     return
 ++    }
 ++    set newviewname($curview) $viewname($curview)
 ++    set newviewperm($curview) $viewperm($curview)
 ++    set newviewargs($curview) [shellarglist $viewargs($curview)]
 ++    set newviewargscmd($curview) $viewargscmd($curview)
 ++    vieweditor $top $curview "Gitk: edit view $viewname($curview)"
 ++}
 ++
 ++proc vieweditor {top n title} {
 ++    global newviewname newviewperm viewfiles bgcolor
 ++
 ++    toplevel $top
 ++    wm title $top $title
 ++    label $top.nl -text [mc "Name"]
 ++    entry $top.name -width 20 -textvariable newviewname($n)
 ++    grid $top.nl $top.name -sticky w -pady 5
 ++    checkbutton $top.perm -text [mc "Remember this view"] \
 ++     -variable newviewperm($n)
 ++    grid $top.perm - -pady 5 -sticky w
 ++    message $top.al -aspect 1000 \
 ++     -text [mc "Commits to include (arguments to git log):"]
 ++    grid $top.al - -sticky w -pady 5
 ++    entry $top.args -width 50 -textvariable newviewargs($n) \
 ++     -background $bgcolor
 ++    grid $top.args - -sticky ew -padx 5
 ++
 ++    message $top.ac -aspect 1000 \
 ++     -text [mc "Command to generate more commits to include:"]
 ++    grid $top.ac - -sticky w -pady 5
 ++    entry $top.argscmd -width 50 -textvariable newviewargscmd($n) \
 ++     -background white
 ++    grid $top.argscmd - -sticky ew -padx 5
 ++
 ++    message $top.l -aspect 1000 \
 ++     -text [mc "Enter files and directories to include, one per line:"]
 ++    grid $top.l - -sticky w
 ++    text $top.t -width 40 -height 10 -background $bgcolor -font uifont
 ++    if {[info exists viewfiles($n)]} {
 ++     foreach f $viewfiles($n) {
 ++         $top.t insert end $f
 ++         $top.t insert end "\n"
 ++     }
 ++     $top.t delete {end - 1c} end
 ++     $top.t mark set insert 0.0
 ++    }
 ++    grid $top.t - -sticky ew -padx 5
 ++    frame $top.buts
 ++    button $top.buts.ok -text [mc "OK"] -command [list newviewok $top $n]
 ++    button $top.buts.can -text [mc "Cancel"] -command [list destroy $top]
 ++    grid $top.buts.ok $top.buts.can
 ++    grid columnconfigure $top.buts 0 -weight 1 -uniform a
 ++    grid columnconfigure $top.buts 1 -weight 1 -uniform a
 ++    grid $top.buts - -pady 10 -sticky ew
 ++    focus $top.t
 ++}
 ++
 ++proc doviewmenu {m first cmd op argv} {
 ++    set nmenu [$m index end]
 ++    for {set i $first} {$i <= $nmenu} {incr i} {
 ++     if {[$m entrycget $i -command] eq $cmd} {
 ++         eval $m $op $i $argv
 ++         break
 ++     }
 ++    }
 ++}
 ++
 ++proc allviewmenus {n op args} {
 ++    # global viewhlmenu
 ++
 ++    doviewmenu .bar.view 5 [list showview $n] $op $args
 ++    # doviewmenu $viewhlmenu 1 [list addvhighlight $n] $op $args
 ++}
 ++
 ++proc newviewok {top n} {
 ++    global nextviewnum newviewperm newviewname newishighlight
 ++    global viewname viewfiles viewperm selectedview curview
 ++    global viewargs newviewargs viewargscmd newviewargscmd viewhlmenu
 ++
 ++    if {[catch {
 ++     set newargs [shellsplit $newviewargs($n)]
 ++    } err]} {
 ++     error_popup "[mc "Error in commit selection arguments:"] $err"
 ++     wm raise $top
 ++     focus $top
 ++     return
 ++    }
 ++    set files {}
 ++    foreach f [split [$top.t get 0.0 end] "\n"] {
 ++     set ft [string trim $f]
 ++     if {$ft ne {}} {
 ++         lappend files $ft
 ++     }
 ++    }
 ++    if {![info exists viewfiles($n)]} {
 ++     # creating a new view
 ++     incr nextviewnum
 ++     set viewname($n) $newviewname($n)
 ++     set viewperm($n) $newviewperm($n)
 ++     set viewfiles($n) $files
 ++     set viewargs($n) $newargs
 ++     set viewargscmd($n) $newviewargscmd($n)
 ++     addviewmenu $n
 ++     if {!$newishighlight} {
 ++         run showview $n
 ++     } else {
 ++         run addvhighlight $n
 ++     }
 ++    } else {
 ++     # editing an existing view
 ++     set viewperm($n) $newviewperm($n)
 ++     if {$newviewname($n) ne $viewname($n)} {
 ++         set viewname($n) $newviewname($n)
 ++         doviewmenu .bar.view 5 [list showview $n] \
 ++             entryconf [list -label $viewname($n)]
 ++         # doviewmenu $viewhlmenu 1 [list addvhighlight $n] \
 ++             # entryconf [list -label $viewname($n) -value $viewname($n)]
 ++     }
 ++     if {$files ne $viewfiles($n) || $newargs ne $viewargs($n) || \
 ++             $newviewargscmd($n) ne $viewargscmd($n)} {
 ++         set viewfiles($n) $files
 ++         set viewargs($n) $newargs
 ++         set viewargscmd($n) $newviewargscmd($n)
 ++         if {$curview == $n} {
 ++             run reloadcommits
 ++         }
 ++     }
 ++    }
 ++    catch {destroy $top}
 ++}
 ++
 ++proc delview {} {
 ++    global curview viewperm hlview selectedhlview
 ++
 ++    if {$curview == 0} return
 ++    if {[info exists hlview] && $hlview == $curview} {
 ++     set selectedhlview [mc "None"]
 ++     unset hlview
 ++    }
 ++    allviewmenus $curview delete
 ++    set viewperm($curview) 0
 ++    showview 0
 ++}
 ++
 ++proc addviewmenu {n} {
 ++    global viewname viewhlmenu
 ++
 ++    .bar.view add radiobutton -label $viewname($n) \
 ++     -command [list showview $n] -variable selectedview -value $n
 ++    #$viewhlmenu add radiobutton -label $viewname($n) \
 ++    #        -command [list addvhighlight $n] -variable selectedhlview
 ++}
 ++
 ++proc showview {n} {
 ++    global curview cached_commitrow ordertok
 ++    global displayorder parentlist rowidlist rowisopt rowfinal
 ++    global colormap rowtextx nextcolor canvxmax
 ++    global numcommits viewcomplete
 ++    global selectedline currentid canv canvy0
 ++    global treediffs
 ++    global pending_select mainheadid
 ++    global commitidx
 ++    global selectedview
 ++    global hlview selectedhlview commitinterest
 ++
 ++    if {$n == $curview} return
 ++    set selid {}
 ++    set ymax [lindex [$canv cget -scrollregion] 3]
 ++    set span [$canv yview]
 ++    set ytop [expr {[lindex $span 0] * $ymax}]
 ++    set ybot [expr {[lindex $span 1] * $ymax}]
 ++    set yscreen [expr {($ybot - $ytop) / 2}]
 ++    if {$selectedline ne {}} {
 ++     set selid $currentid
 ++     set y [yc $selectedline]
 ++     if {$ytop < $y && $y < $ybot} {
 ++         set yscreen [expr {$y - $ytop}]
 ++     }
 ++    } elseif {[info exists pending_select]} {
 ++     set selid $pending_select
 ++     unset pending_select
 ++    }
 ++    unselectline
 ++    normalline
 ++    catch {unset treediffs}
 ++    clear_display
 ++    if {[info exists hlview] && $hlview == $n} {
 ++     unset hlview
 ++     set selectedhlview [mc "None"]
 ++    }
 ++    catch {unset commitinterest}
 ++    catch {unset cached_commitrow}
 ++    catch {unset ordertok}
 ++
 ++    set curview $n
 ++    set selectedview $n
 ++    .bar.view entryconf [mc "Edit view..."] -state [expr {$n == 0? "disabled": "normal"}]
 ++    .bar.view entryconf [mc "Delete view"] -state [expr {$n == 0? "disabled": "normal"}]
 ++
 ++    run refill_reflist
 ++    if {![info exists viewcomplete($n)]} {
 ++     getcommits $selid
 ++     return
 ++    }
 ++
 ++    set displayorder {}
 ++    set parentlist {}
 ++    set rowidlist {}
 ++    set rowisopt {}
 ++    set rowfinal {}
 ++    set numcommits $commitidx($n)
 ++
 ++    catch {unset colormap}
 ++    catch {unset rowtextx}
 ++    set nextcolor 0
 ++    set canvxmax [$canv cget -width]
 ++    set curview $n
 ++    set row 0
 ++    setcanvscroll
 ++    set yf 0
 ++    set row {}
 ++    if {$selid ne {} && [commitinview $selid $n]} {
 ++     set row [rowofcommit $selid]
 ++     # try to get the selected row in the same position on the screen
 ++     set ymax [lindex [$canv cget -scrollregion] 3]
 ++     set ytop [expr {[yc $row] - $yscreen}]
 ++     if {$ytop < 0} {
 ++         set ytop 0
 ++     }
 ++     set yf [expr {$ytop * 1.0 / $ymax}]
 ++    }
 ++    allcanvs yview moveto $yf
 ++    drawvisible
 ++    if {$row ne {}} {
 ++     selectline $row 0
 ++    } elseif {!$viewcomplete($n)} {
 ++     reset_pending_select $selid
 ++    } else {
 ++     reset_pending_select {}
 ++
 ++     if {[commitinview $pending_select $curview]} {
 ++         selectline [rowofcommit $pending_select] 1
 ++     } else {
 ++         set row [first_real_row]
 ++         if {$row < $numcommits} {
 ++             selectline $row 0
 ++         }
 ++     }
 ++    }
 ++    if {!$viewcomplete($n)} {
 ++     if {$numcommits == 0} {
 ++         show_status [mc "Reading commits..."]
 ++     }
 ++    } elseif {$numcommits == 0} {
 ++     show_status [mc "No commits selected"]
 ++    }
 ++}
 ++
 ++# Stuff relating to the highlighting facility
 ++
 ++proc ishighlighted {id} {
 ++    global vhighlights fhighlights nhighlights rhighlights
 ++
 ++    if {[info exists nhighlights($id)] && $nhighlights($id) > 0} {
 ++     return $nhighlights($id)
 ++    }
 ++    if {[info exists vhighlights($id)] && $vhighlights($id) > 0} {
 ++     return $vhighlights($id)
 ++    }
 ++    if {[info exists fhighlights($id)] && $fhighlights($id) > 0} {
 ++     return $fhighlights($id)
 ++    }
 ++    if {[info exists rhighlights($id)] && $rhighlights($id) > 0} {
 ++     return $rhighlights($id)
 ++    }
 ++    return 0
 ++}
 ++
 ++proc bolden {row font} {
 ++    global canv linehtag selectedline boldrows
 ++
 ++    lappend boldrows $row
 ++    $canv itemconf $linehtag($row) -font $font
 ++    if {$row == $selectedline} {
 ++     $canv delete secsel
 ++     set t [eval $canv create rect [$canv bbox $linehtag($row)] \
 ++                -outline {{}} -tags secsel \
 ++                -fill [$canv cget -selectbackground]]
 ++     $canv lower $t
 ++    }
 ++}
 ++
 ++proc bolden_name {row font} {
 ++    global canv2 linentag selectedline boldnamerows
 ++
 ++    lappend boldnamerows $row
 ++    $canv2 itemconf $linentag($row) -font $font
 ++    if {$row == $selectedline} {
 ++     $canv2 delete secsel
 ++     set t [eval $canv2 create rect [$canv2 bbox $linentag($row)] \
 ++                -outline {{}} -tags secsel \
 ++                -fill [$canv2 cget -selectbackground]]
 ++     $canv2 lower $t
 ++    }
 ++}
 ++
 ++proc unbolden {} {
 ++    global boldrows
 ++
 ++    set stillbold {}
 ++    foreach row $boldrows {
 ++     if {![ishighlighted [commitonrow $row]]} {
 ++         bolden $row mainfont
 ++     } else {
 ++         lappend stillbold $row
 ++     }
 ++    }
 ++    set boldrows $stillbold
 ++}
 ++
 ++proc addvhighlight {n} {
 ++    global hlview viewcomplete curview vhl_done commitidx
 ++
 ++    if {[info exists hlview]} {
 ++     delvhighlight
 ++    }
 ++    set hlview $n
 ++    if {$n != $curview && ![info exists viewcomplete($n)]} {
 ++     start_rev_list $n
 ++    }
 ++    set vhl_done $commitidx($hlview)
 ++    if {$vhl_done > 0} {
 ++     drawvisible
 ++    }
 ++}
 ++
 ++proc delvhighlight {} {
 ++    global hlview vhighlights
 ++
 ++    if {![info exists hlview]} return
 ++    unset hlview
 ++    catch {unset vhighlights}
 ++    unbolden
 ++}
 ++
 ++proc vhighlightmore {} {
 ++    global hlview vhl_done commitidx vhighlights curview
 ++
 ++    set max $commitidx($hlview)
 ++    set vr [visiblerows]
 ++    set r0 [lindex $vr 0]
 ++    set r1 [lindex $vr 1]
 ++    for {set i $vhl_done} {$i < $max} {incr i} {
 ++     set id [commitonrow $i $hlview]
 ++     if {[commitinview $id $curview]} {
 ++         set row [rowofcommit $id]
 ++         if {$r0 <= $row && $row <= $r1} {
 ++             if {![highlighted $row]} {
 ++                 bolden $row mainfontbold
 ++             }
 ++             set vhighlights($id) 1
 ++         }
 ++     }
 ++    }
 ++    set vhl_done $max
 ++    return 0
 ++}
 ++
 ++proc askvhighlight {row id} {
 ++    global hlview vhighlights iddrawn
 ++
 ++    if {[commitinview $id $hlview]} {
 ++     if {[info exists iddrawn($id)] && ![ishighlighted $id]} {
 ++         bolden $row mainfontbold
 ++     }
 ++     set vhighlights($id) 1
 ++    } else {
 ++     set vhighlights($id) 0
 ++    }
 ++}
 ++
 ++proc hfiles_change {} {
 ++    global highlight_files filehighlight fhighlights fh_serial
 ++    global highlight_paths gdttype
 ++
 ++    if {[info exists filehighlight]} {
 ++     # delete previous highlights
 ++     catch {close $filehighlight}
 ++     unset filehighlight
 ++     catch {unset fhighlights}
 ++     unbolden
 ++     unhighlight_filelist
 ++    }
 ++    set highlight_paths {}
 ++    after cancel do_file_hl $fh_serial
 ++    incr fh_serial
 ++    if {$highlight_files ne {}} {
 ++     after 300 do_file_hl $fh_serial
 ++    }
 ++}
 ++
 ++proc gdttype_change {name ix op} {
 ++    global gdttype highlight_files findstring findpattern
 ++
 ++    stopfinding
 ++    if {$findstring ne {}} {
 ++     if {$gdttype eq [mc "containing:"]} {
 ++         if {$highlight_files ne {}} {
 ++             set highlight_files {}
 ++             hfiles_change
 ++         }
 ++         findcom_change
 ++     } else {
 ++         if {$findpattern ne {}} {
 ++             set findpattern {}
 ++             findcom_change
 ++         }
 ++         set highlight_files $findstring
 ++         hfiles_change
 ++     }
 ++     drawvisible
 ++    }
 ++    # enable/disable findtype/findloc menus too
 ++}
 ++
 ++proc find_change {name ix op} {
 ++    global gdttype findstring highlight_files
 ++
 ++    stopfinding
 ++    if {$gdttype eq [mc "containing:"]} {
 ++     findcom_change
 ++    } else {
 ++     if {$highlight_files ne $findstring} {
 ++         set highlight_files $findstring
 ++         hfiles_change
 ++     }
 ++    }
 ++    drawvisible
 ++}
 ++
 ++proc findcom_change args {
 ++    global nhighlights boldnamerows
 ++    global findpattern findtype findstring gdttype
 ++
 ++    stopfinding
 ++    # delete previous highlights, if any
 ++    foreach row $boldnamerows {
 ++     bolden_name $row mainfont
 ++    }
 ++    set boldnamerows {}
 ++    catch {unset nhighlights}
 ++    unbolden
 ++    unmarkmatches
 ++    if {$gdttype ne [mc "containing:"] || $findstring eq {}} {
 ++     set findpattern {}
 ++    } elseif {$findtype eq [mc "Regexp"]} {
 ++     set findpattern $findstring
 ++    } else {
 ++     set e [string map {"*" "\\*" "?" "\\?" "\[" "\\\[" "\\" "\\\\"} \
 ++                $findstring]
 ++     set findpattern "*$e*"
 ++    }
 ++}
 ++
 ++proc makepatterns {l} {
 ++    set ret {}
 ++    foreach e $l {
 ++     set ee [string map {"*" "\\*" "?" "\\?" "\[" "\\\[" "\\" "\\\\"} $e]
 ++     if {[string index $ee end] eq "/"} {
 ++         lappend ret "$ee*"
 ++     } else {
 ++         lappend ret $ee
 ++         lappend ret "$ee/*"
 ++     }
 ++    }
 ++    return $ret
 ++}
 ++
 ++proc do_file_hl {serial} {
 ++    global highlight_files filehighlight highlight_paths gdttype fhl_list
 ++
 ++    if {$gdttype eq [mc "touching paths:"]} {
 ++     if {[catch {set paths [shellsplit $highlight_files]}]} return
 ++     set highlight_paths [makepatterns $paths]
 ++     highlight_filelist
 ++     set gdtargs [concat -- $paths]
 ++    } elseif {$gdttype eq [mc "adding/removing string:"]} {
 ++     set gdtargs [list "-S$highlight_files"]
 ++    } else {
 ++     # must be "containing:", i.e. we're searching commit info
 ++     return
 ++    }
 ++    set cmd [concat | git diff-tree -r -s --stdin $gdtargs]
 ++    set filehighlight [open $cmd r+]
 ++    fconfigure $filehighlight -blocking 0
 ++    filerun $filehighlight readfhighlight
 ++    set fhl_list {}
 ++    drawvisible
 ++    flushhighlights
 ++}
 ++
 ++proc flushhighlights {} {
 ++    global filehighlight fhl_list
 ++
 ++    if {[info exists filehighlight]} {
 ++     lappend fhl_list {}
 ++     puts $filehighlight ""
 ++     flush $filehighlight
 ++    }
 ++}
 ++
 ++proc askfilehighlight {row id} {
 ++    global filehighlight fhighlights fhl_list
 ++
 ++    lappend fhl_list $id
 ++    set fhighlights($id) -1
 ++    puts $filehighlight $id
 ++}
 ++
 ++proc readfhighlight {} {
 ++    global filehighlight fhighlights curview iddrawn
 ++    global fhl_list find_dirn
 ++
 ++    if {![info exists filehighlight]} {
 ++     return 0
 ++    }
 ++    set nr 0
 ++    while {[incr nr] <= 100 && [gets $filehighlight line] >= 0} {
 ++     set line [string trim $line]
 ++     set i [lsearch -exact $fhl_list $line]
 ++     if {$i < 0} continue
 ++     for {set j 0} {$j < $i} {incr j} {
 ++         set id [lindex $fhl_list $j]
 ++         set fhighlights($id) 0
 ++     }
 ++     set fhl_list [lrange $fhl_list [expr {$i+1}] end]
 ++     if {$line eq {}} continue
 ++     if {![commitinview $line $curview]} continue
 ++     set row [rowofcommit $line]
 ++     if {[info exists iddrawn($line)] && ![ishighlighted $line]} {
 ++         bolden $row mainfontbold
 ++     }
 ++     set fhighlights($line) 1
 ++    }
 ++    if {[eof $filehighlight]} {
 ++     # strange...
 ++     puts "oops, git diff-tree died"
 ++     catch {close $filehighlight}
 ++     unset filehighlight
 ++     return 0
 ++    }
 ++    if {[info exists find_dirn]} {
 ++     run findmore
 ++    }
 ++    return 1
 ++}
 ++
 ++proc doesmatch {f} {
 ++    global findtype findpattern
 ++
 ++    if {$findtype eq [mc "Regexp"]} {
 ++     return [regexp $findpattern $f]
 ++    } elseif {$findtype eq [mc "IgnCase"]} {
 ++     return [string match -nocase $findpattern $f]
 ++    } else {
 ++     return [string match $findpattern $f]
 ++    }
 ++}
 ++
 ++proc askfindhighlight {row id} {
 ++    global nhighlights commitinfo iddrawn
 ++    global findloc
 ++    global markingmatches
 ++
 ++    if {![info exists commitinfo($id)]} {
 ++     getcommit $id
 ++    }
 ++    set info $commitinfo($id)
 ++    set isbold 0
 ++    set fldtypes [list [mc Headline] [mc Author] [mc Date] [mc Committer] [mc CDate] [mc Comments]]
 ++    foreach f $info ty $fldtypes {
 ++     if {($findloc eq [mc "All fields"] || $findloc eq $ty) &&
 ++         [doesmatch $f]} {
 ++         if {$ty eq [mc "Author"]} {
 ++             set isbold 2
 ++             break
 ++         }
 ++         set isbold 1
 ++     }
 ++    }
 ++    if {$isbold && [info exists iddrawn($id)]} {
 ++     if {![ishighlighted $id]} {
 ++         bolden $row mainfontbold
 ++         if {$isbold > 1} {
 ++             bolden_name $row mainfontbold
 ++         }
 ++     }
 ++     if {$markingmatches} {
 ++         markrowmatches $row $id
 ++     }
 ++    }
 ++    set nhighlights($id) $isbold
 ++}
 ++
 ++proc markrowmatches {row id} {
 ++    global canv canv2 linehtag linentag commitinfo findloc
 ++
 ++    set headline [lindex $commitinfo($id) 0]
 ++    set author [lindex $commitinfo($id) 1]
 ++    $canv delete match$row
 ++    $canv2 delete match$row
 ++    if {$findloc eq [mc "All fields"] || $findloc eq [mc "Headline"]} {
 ++     set m [findmatches $headline]
 ++     if {$m ne {}} {
 ++         markmatches $canv $row $headline $linehtag($row) $m \
 ++             [$canv itemcget $linehtag($row) -font] $row
 ++     }
 ++    }
 ++    if {$findloc eq [mc "All fields"] || $findloc eq [mc "Author"]} {
 ++     set m [findmatches $author]
 ++     if {$m ne {}} {
 ++         markmatches $canv2 $row $author $linentag($row) $m \
 ++             [$canv2 itemcget $linentag($row) -font] $row
 ++     }
 ++    }
 ++}
 ++
 ++proc vrel_change {name ix op} {
 ++    global highlight_related
 ++
 ++    rhighlight_none
 ++    if {$highlight_related ne [mc "None"]} {
 ++     run drawvisible
 ++    }
 ++}
 ++
 ++# prepare for testing whether commits are descendents or ancestors of a
 ++proc rhighlight_sel {a} {
 ++    global descendent desc_todo ancestor anc_todo
 ++    global highlight_related
 ++
 ++    catch {unset descendent}
 ++    set desc_todo [list $a]
 ++    catch {unset ancestor}
 ++    set anc_todo [list $a]
 ++    if {$highlight_related ne [mc "None"]} {
 ++     rhighlight_none
 ++     run drawvisible
 ++    }
 ++}
 ++
 ++proc rhighlight_none {} {
 ++    global rhighlights
 ++
 ++    catch {unset rhighlights}
 ++    unbolden
 ++}
 ++
 ++proc is_descendent {a} {
 ++    global curview children descendent desc_todo
 ++
 ++    set v $curview
 ++    set la [rowofcommit $a]
 ++    set todo $desc_todo
 ++    set leftover {}
 ++    set done 0
 ++    for {set i 0} {$i < [llength $todo]} {incr i} {
 ++     set do [lindex $todo $i]
 ++     if {[rowofcommit $do] < $la} {
 ++         lappend leftover $do
 ++         continue
 ++     }
 ++     foreach nk $children($v,$do) {
 ++         if {![info exists descendent($nk)]} {
 ++             set descendent($nk) 1
 ++             lappend todo $nk
 ++             if {$nk eq $a} {
 ++                 set done 1
 ++             }
 ++         }
 ++     }
 ++     if {$done} {
 ++         set desc_todo [concat $leftover [lrange $todo [expr {$i+1}] end]]
 ++         return
 ++     }
 ++    }
 ++    set descendent($a) 0
 ++    set desc_todo $leftover
 ++}
 ++
 ++proc is_ancestor {a} {
 ++    global curview parents ancestor anc_todo
 ++
 ++    set v $curview
 ++    set la [rowofcommit $a]
 ++    set todo $anc_todo
 ++    set leftover {}
 ++    set done 0
 ++    for {set i 0} {$i < [llength $todo]} {incr i} {
 ++     set do [lindex $todo $i]
 ++     if {![commitinview $do $v] || [rowofcommit $do] > $la} {
 ++         lappend leftover $do
 ++         continue
 ++     }
 ++     foreach np $parents($v,$do) {
 ++         if {![info exists ancestor($np)]} {
 ++             set ancestor($np) 1
 ++             lappend todo $np
 ++             if {$np eq $a} {
 ++                 set done 1
 ++             }
 ++         }
 ++     }
 ++     if {$done} {
 ++         set anc_todo [concat $leftover [lrange $todo [expr {$i+1}] end]]
 ++         return
 ++     }
 ++    }
 ++    set ancestor($a) 0
 ++    set anc_todo $leftover
 ++}
 ++
 ++proc askrelhighlight {row id} {
 ++    global descendent highlight_related iddrawn rhighlights
 ++    global selectedline ancestor
 ++
 ++    if {$selectedline eq {}} return
 ++    set isbold 0
 ++    if {$highlight_related eq [mc "Descendant"] ||
 ++     $highlight_related eq [mc "Not descendant"]} {
 ++     if {![info exists descendent($id)]} {
 ++         is_descendent $id
 ++     }
 ++     if {$descendent($id) == ($highlight_related eq [mc "Descendant"])} {
 ++         set isbold 1
 ++     }
 ++    } elseif {$highlight_related eq [mc "Ancestor"] ||
 ++           $highlight_related eq [mc "Not ancestor"]} {
 ++     if {![info exists ancestor($id)]} {
 ++         is_ancestor $id
 ++     }
 ++     if {$ancestor($id) == ($highlight_related eq [mc "Ancestor"])} {
 ++         set isbold 1
 ++     }
 ++    }
 ++    if {[info exists iddrawn($id)]} {
 ++     if {$isbold && ![ishighlighted $id]} {
 ++         bolden $row mainfontbold
 ++     }
 ++    }
 ++    set rhighlights($id) $isbold
 ++}
 ++
 ++# Graph layout functions
 ++
 ++proc shortids {ids} {
 ++    set res {}
 ++    foreach id $ids {
 ++     if {[llength $id] > 1} {
 ++         lappend res [shortids $id]
 ++     } elseif {[regexp {^[0-9a-f]{40}$} $id]} {
 ++         lappend res [string range $id 0 7]
 ++     } else {
 ++         lappend res $id
 ++     }
 ++    }
 ++    return $res
 ++}
 ++
 ++proc ntimes {n o} {
 ++    set ret {}
 ++    set o [list $o]
 ++    for {set mask 1} {$mask <= $n} {incr mask $mask} {
 ++     if {($n & $mask) != 0} {
 ++         set ret [concat $ret $o]
 ++     }
 ++     set o [concat $o $o]
 ++    }
 ++    return $ret
 ++}
 ++
 ++proc ordertoken {id} {
 ++    global ordertok curview varcid varcstart varctok curview parents children
 ++    global nullid nullid2
 ++
 ++    if {[info exists ordertok($id)]} {
 ++     return $ordertok($id)
 ++    }
 ++    set origid $id
 ++    set todo {}
 ++    while {1} {
 ++     if {[info exists varcid($curview,$id)]} {
 ++         set a $varcid($curview,$id)
 ++         set p [lindex $varcstart($curview) $a]
 ++     } else {
 ++         set p [lindex $children($curview,$id) 0]
 ++     }
 ++     if {[info exists ordertok($p)]} {
 ++         set tok $ordertok($p)
 ++         break
 ++     }
 ++     set id [first_real_child $curview,$p]
 ++     if {$id eq {}} {
 ++         # it's a root
 ++         set tok [lindex $varctok($curview) $varcid($curview,$p)]
 ++         break
 ++     }
 ++     if {[llength $parents($curview,$id)] == 1} {
 ++         lappend todo [list $p {}]
 ++     } else {
 ++         set j [lsearch -exact $parents($curview,$id) $p]
 ++         if {$j < 0} {
 ++             puts "oops didn't find [shortids $p] in parents of [shortids $id]"
 ++         }
 ++         lappend todo [list $p [strrep $j]]
 ++     }
 ++    }
 ++    for {set i [llength $todo]} {[incr i -1] >= 0} {} {
 ++     set p [lindex $todo $i 0]
 ++     append tok [lindex $todo $i 1]
 ++     set ordertok($p) $tok
 ++    }
 ++    set ordertok($origid) $tok
 ++    return $tok
 ++}
 ++
 ++# Work out where id should go in idlist so that order-token
 ++# values increase from left to right
 ++proc idcol {idlist id {i 0}} {
 ++    set t [ordertoken $id]
 ++    if {$i < 0} {
 ++     set i 0
 ++    }
 ++    if {$i >= [llength $idlist] || $t < [ordertoken [lindex $idlist $i]]} {
 ++     if {$i > [llength $idlist]} {
 ++         set i [llength $idlist]
 ++     }
 ++     while {[incr i -1] >= 0 && $t < [ordertoken [lindex $idlist $i]]} {}
 ++     incr i
 ++    } else {
 ++     if {$t > [ordertoken [lindex $idlist $i]]} {
 ++         while {[incr i] < [llength $idlist] &&
 ++                $t >= [ordertoken [lindex $idlist $i]]} {}
 ++     }
 ++    }
 ++    return $i
 ++}
 ++
 ++proc initlayout {} {
 ++    global rowidlist rowisopt rowfinal displayorder parentlist
 ++    global numcommits canvxmax canv
 ++    global nextcolor
 ++    global colormap rowtextx
 ++
 ++    set numcommits 0
 ++    set displayorder {}
 ++    set parentlist {}
 ++    set nextcolor 0
 ++    set rowidlist {}
 ++    set rowisopt {}
 ++    set rowfinal {}
 ++    set canvxmax [$canv cget -width]
 ++    catch {unset colormap}
 ++    catch {unset rowtextx}
 ++    setcanvscroll
 ++}
 ++
 ++proc setcanvscroll {} {
 ++    global canv canv2 canv3 numcommits linespc canvxmax canvy0
 ++    global lastscrollset lastscrollrows
 ++
 ++    set ymax [expr {$canvy0 + ($numcommits - 0.5) * $linespc + 2}]
 ++    $canv conf -scrollregion [list 0 0 $canvxmax $ymax]
 ++    $canv2 conf -scrollregion [list 0 0 0 $ymax]
 ++    $canv3 conf -scrollregion [list 0 0 0 $ymax]
 ++    set lastscrollset [clock clicks -milliseconds]
 ++    set lastscrollrows $numcommits
 ++}
 ++
 ++proc visiblerows {} {
 ++    global canv numcommits linespc
 ++
 ++    set ymax [lindex [$canv cget -scrollregion] 3]
 ++    if {$ymax eq {} || $ymax == 0} return
 ++    set f [$canv yview]
 ++    set y0 [expr {int([lindex $f 0] * $ymax)}]
 ++    set r0 [expr {int(($y0 - 3) / $linespc) - 1}]
 ++    if {$r0 < 0} {
 ++     set r0 0
 ++    }
 ++    set y1 [expr {int([lindex $f 1] * $ymax)}]
 ++    set r1 [expr {int(($y1 - 3) / $linespc) + 1}]
 ++    if {$r1 >= $numcommits} {
 ++     set r1 [expr {$numcommits - 1}]
 ++    }
 ++    return [list $r0 $r1]
 ++}
 ++
 ++proc layoutmore {} {
 ++    global commitidx viewcomplete curview
 ++    global numcommits pending_select curview
 ++    global lastscrollset lastscrollrows commitinterest
 ++
 ++    if {$lastscrollrows < 100 || $viewcomplete($curview) ||
 ++     [clock clicks -milliseconds] - $lastscrollset > 500} {
 ++     setcanvscroll
 ++    }
 ++    if {[info exists pending_select] &&
 ++     [commitinview $pending_select $curview]} {
 ++     update
 ++     selectline [rowofcommit $pending_select] 1
 ++    }
 ++    drawvisible
 ++}
 ++
 ++proc doshowlocalchanges {} {
 ++    global curview mainheadid
 ++
 ++    if {$mainheadid eq {}} return
 ++    if {[commitinview $mainheadid $curview]} {
 ++     dodiffindex
 ++    } else {
 ++     lappend commitinterest($mainheadid) {dodiffindex}
 ++    }
 ++}
 ++
 ++proc dohidelocalchanges {} {
 ++    global nullid nullid2 lserial curview
 ++
 ++    if {[commitinview $nullid $curview]} {
 ++     removefakerow $nullid
 ++    }
 ++    if {[commitinview $nullid2 $curview]} {
 ++     removefakerow $nullid2
 ++    }
 ++    incr lserial
 ++}
 ++
 ++# spawn off a process to do git diff-index --cached HEAD
 ++proc dodiffindex {} {
 ++    global lserial showlocalchanges
 ++    global isworktree
 ++
 ++    if {!$showlocalchanges || !$isworktree} return
 ++    incr lserial
 ++    set fd [open "|git diff-index --cached HEAD" r]
 ++    fconfigure $fd -blocking 0
 ++    set i [reg_instance $fd]
 ++    filerun $fd [list readdiffindex $fd $lserial $i]
 ++}
 ++
 ++proc readdiffindex {fd serial inst} {
 ++    global mainheadid nullid nullid2 curview commitinfo commitdata lserial
 ++
 ++    set isdiff 1
 ++    if {[gets $fd line] < 0} {
 ++     if {![eof $fd]} {
 ++         return 1
 ++     }
 ++     set isdiff 0
 ++    }
 ++    # we only need to see one line and we don't really care what it says...
 ++    stop_instance $inst
 ++
 ++    if {$serial != $lserial} {
 ++     return 0
 ++    }
 ++
 ++    # now see if there are any local changes not checked in to the index
 ++    set fd [open "|git diff-files" r]
 ++    fconfigure $fd -blocking 0
 ++    set i [reg_instance $fd]
 ++    filerun $fd [list readdifffiles $fd $serial $i]
 ++
 ++    if {$isdiff && ![commitinview $nullid2 $curview]} {
 ++     # add the line for the changes in the index to the graph
 ++     set hl [mc "Local changes checked in to index but not committed"]
 ++     set commitinfo($nullid2) [list  $hl {} {} {} {} "    $hl\n"]
 ++     set commitdata($nullid2) "\n    $hl\n"
 ++     if {[commitinview $nullid $curview]} {
 ++         removefakerow $nullid
 ++     }
 ++     insertfakerow $nullid2 $mainheadid
 ++    } elseif {!$isdiff && [commitinview $nullid2 $curview]} {
 ++     removefakerow $nullid2
 ++    }
 ++    return 0
 ++}
 ++
 ++proc readdifffiles {fd serial inst} {
 ++    global mainheadid nullid nullid2 curview
 ++    global commitinfo commitdata lserial
 ++
 ++    set isdiff 1
 ++    if {[gets $fd line] < 0} {
 ++     if {![eof $fd]} {
 ++         return 1
 ++     }
 ++     set isdiff 0
 ++    }
 ++    # we only need to see one line and we don't really care what it says...
 ++    stop_instance $inst
 ++
 ++    if {$serial != $lserial} {
 ++     return 0
 ++    }
 ++
 ++    if {$isdiff && ![commitinview $nullid $curview]} {
 ++     # add the line for the local diff to the graph
 ++     set hl [mc "Local uncommitted changes, not checked in to index"]
 ++     set commitinfo($nullid) [list  $hl {} {} {} {} "    $hl\n"]
 ++     set commitdata($nullid) "\n    $hl\n"
 ++     if {[commitinview $nullid2 $curview]} {
 ++         set p $nullid2
 ++     } else {
 ++         set p $mainheadid
 ++     }
 ++     insertfakerow $nullid $p
 ++    } elseif {!$isdiff && [commitinview $nullid $curview]} {
 ++     removefakerow $nullid
 ++    }
 ++    return 0
 ++}
 ++
 ++proc nextuse {id row} {
 ++    global curview children
 ++
 ++    if {[info exists children($curview,$id)]} {
 ++     foreach kid $children($curview,$id) {
 ++         if {![commitinview $kid $curview]} {
 ++             return -1
 ++         }
 ++         if {[rowofcommit $kid] > $row} {
 ++             return [rowofcommit $kid]
 ++         }
 ++     }
 ++    }
 ++    if {[commitinview $id $curview]} {
 ++     return [rowofcommit $id]
 ++    }
 ++    return -1
 ++}
 ++
 ++proc prevuse {id row} {
 ++    global curview children
 ++
 ++    set ret -1
 ++    if {[info exists children($curview,$id)]} {
 ++     foreach kid $children($curview,$id) {
 ++         if {![commitinview $kid $curview]} break
 ++         if {[rowofcommit $kid] < $row} {
 ++             set ret [rowofcommit $kid]
 ++         }
 ++     }
 ++    }
 ++    return $ret
 ++}
 ++
 ++proc make_idlist {row} {
 ++    global displayorder parentlist uparrowlen downarrowlen mingaplen
 ++    global commitidx curview children
 ++
 ++    set r [expr {$row - $mingaplen - $downarrowlen - 1}]
 ++    if {$r < 0} {
 ++     set r 0
 ++    }
 ++    set ra [expr {$row - $downarrowlen}]
 ++    if {$ra < 0} {
 ++     set ra 0
 ++    }
 ++    set rb [expr {$row + $uparrowlen}]
 ++    if {$rb > $commitidx($curview)} {
 ++     set rb $commitidx($curview)
 ++    }
 ++    make_disporder $r [expr {$rb + 1}]
 ++    set ids {}
 ++    for {} {$r < $ra} {incr r} {
 ++     set nextid [lindex $displayorder [expr {$r + 1}]]
 ++     foreach p [lindex $parentlist $r] {
 ++         if {$p eq $nextid} continue
 ++         set rn [nextuse $p $r]
 ++         if {$rn >= $row &&
 ++             $rn <= $r + $downarrowlen + $mingaplen + $uparrowlen} {
 ++             lappend ids [list [ordertoken $p] $p]
 ++         }
 ++     }
 ++    }
 ++    for {} {$r < $row} {incr r} {
 ++     set nextid [lindex $displayorder [expr {$r + 1}]]
 ++     foreach p [lindex $parentlist $r] {
 ++         if {$p eq $nextid} continue
 ++         set rn [nextuse $p $r]
 ++         if {$rn < 0 || $rn >= $row} {
 ++             lappend ids [list [ordertoken $p] $p]
 ++         }
 ++     }
 ++    }
 ++    set id [lindex $displayorder $row]
 ++    lappend ids [list [ordertoken $id] $id]
 ++    while {$r < $rb} {
 ++     foreach p [lindex $parentlist $r] {
 ++         set firstkid [lindex $children($curview,$p) 0]
 ++         if {[rowofcommit $firstkid] < $row} {
 ++             lappend ids [list [ordertoken $p] $p]
 ++         }
 ++     }
 ++     incr r
 ++     set id [lindex $displayorder $r]
 ++     if {$id ne {}} {
 ++         set firstkid [lindex $children($curview,$id) 0]
 ++         if {$firstkid ne {} && [rowofcommit $firstkid] < $row} {
 ++             lappend ids [list [ordertoken $id] $id]
 ++         }
 ++     }
 ++    }
 ++    set idlist {}
 ++    foreach idx [lsort -unique $ids] {
 ++     lappend idlist [lindex $idx 1]
 ++    }
 ++    return $idlist
 ++}
 ++
 ++proc rowsequal {a b} {
 ++    while {[set i [lsearch -exact $a {}]] >= 0} {
 ++     set a [lreplace $a $i $i]
 ++    }
 ++    while {[set i [lsearch -exact $b {}]] >= 0} {
 ++     set b [lreplace $b $i $i]
 ++    }
 ++    return [expr {$a eq $b}]
 ++}
 ++
 ++proc makeupline {id row rend col} {
 ++    global rowidlist uparrowlen downarrowlen mingaplen
 ++
 ++    for {set r $rend} {1} {set r $rstart} {
 ++     set rstart [prevuse $id $r]
 ++     if {$rstart < 0} return
 ++     if {$rstart < $row} break
 ++    }
 ++    if {$rstart + $uparrowlen + $mingaplen + $downarrowlen < $rend} {
 ++     set rstart [expr {$rend - $uparrowlen - 1}]
 ++    }
 ++    for {set r $rstart} {[incr r] <= $row} {} {
 ++     set idlist [lindex $rowidlist $r]
 ++     if {$idlist ne {} && [lsearch -exact $idlist $id] < 0} {
 ++         set col [idcol $idlist $id $col]
 ++         lset rowidlist $r [linsert $idlist $col $id]
 ++         changedrow $r
 ++     }
 ++    }
 ++}
 ++
 ++proc layoutrows {row endrow} {
 ++    global rowidlist rowisopt rowfinal displayorder
 ++    global uparrowlen downarrowlen maxwidth mingaplen
 ++    global children parentlist
 ++    global commitidx viewcomplete curview
 ++
 ++    make_disporder [expr {$row - 1}] [expr {$endrow + $uparrowlen}]
 ++    set idlist {}
 ++    if {$row > 0} {
 ++     set rm1 [expr {$row - 1}]
 ++     foreach id [lindex $rowidlist $rm1] {
 ++         if {$id ne {}} {
 ++             lappend idlist $id
 ++         }
 ++     }
 ++     set final [lindex $rowfinal $rm1]
 ++    }
 ++    for {} {$row < $endrow} {incr row} {
 ++     set rm1 [expr {$row - 1}]
 ++     if {$rm1 < 0 || $idlist eq {}} {
 ++         set idlist [make_idlist $row]
 ++         set final 1
 ++     } else {
 ++         set id [lindex $displayorder $rm1]
 ++         set col [lsearch -exact $idlist $id]
 ++         set idlist [lreplace $idlist $col $col]
 ++         foreach p [lindex $parentlist $rm1] {
 ++             if {[lsearch -exact $idlist $p] < 0} {
 ++                 set col [idcol $idlist $p $col]
 ++                 set idlist [linsert $idlist $col $p]
 ++                 # if not the first child, we have to insert a line going up
 ++                 if {$id ne [lindex $children($curview,$p) 0]} {
 ++                     makeupline $p $rm1 $row $col
 ++                 }
 ++             }
 ++         }
 ++         set id [lindex $displayorder $row]
 ++         if {$row > $downarrowlen} {
 ++             set termrow [expr {$row - $downarrowlen - 1}]
 ++             foreach p [lindex $parentlist $termrow] {
 ++                 set i [lsearch -exact $idlist $p]
 ++                 if {$i < 0} continue
 ++                 set nr [nextuse $p $termrow]
 ++                 if {$nr < 0 || $nr >= $row + $mingaplen + $uparrowlen} {
 ++                     set idlist [lreplace $idlist $i $i]
 ++                 }
 ++             }
 ++         }
 ++         set col [lsearch -exact $idlist $id]
 ++         if {$col < 0} {
 ++             set col [idcol $idlist $id]
 ++             set idlist [linsert $idlist $col $id]
 ++             if {$children($curview,$id) ne {}} {
 ++                 makeupline $id $rm1 $row $col
 ++             }
 ++         }
 ++         set r [expr {$row + $uparrowlen - 1}]
 ++         if {$r < $commitidx($curview)} {
 ++             set x $col
 ++             foreach p [lindex $parentlist $r] {
 ++                 if {[lsearch -exact $idlist $p] >= 0} continue
 ++                 set fk [lindex $children($curview,$p) 0]
 ++                 if {[rowofcommit $fk] < $row} {
 ++                     set x [idcol $idlist $p $x]
 ++                     set idlist [linsert $idlist $x $p]
 ++                 }
 ++             }
 ++             if {[incr r] < $commitidx($curview)} {
 ++                 set p [lindex $displayorder $r]
 ++                 if {[lsearch -exact $idlist $p] < 0} {
 ++                     set fk [lindex $children($curview,$p) 0]
 ++                     if {$fk ne {} && [rowofcommit $fk] < $row} {
 ++                         set x [idcol $idlist $p $x]
 ++                         set idlist [linsert $idlist $x $p]
 ++                     }
 ++                 }
 ++             }
 ++         }
 ++     }
 ++     if {$final && !$viewcomplete($curview) &&
 ++         $row + $uparrowlen + $mingaplen + $downarrowlen
 ++             >= $commitidx($curview)} {
 ++         set final 0
 ++     }
 ++     set l [llength $rowidlist]
 ++     if {$row == $l} {
 ++         lappend rowidlist $idlist
 ++         lappend rowisopt 0
 ++         lappend rowfinal $final
 ++     } elseif {$row < $l} {
 ++         if {![rowsequal $idlist [lindex $rowidlist $row]]} {
 ++             lset rowidlist $row $idlist
 ++             changedrow $row
 ++         }
 ++         lset rowfinal $row $final
 ++     } else {
 ++         set pad [ntimes [expr {$row - $l}] {}]
 ++         set rowidlist [concat $rowidlist $pad]
 ++         lappend rowidlist $idlist
 ++         set rowfinal [concat $rowfinal $pad]
 ++         lappend rowfinal $final
 ++         set rowisopt [concat $rowisopt [ntimes [expr {$row - $l + 1}] 0]]
 ++     }
 ++    }
 ++    return $row
 ++}
 ++
 ++proc changedrow {row} {
 ++    global displayorder iddrawn rowisopt need_redisplay
 ++
 ++    set l [llength $rowisopt]
 ++    if {$row < $l} {
 ++     lset rowisopt $row 0
 ++     if {$row + 1 < $l} {
 ++         lset rowisopt [expr {$row + 1}] 0
 ++         if {$row + 2 < $l} {
 ++             lset rowisopt [expr {$row + 2}] 0
 ++         }
 ++     }
 ++    }
 ++    set id [lindex $displayorder $row]
 ++    if {[info exists iddrawn($id)]} {
 ++     set need_redisplay 1
 ++    }
 ++}
 ++
 ++proc insert_pad {row col npad} {
 ++    global rowidlist
 ++
 ++    set pad [ntimes $npad {}]
 ++    set idlist [lindex $rowidlist $row]
 ++    set bef [lrange $idlist 0 [expr {$col - 1}]]
 ++    set aft [lrange $idlist $col end]
 ++    set i [lsearch -exact $aft {}]
 ++    if {$i > 0} {
 ++     set aft [lreplace $aft $i $i]
 ++    }
 ++    lset rowidlist $row [concat $bef $pad $aft]
 ++    changedrow $row
 ++}
 ++
 ++proc optimize_rows {row col endrow} {
 ++    global rowidlist rowisopt displayorder curview children
 ++
 ++    if {$row < 1} {
 ++     set row 1
 ++    }
 ++    for {} {$row < $endrow} {incr row; set col 0} {
 ++     if {[lindex $rowisopt $row]} continue
 ++     set haspad 0
 ++     set y0 [expr {$row - 1}]
 ++     set ym [expr {$row - 2}]
 ++     set idlist [lindex $rowidlist $row]
 ++     set previdlist [lindex $rowidlist $y0]
 ++     if {$idlist eq {} || $previdlist eq {}} continue
 ++     if {$ym >= 0} {
 ++         set pprevidlist [lindex $rowidlist $ym]
 ++         if {$pprevidlist eq {}} continue
 ++     } else {
 ++         set pprevidlist {}
 ++     }
 ++     set x0 -1
 ++     set xm -1
 ++     for {} {$col < [llength $idlist]} {incr col} {
 ++         set id [lindex $idlist $col]
 ++         if {[lindex $previdlist $col] eq $id} continue
 ++         if {$id eq {}} {
 ++             set haspad 1
 ++             continue
 ++         }
 ++         set x0 [lsearch -exact $previdlist $id]
 ++         if {$x0 < 0} continue
 ++         set z [expr {$x0 - $col}]
 ++         set isarrow 0
 ++         set z0 {}
 ++         if {$ym >= 0} {
 ++             set xm [lsearch -exact $pprevidlist $id]
 ++             if {$xm >= 0} {
 ++                 set z0 [expr {$xm - $x0}]
 ++             }
 ++         }
 ++         if {$z0 eq {}} {
 ++             # if row y0 is the first child of $id then it's not an arrow
 ++             if {[lindex $children($curview,$id) 0] ne
 ++                 [lindex $displayorder $y0]} {
 ++                 set isarrow 1
 ++             }
 ++         }
 ++         if {!$isarrow && $id ne [lindex $displayorder $row] &&
 ++             [lsearch -exact [lindex $rowidlist [expr {$row+1}]] $id] < 0} {
 ++             set isarrow 1
 ++         }
 ++         # Looking at lines from this row to the previous row,
 ++         # make them go straight up if they end in an arrow on
 ++         # the previous row; otherwise make them go straight up
 ++         # or at 45 degrees.
 ++         if {$z < -1 || ($z < 0 && $isarrow)} {
 ++             # Line currently goes left too much;
 ++             # insert pads in the previous row, then optimize it
 ++             set npad [expr {-1 - $z + $isarrow}]
 ++             insert_pad $y0 $x0 $npad
 ++             if {$y0 > 0} {
 ++                 optimize_rows $y0 $x0 $row
 ++             }
 ++             set previdlist [lindex $rowidlist $y0]
 ++             set x0 [lsearch -exact $previdlist $id]
 ++             set z [expr {$x0 - $col}]
 ++             if {$z0 ne {}} {
 ++                 set pprevidlist [lindex $rowidlist $ym]
 ++                 set xm [lsearch -exact $pprevidlist $id]
 ++                 set z0 [expr {$xm - $x0}]
 ++             }
 ++         } elseif {$z > 1 || ($z > 0 && $isarrow)} {
 ++             # Line currently goes right too much;
 ++             # insert pads in this line
 ++             set npad [expr {$z - 1 + $isarrow}]
 ++             insert_pad $row $col $npad
 ++             set idlist [lindex $rowidlist $row]
 ++             incr col $npad
 ++             set z [expr {$x0 - $col}]
 ++             set haspad 1
 ++         }
 ++         if {$z0 eq {} && !$isarrow && $ym >= 0} {
 ++             # this line links to its first child on row $row-2
 ++             set id [lindex $displayorder $ym]
 ++             set xc [lsearch -exact $pprevidlist $id]
 ++             if {$xc >= 0} {
 ++                 set z0 [expr {$xc - $x0}]
 ++             }
 ++         }
 ++         # avoid lines jigging left then immediately right
 ++         if {$z0 ne {} && $z < 0 && $z0 > 0} {
 ++             insert_pad $y0 $x0 1
 ++             incr x0
 ++             optimize_rows $y0 $x0 $row
 ++             set previdlist [lindex $rowidlist $y0]
 ++         }
 ++     }
 ++     if {!$haspad} {
 ++         # Find the first column that doesn't have a line going right
 ++         for {set col [llength $idlist]} {[incr col -1] >= 0} {} {
 ++             set id [lindex $idlist $col]
 ++             if {$id eq {}} break
 ++             set x0 [lsearch -exact $previdlist $id]
 ++             if {$x0 < 0} {
 ++                 # check if this is the link to the first child
 ++                 set kid [lindex $displayorder $y0]
 ++                 if {[lindex $children($curview,$id) 0] eq $kid} {
 ++                     # it is, work out offset to child
 ++                     set x0 [lsearch -exact $previdlist $kid]
 ++                 }
 ++             }
 ++             if {$x0 <= $col} break
 ++         }
 ++         # Insert a pad at that column as long as it has a line and
 ++         # isn't the last column
 ++         if {$x0 >= 0 && [incr col] < [llength $idlist]} {
 ++             set idlist [linsert $idlist $col {}]
 ++             lset rowidlist $row $idlist
 ++             changedrow $row
 ++         }
 ++     }
 ++    }
 ++}
 ++
 ++proc xc {row col} {
 ++    global canvx0 linespc
 ++    return [expr {$canvx0 + $col * $linespc}]
 ++}
 ++
 ++proc yc {row} {
 ++    global canvy0 linespc
 ++    return [expr {$canvy0 + $row * $linespc}]
 ++}
 ++
 ++proc linewidth {id} {
 ++    global thickerline lthickness
 ++
 ++    set wid $lthickness
 ++    if {[info exists thickerline] && $id eq $thickerline} {
 ++     set wid [expr {2 * $lthickness}]
 ++    }
 ++    return $wid
 ++}
 ++
 ++proc rowranges {id} {
 ++    global curview children uparrowlen downarrowlen
 ++    global rowidlist
 ++
 ++    set kids $children($curview,$id)
 ++    if {$kids eq {}} {
 ++     return {}
 ++    }
 ++    set ret {}
 ++    lappend kids $id
 ++    foreach child $kids {
 ++     if {![commitinview $child $curview]} break
 ++     set row [rowofcommit $child]
 ++     if {![info exists prev]} {
 ++         lappend ret [expr {$row + 1}]
 ++     } else {
 ++         if {$row <= $prevrow} {
 ++             puts "oops children of [shortids $id] out of order [shortids $child] $row <= [shortids $prev] $prevrow"
 ++         }
 ++         # see if the line extends the whole way from prevrow to row
 ++         if {$row > $prevrow + $uparrowlen + $downarrowlen &&
 ++             [lsearch -exact [lindex $rowidlist \
 ++                         [expr {int(($row + $prevrow) / 2)}]] $id] < 0} {
 ++             # it doesn't, see where it ends
 ++             set r [expr {$prevrow + $downarrowlen}]
 ++             if {[lsearch -exact [lindex $rowidlist $r] $id] < 0} {
 ++                 while {[incr r -1] > $prevrow &&
 ++                        [lsearch -exact [lindex $rowidlist $r] $id] < 0} {}
 ++             } else {
 ++                 while {[incr r] <= $row &&
 ++                        [lsearch -exact [lindex $rowidlist $r] $id] >= 0} {}
 ++                 incr r -1
 ++             }
 ++             lappend ret $r
 ++             # see where it starts up again
 ++             set r [expr {$row - $uparrowlen}]
 ++             if {[lsearch -exact [lindex $rowidlist $r] $id] < 0} {
 ++                 while {[incr r] < $row &&
 ++                        [lsearch -exact [lindex $rowidlist $r] $id] < 0} {}
 ++             } else {
 ++                 while {[incr r -1] >= $prevrow &&
 ++                        [lsearch -exact [lindex $rowidlist $r] $id] >= 0} {}
 ++                 incr r
 ++             }
 ++             lappend ret $r
 ++         }
 ++     }
 ++     if {$child eq $id} {
 ++         lappend ret $row
 ++     }
 ++     set prev $child
 ++     set prevrow $row
 ++    }
 ++    return $ret
 ++}
 ++
 ++proc drawlineseg {id row endrow arrowlow} {
 ++    global rowidlist displayorder iddrawn linesegs
 ++    global canv colormap linespc curview maxlinelen parentlist
 ++
 ++    set cols [list [lsearch -exact [lindex $rowidlist $row] $id]]
 ++    set le [expr {$row + 1}]
 ++    set arrowhigh 1
 ++    while {1} {
 ++     set c [lsearch -exact [lindex $rowidlist $le] $id]
 ++     if {$c < 0} {
 ++         incr le -1
 ++         break
 ++     }
 ++     lappend cols $c
 ++     set x [lindex $displayorder $le]
 ++     if {$x eq $id} {
 ++         set arrowhigh 0
 ++         break
 ++     }
 ++     if {[info exists iddrawn($x)] || $le == $endrow} {
 ++         set c [lsearch -exact [lindex $rowidlist [expr {$le+1}]] $id]
 ++         if {$c >= 0} {
 ++             lappend cols $c
 ++             set arrowhigh 0
 ++         }
 ++         break
 ++     }
 ++     incr le
 ++    }
 ++    if {$le <= $row} {
 ++     return $row
 ++    }
 ++
 ++    set lines {}
 ++    set i 0
 ++    set joinhigh 0
 ++    if {[info exists linesegs($id)]} {
 ++     set lines $linesegs($id)
 ++     foreach li $lines {
 ++         set r0 [lindex $li 0]
 ++         if {$r0 > $row} {
 ++             if {$r0 == $le && [lindex $li 1] - $row <= $maxlinelen} {
 ++                 set joinhigh 1
 ++             }
 ++             break
 ++         }
 ++         incr i
 ++     }
 ++    }
 ++    set joinlow 0
 ++    if {$i > 0} {
 ++     set li [lindex $lines [expr {$i-1}]]
 ++     set r1 [lindex $li 1]
 ++     if {$r1 == $row && $le - [lindex $li 0] <= $maxlinelen} {
 ++         set joinlow 1
 ++     }
 ++    }
 ++
 ++    set x [lindex $cols [expr {$le - $row}]]
 ++    set xp [lindex $cols [expr {$le - 1 - $row}]]
 ++    set dir [expr {$xp - $x}]
 ++    if {$joinhigh} {
 ++     set ith [lindex $lines $i 2]
 ++     set coords [$canv coords $ith]
 ++     set ah [$canv itemcget $ith -arrow]
 ++     set arrowhigh [expr {$ah eq "first" || $ah eq "both"}]
 ++     set x2 [lindex $cols [expr {$le + 1 - $row}]]
 ++     if {$x2 ne {} && $x - $x2 == $dir} {
 ++         set coords [lrange $coords 0 end-2]
 ++     }
 ++    } else {
 ++     set coords [list [xc $le $x] [yc $le]]
 ++    }
 ++    if {$joinlow} {
 ++     set itl [lindex $lines [expr {$i-1}] 2]
 ++     set al [$canv itemcget $itl -arrow]
 ++     set arrowlow [expr {$al eq "last" || $al eq "both"}]
 ++    } elseif {$arrowlow} {
 ++     if {[lsearch -exact [lindex $rowidlist [expr {$row-1}]] $id] >= 0 ||
 ++         [lsearch -exact [lindex $parentlist [expr {$row-1}]] $id] >= 0} {
 ++         set arrowlow 0
 ++     }
 ++    }
 ++    set arrow [lindex {none first last both} [expr {$arrowhigh + 2*$arrowlow}]]
 ++    for {set y $le} {[incr y -1] > $row} {} {
 ++     set x $xp
 ++     set xp [lindex $cols [expr {$y - 1 - $row}]]
 ++     set ndir [expr {$xp - $x}]
 ++     if {$dir != $ndir || $xp < 0} {
 ++         lappend coords [xc $y $x] [yc $y]
 ++     }
 ++     set dir $ndir
 ++    }
 ++    if {!$joinlow} {
 ++     if {$xp < 0} {
 ++         # join parent line to first child
 ++         set ch [lindex $displayorder $row]
 ++         set xc [lsearch -exact [lindex $rowidlist $row] $ch]
 ++         if {$xc < 0} {
 ++             puts "oops: drawlineseg: child $ch not on row $row"
 ++         } elseif {$xc != $x} {
 ++             if {($arrowhigh && $le == $row + 1) || $dir == 0} {
 ++                 set d [expr {int(0.5 * $linespc)}]
 ++                 set x1 [xc $row $x]
 ++                 if {$xc < $x} {
 ++                     set x2 [expr {$x1 - $d}]
 ++                 } else {
 ++                     set x2 [expr {$x1 + $d}]
 ++                 }
 ++                 set y2 [yc $row]
 ++                 set y1 [expr {$y2 + $d}]
 ++                 lappend coords $x1 $y1 $x2 $y2
 ++             } elseif {$xc < $x - 1} {
 ++                 lappend coords [xc $row [expr {$x-1}]] [yc $row]
 ++             } elseif {$xc > $x + 1} {
 ++                 lappend coords [xc $row [expr {$x+1}]] [yc $row]
 ++             }
 ++             set x $xc
 ++         }
 ++         lappend coords [xc $row $x] [yc $row]
 ++     } else {
 ++         set xn [xc $row $xp]
 ++         set yn [yc $row]
 ++         lappend coords $xn $yn
 ++     }
 ++     if {!$joinhigh} {
 ++         assigncolor $id
 ++         set t [$canv create line $coords -width [linewidth $id] \
 ++                    -fill $colormap($id) -tags lines.$id -arrow $arrow]
 ++         $canv lower $t
 ++         bindline $t $id
 ++         set lines [linsert $lines $i [list $row $le $t]]
 ++     } else {
 ++         $canv coords $ith $coords
 ++         if {$arrow ne $ah} {
 ++             $canv itemconf $ith -arrow $arrow
 ++         }
 ++         lset lines $i 0 $row
 ++     }
 ++    } else {
 ++     set xo [lsearch -exact [lindex $rowidlist [expr {$row - 1}]] $id]
 ++     set ndir [expr {$xo - $xp}]
 ++     set clow [$canv coords $itl]
 ++     if {$dir == $ndir} {
 ++         set clow [lrange $clow 2 end]
 ++     }
 ++     set coords [concat $coords $clow]
 ++     if {!$joinhigh} {
 ++         lset lines [expr {$i-1}] 1 $le
 ++     } else {
 ++         # coalesce two pieces
 ++         $canv delete $ith
 ++         set b [lindex $lines [expr {$i-1}] 0]
 ++         set e [lindex $lines $i 1]
 ++         set lines [lreplace $lines [expr {$i-1}] $i [list $b $e $itl]]
 ++     }
 ++     $canv coords $itl $coords
 ++     if {$arrow ne $al} {
 ++         $canv itemconf $itl -arrow $arrow
 ++     }
 ++    }
 ++
 ++    set linesegs($id) $lines
 ++    return $le
 ++}
 ++
 ++proc drawparentlinks {id row} {
 ++    global rowidlist canv colormap curview parentlist
 ++    global idpos linespc
 ++
 ++    set rowids [lindex $rowidlist $row]
 ++    set col [lsearch -exact $rowids $id]
 ++    if {$col < 0} return
 ++    set olds [lindex $parentlist $row]
 ++    set row2 [expr {$row + 1}]
 ++    set x [xc $row $col]
 ++    set y [yc $row]
 ++    set y2 [yc $row2]
 ++    set d [expr {int(0.5 * $linespc)}]
 ++    set ymid [expr {$y + $d}]
 ++    set ids [lindex $rowidlist $row2]
 ++    # rmx = right-most X coord used
 ++    set rmx 0
 ++    foreach p $olds {
 ++     set i [lsearch -exact $ids $p]
 ++     if {$i < 0} {
 ++         puts "oops, parent $p of $id not in list"
 ++         continue
 ++     }
 ++     set x2 [xc $row2 $i]
 ++     if {$x2 > $rmx} {
 ++         set rmx $x2
 ++     }
 ++     set j [lsearch -exact $rowids $p]
 ++     if {$j < 0} {
 ++         # drawlineseg will do this one for us
 ++         continue
 ++     }
 ++     assigncolor $p
 ++     # should handle duplicated parents here...
 ++     set coords [list $x $y]
 ++     if {$i != $col} {
 ++         # if attaching to a vertical segment, draw a smaller
 ++         # slant for visual distinctness
 ++         if {$i == $j} {
 ++             if {$i < $col} {
 ++                 lappend coords [expr {$x2 + $d}] $y $x2 $ymid
 ++             } else {
 ++                 lappend coords [expr {$x2 - $d}] $y $x2 $ymid
 ++             }
 ++         } elseif {$i < $col && $i < $j} {
 ++             # segment slants towards us already
 ++             lappend coords [xc $row $j] $y
 ++         } else {
 ++             if {$i < $col - 1} {
 ++                 lappend coords [expr {$x2 + $linespc}] $y
 ++             } elseif {$i > $col + 1} {
 ++                 lappend coords [expr {$x2 - $linespc}] $y
 ++             }
 ++             lappend coords $x2 $y2
 ++         }
 ++     } else {
 ++         lappend coords $x2 $y2
 ++     }
 ++     set t [$canv create line $coords -width [linewidth $p] \
 ++                -fill $colormap($p) -tags lines.$p]
 ++     $canv lower $t
 ++     bindline $t $p
 ++    }
 ++    if {$rmx > [lindex $idpos($id) 1]} {
 ++     lset idpos($id) 1 $rmx
 ++     redrawtags $id
 ++    }
 ++}
 ++
 ++proc drawlines {id} {
 ++    global canv
 ++
 ++    $canv itemconf lines.$id -width [linewidth $id]
 ++}
 ++
 ++proc drawcmittext {id row col} {
 ++    global linespc canv canv2 canv3 fgcolor curview
 ++    global cmitlisted commitinfo rowidlist parentlist
 ++    global rowtextx idpos idtags idheads idotherrefs
 ++    global linehtag linentag linedtag selectedline
 ++    global canvxmax boldrows boldnamerows fgcolor
 ++    global mainheadid nullid nullid2 circleitem circlecolors
 ++
 ++    # listed is 0 for boundary, 1 for normal, 2 for negative, 3 for left, 4 for right
 ++    set listed $cmitlisted($curview,$id)
 ++    if {$id eq $nullid} {
 ++     set ofill red
 ++    } elseif {$id eq $nullid2} {
 ++     set ofill green
 ++    } elseif {$id eq $mainheadid} {
 ++     set ofill yellow
 ++    } else {
 ++     set ofill [lindex $circlecolors $listed]
 ++    }
 ++    set x [xc $row $col]
 ++    set y [yc $row]
 ++    set orad [expr {$linespc / 3}]
 ++    if {$listed <= 2} {
 ++     set t [$canv create oval [expr {$x - $orad}] [expr {$y - $orad}] \
 ++                [expr {$x + $orad - 1}] [expr {$y + $orad - 1}] \
 ++                -fill $ofill -outline $fgcolor -width 1 -tags circle]
 ++    } elseif {$listed == 3} {
 ++     # triangle pointing left for left-side commits
 ++     set t [$canv create polygon \
 ++                [expr {$x - $orad}] $y \
 ++                [expr {$x + $orad - 1}] [expr {$y - $orad}] \
 ++                [expr {$x + $orad - 1}] [expr {$y + $orad - 1}] \
 ++                -fill $ofill -outline $fgcolor -width 1 -tags circle]
 ++    } else {
 ++     # triangle pointing right for right-side commits
 ++     set t [$canv create polygon \
 ++                [expr {$x + $orad - 1}] $y \
 ++                [expr {$x - $orad}] [expr {$y - $orad}] \
 ++                [expr {$x - $orad}] [expr {$y + $orad - 1}] \
 ++                -fill $ofill -outline $fgcolor -width 1 -tags circle]
 ++    }
 ++    set circleitem($row) $t
 ++    $canv raise $t
 ++    $canv bind $t <1> {selcanvline {} %x %y}
 ++    set rmx [llength [lindex $rowidlist $row]]
 ++    set olds [lindex $parentlist $row]
 ++    if {$olds ne {}} {
 ++     set nextids [lindex $rowidlist [expr {$row + 1}]]
 ++     foreach p $olds {
 ++         set i [lsearch -exact $nextids $p]
 ++         if {$i > $rmx} {
 ++             set rmx $i
 ++         }
 ++     }
 ++    }
 ++    set xt [xc $row $rmx]
 ++    set rowtextx($row) $xt
 ++    set idpos($id) [list $x $xt $y]
 ++    if {[info exists idtags($id)] || [info exists idheads($id)]
 ++     || [info exists idotherrefs($id)]} {
 ++     set xt [drawtags $id $x $xt $y]
 ++    }
 ++    set headline [lindex $commitinfo($id) 0]
 ++    set name [lindex $commitinfo($id) 1]
 ++    set date [lindex $commitinfo($id) 2]
 ++    set date [formatdate $date]
 ++    set font mainfont
 ++    set nfont mainfont
 ++    set isbold [ishighlighted $id]
 ++    if {$isbold > 0} {
 ++     lappend boldrows $row
 ++     set font mainfontbold
 ++     if {$isbold > 1} {
 ++         lappend boldnamerows $row
 ++         set nfont mainfontbold
 ++     }
 ++    }
 ++    set linehtag($row) [$canv create text $xt $y -anchor w -fill $fgcolor \
 ++                         -text $headline -font $font -tags text]
 ++    $canv bind $linehtag($row) <Button-3> "rowmenu %X %Y $id"
 ++    set linentag($row) [$canv2 create text 3 $y -anchor w -fill $fgcolor \
 ++                         -text $name -font $nfont -tags text]
 ++    set linedtag($row) [$canv3 create text 3 $y -anchor w -fill $fgcolor \
 ++                         -text $date -font mainfont -tags text]
 ++    if {$selectedline == $row} {
 ++     make_secsel $row
 ++    }
 ++    set xr [expr {$xt + [font measure $font $headline]}]
 ++    if {$xr > $canvxmax} {
 ++     set canvxmax $xr
 ++     setcanvscroll
 ++    }
 ++}
 ++
 ++proc drawcmitrow {row} {
 ++    global displayorder rowidlist nrows_drawn
 ++    global iddrawn markingmatches
 ++    global commitinfo numcommits
 ++    global filehighlight fhighlights findpattern nhighlights
 ++    global hlview vhighlights
 ++    global highlight_related rhighlights
 ++
 ++    if {$row >= $numcommits} return
 ++
 ++    set id [lindex $displayorder $row]
 ++    if {[info exists hlview] && ![info exists vhighlights($id)]} {
 ++     askvhighlight $row $id
 ++    }
 ++    if {[info exists filehighlight] && ![info exists fhighlights($id)]} {
 ++     askfilehighlight $row $id
 ++    }
 ++    if {$findpattern ne {} && ![info exists nhighlights($id)]} {
 ++     askfindhighlight $row $id
 ++    }
 ++    if {$highlight_related ne [mc "None"] && ![info exists rhighlights($id)]} {
 ++     askrelhighlight $row $id
 ++    }
 ++    if {![info exists iddrawn($id)]} {
 ++     set col [lsearch -exact [lindex $rowidlist $row] $id]
 ++     if {$col < 0} {
 ++         puts "oops, row $row id $id not in list"
 ++         return
 ++     }
 ++     if {![info exists commitinfo($id)]} {
 ++         getcommit $id
 ++     }
 ++     assigncolor $id
 ++     drawcmittext $id $row $col
 ++     set iddrawn($id) 1
 ++     incr nrows_drawn
 ++    }
 ++    if {$markingmatches} {
 ++     markrowmatches $row $id
 ++    }
 ++}
 ++
 ++proc drawcommits {row {endrow {}}} {
 ++    global numcommits iddrawn displayorder curview need_redisplay
 ++    global parentlist rowidlist rowfinal uparrowlen downarrowlen nrows_drawn
 ++
 ++    if {$row < 0} {
 ++     set row 0
 ++    }
 ++    if {$endrow eq {}} {
 ++     set endrow $row
 ++    }
 ++    if {$endrow >= $numcommits} {
 ++     set endrow [expr {$numcommits - 1}]
 ++    }
 ++
 ++    set rl1 [expr {$row - $downarrowlen - 3}]
 ++    if {$rl1 < 0} {
 ++     set rl1 0
 ++    }
 ++    set ro1 [expr {$row - 3}]
 ++    if {$ro1 < 0} {
 ++     set ro1 0
 ++    }
 ++    set r2 [expr {$endrow + $uparrowlen + 3}]
 ++    if {$r2 > $numcommits} {
 ++     set r2 $numcommits
 ++    }
 ++    for {set r $rl1} {$r < $r2} {incr r} {
 ++     if {[lindex $rowidlist $r] ne {} && [lindex $rowfinal $r]} {
 ++         if {$rl1 < $r} {
 ++             layoutrows $rl1 $r
 ++         }
 ++         set rl1 [expr {$r + 1}]
 ++     }
 ++    }
 ++    if {$rl1 < $r} {
 ++     layoutrows $rl1 $r
 ++    }
 ++    optimize_rows $ro1 0 $r2
 ++    if {$need_redisplay || $nrows_drawn > 2000} {
 ++     clear_display
 ++     drawvisible
 ++    }
 ++
 ++    # make the lines join to already-drawn rows either side
 ++    set r [expr {$row - 1}]
 ++    if {$r < 0 || ![info exists iddrawn([lindex $displayorder $r])]} {
 ++     set r $row
 ++    }
 ++    set er [expr {$endrow + 1}]
 ++    if {$er >= $numcommits ||
 ++     ![info exists iddrawn([lindex $displayorder $er])]} {
 ++     set er $endrow
 ++    }
 ++    for {} {$r <= $er} {incr r} {
 ++     set id [lindex $displayorder $r]
 ++     set wasdrawn [info exists iddrawn($id)]
 ++     drawcmitrow $r
 ++     if {$r == $er} break
 ++     set nextid [lindex $displayorder [expr {$r + 1}]]
 ++     if {$wasdrawn && [info exists iddrawn($nextid)]} continue
 ++     drawparentlinks $id $r
 ++
 ++     set rowids [lindex $rowidlist $r]
 ++     foreach lid $rowids {
 ++         if {$lid eq {}} continue
 ++         if {[info exists lineend($lid)] && $lineend($lid) > $r} continue
 ++         if {$lid eq $id} {
 ++             # see if this is the first child of any of its parents
 ++             foreach p [lindex $parentlist $r] {
 ++                 if {[lsearch -exact $rowids $p] < 0} {
 ++                     # make this line extend up to the child
 ++                     set lineend($p) [drawlineseg $p $r $er 0]
 ++                 }
 ++             }
 ++         } else {
 ++             set lineend($lid) [drawlineseg $lid $r $er 1]
 ++         }
 ++     }
 ++    }
 ++}
 ++
 ++proc undolayout {row} {
 ++    global uparrowlen mingaplen downarrowlen
 ++    global rowidlist rowisopt rowfinal need_redisplay
 ++
 ++    set r [expr {$row - ($uparrowlen + $mingaplen + $downarrowlen)}]
 ++    if {$r < 0} {
 ++     set r 0
 ++    }
 ++    if {[llength $rowidlist] > $r} {
 ++     incr r -1
 ++     set rowidlist [lrange $rowidlist 0 $r]
 ++     set rowfinal [lrange $rowfinal 0 $r]
 ++     set rowisopt [lrange $rowisopt 0 $r]
 ++     set need_redisplay 1
 ++     run drawvisible
 ++    }
 ++}
 ++
 ++proc drawvisible {} {
 ++    global canv linespc curview vrowmod selectedline targetrow targetid
 ++    global need_redisplay cscroll numcommits
 ++
 ++    set fs [$canv yview]
 ++    set ymax [lindex [$canv cget -scrollregion] 3]
 ++    if {$ymax eq {} || $ymax == 0 || $numcommits == 0} return
 ++    set f0 [lindex $fs 0]
 ++    set f1 [lindex $fs 1]
 ++    set y0 [expr {int($f0 * $ymax)}]
 ++    set y1 [expr {int($f1 * $ymax)}]
 ++
 ++    if {[info exists targetid]} {
 ++     if {[commitinview $targetid $curview]} {
 ++         set r [rowofcommit $targetid]
 ++         if {$r != $targetrow} {
 ++             # Fix up the scrollregion and change the scrolling position
 ++             # now that our target row has moved.
 ++             set diff [expr {($r - $targetrow) * $linespc}]
 ++             set targetrow $r
 ++             setcanvscroll
 ++             set ymax [lindex [$canv cget -scrollregion] 3]
 ++             incr y0 $diff
 ++             incr y1 $diff
 ++             set f0 [expr {$y0 / $ymax}]
 ++             set f1 [expr {$y1 / $ymax}]
 ++             allcanvs yview moveto $f0
 ++             $cscroll set $f0 $f1
 ++             set need_redisplay 1
 ++         }
 ++     } else {
 ++         unset targetid
 ++     }
 ++    }
 ++
 ++    set row [expr {int(($y0 - 3) / $linespc) - 1}]
 ++    set endrow [expr {int(($y1 - 3) / $linespc) + 1}]
 ++    if {$endrow >= $vrowmod($curview)} {
 ++     update_arcrows $curview
 ++    }
 ++    if {$selectedline ne {} &&
 ++     $row <= $selectedline && $selectedline <= $endrow} {
 ++     set targetrow $selectedline
 ++    } elseif {[info exists targetid]} {
 ++     set targetrow [expr {int(($row + $endrow) / 2)}]
 ++    }
 ++    if {[info exists targetrow]} {
 ++     if {$targetrow >= $numcommits} {
 ++         set targetrow [expr {$numcommits - 1}]
 ++     }
 ++     set targetid [commitonrow $targetrow]
 ++    }
 ++    drawcommits $row $endrow
 ++}
 ++
 ++proc clear_display {} {
 ++    global iddrawn linesegs need_redisplay nrows_drawn
 ++    global vhighlights fhighlights nhighlights rhighlights
 ++    global linehtag linentag linedtag boldrows boldnamerows
 ++
 ++    allcanvs delete all
 ++    catch {unset iddrawn}
 ++    catch {unset linesegs}
 ++    catch {unset linehtag}
 ++    catch {unset linentag}
 ++    catch {unset linedtag}
 ++    set boldrows {}
 ++    set boldnamerows {}
 ++    catch {unset vhighlights}
 ++    catch {unset fhighlights}
 ++    catch {unset nhighlights}
 ++    catch {unset rhighlights}
 ++    set need_redisplay 0
 ++    set nrows_drawn 0
 ++}
 ++
 ++proc findcrossings {id} {
 ++    global rowidlist parentlist numcommits displayorder
 ++
 ++    set cross {}
 ++    set ccross {}
 ++    foreach {s e} [rowranges $id] {
 ++     if {$e >= $numcommits} {
 ++         set e [expr {$numcommits - 1}]
 ++     }
 ++     if {$e <= $s} continue
 ++     for {set row $e} {[incr row -1] >= $s} {} {
 ++         set x [lsearch -exact [lindex $rowidlist $row] $id]
 ++         if {$x < 0} break
 ++         set olds [lindex $parentlist $row]
 ++         set kid [lindex $displayorder $row]
 ++         set kidx [lsearch -exact [lindex $rowidlist $row] $kid]
 ++         if {$kidx < 0} continue
 ++         set nextrow [lindex $rowidlist [expr {$row + 1}]]
 ++         foreach p $olds {
 ++             set px [lsearch -exact $nextrow $p]
 ++             if {$px < 0} continue
 ++             if {($kidx < $x && $x < $px) || ($px < $x && $x < $kidx)} {
 ++                 if {[lsearch -exact $ccross $p] >= 0} continue
 ++                 if {$x == $px + ($kidx < $px? -1: 1)} {
 ++                     lappend ccross $p
 ++                 } elseif {[lsearch -exact $cross $p] < 0} {
 ++                     lappend cross $p
 ++                 }
 ++             }
 ++         }
 ++     }
 ++    }
 ++    return [concat $ccross {{}} $cross]
 ++}
 ++
 ++proc assigncolor {id} {
 ++    global colormap colors nextcolor
 ++    global parents children children curview
 ++
 ++    if {[info exists colormap($id)]} return
 ++    set ncolors [llength $colors]
 ++    if {[info exists children($curview,$id)]} {
 ++     set kids $children($curview,$id)
 ++    } else {
 ++     set kids {}
 ++    }
 ++    if {[llength $kids] == 1} {
 ++     set child [lindex $kids 0]
 ++     if {[info exists colormap($child)]
 ++         && [llength $parents($curview,$child)] == 1} {
 ++         set colormap($id) $colormap($child)
 ++         return
 ++     }
 ++    }
 ++    set badcolors {}
 ++    set origbad {}
 ++    foreach x [findcrossings $id] {
 ++     if {$x eq {}} {
 ++         # delimiter between corner crossings and other crossings
 ++         if {[llength $badcolors] >= $ncolors - 1} break
 ++         set origbad $badcolors
 ++     }
 ++     if {[info exists colormap($x)]
 ++         && [lsearch -exact $badcolors $colormap($x)] < 0} {
 ++         lappend badcolors $colormap($x)
 ++     }
 ++    }
 ++    if {[llength $badcolors] >= $ncolors} {
 ++     set badcolors $origbad
 ++    }
 ++    set origbad $badcolors
 ++    if {[llength $badcolors] < $ncolors - 1} {
 ++     foreach child $kids {
 ++         if {[info exists colormap($child)]
 ++             && [lsearch -exact $badcolors $colormap($child)] < 0} {
 ++             lappend badcolors $colormap($child)
 ++         }
 ++         foreach p $parents($curview,$child) {
 ++             if {[info exists colormap($p)]
 ++                 && [lsearch -exact $badcolors $colormap($p)] < 0} {
 ++                 lappend badcolors $colormap($p)
 ++             }
 ++         }
 ++     }
 ++     if {[llength $badcolors] >= $ncolors} {
 ++         set badcolors $origbad
 ++     }
 ++    }
 ++    for {set i 0} {$i <= $ncolors} {incr i} {
 ++     set c [lindex $colors $nextcolor]
 ++     if {[incr nextcolor] >= $ncolors} {
 ++         set nextcolor 0
 ++     }
 ++     if {[lsearch -exact $badcolors $c]} break
 ++    }
 ++    set colormap($id) $c
 ++}
 ++
 ++proc bindline {t id} {
 ++    global canv
 ++
 ++    $canv bind $t <Enter> "lineenter %x %y $id"
 ++    $canv bind $t <Motion> "linemotion %x %y $id"
 ++    $canv bind $t <Leave> "lineleave $id"
 ++    $canv bind $t <Button-1> "lineclick %x %y $id 1"
 ++}
 ++
 ++proc drawtags {id x xt y1} {
 ++    global idtags idheads idotherrefs mainhead
 ++    global linespc lthickness
 ++    global canv rowtextx curview fgcolor bgcolor
 ++
 ++    set marks {}
 ++    set ntags 0
 ++    set nheads 0
 ++    if {[info exists idtags($id)]} {
 ++     set marks $idtags($id)
 ++     set ntags [llength $marks]
 ++    }
 ++    if {[info exists idheads($id)]} {
 ++     set marks [concat $marks $idheads($id)]
 ++     set nheads [llength $idheads($id)]
 ++    }
 ++    if {[info exists idotherrefs($id)]} {
 ++     set marks [concat $marks $idotherrefs($id)]
 ++    }
 ++    if {$marks eq {}} {
 ++     return $xt
 ++    }
 ++
 ++    set delta [expr {int(0.5 * ($linespc - $lthickness))}]
 ++    set yt [expr {$y1 - 0.5 * $linespc}]
 ++    set yb [expr {$yt + $linespc - 1}]
 ++    set xvals {}
 ++    set wvals {}
 ++    set i -1
 ++    foreach tag $marks {
 ++     incr i
 ++     if {$i >= $ntags && $i < $ntags + $nheads && $tag eq $mainhead} {
 ++         set wid [font measure mainfontbold $tag]
 ++     } else {
 ++         set wid [font measure mainfont $tag]
 ++     }
 ++     lappend xvals $xt
 ++     lappend wvals $wid
 ++     set xt [expr {$xt + $delta + $wid + $lthickness + $linespc}]
 ++    }
 ++    set t [$canv create line $x $y1 [lindex $xvals end] $y1 \
 ++            -width $lthickness -fill black -tags tag.$id]
 ++    $canv lower $t
 ++    foreach tag $marks x $xvals wid $wvals {
 ++     set xl [expr {$x + $delta}]
 ++     set xr [expr {$x + $delta + $wid + $lthickness}]
 ++     set font mainfont
 ++     if {[incr ntags -1] >= 0} {
 ++         # draw a tag
 ++         set t [$canv create polygon $x [expr {$yt + $delta}] $xl $yt \
 ++                    $xr $yt $xr $yb $xl $yb $x [expr {$yb - $delta}] \
 ++                    -width 1 -outline black -fill yellow -tags tag.$id]
 ++         $canv bind $t <1> [list showtag $tag 1]
 ++         set rowtextx([rowofcommit $id]) [expr {$xr + $linespc}]
 ++     } else {
 ++         # draw a head or other ref
 ++         if {[incr nheads -1] >= 0} {
 ++             set col green
 ++             if {$tag eq $mainhead} {
 ++                 set font mainfontbold
 ++             }
 ++         } else {
 ++             set col "#ddddff"
 ++         }
 ++         set xl [expr {$xl - $delta/2}]
 ++         $canv create polygon $x $yt $xr $yt $xr $yb $x $yb \
 ++             -width 1 -outline black -fill $col -tags tag.$id
 ++         if {[regexp {^(remotes/.*/|remotes/)} $tag match remoteprefix]} {
 ++             set rwid [font measure mainfont $remoteprefix]
 ++             set xi [expr {$x + 1}]
 ++             set yti [expr {$yt + 1}]
 ++             set xri [expr {$x + $rwid}]
 ++             $canv create polygon $xi $yti $xri $yti $xri $yb $xi $yb \
 ++                     -width 0 -fill "#ffddaa" -tags tag.$id
 ++         }
 ++     }
 ++     set t [$canv create text $xl $y1 -anchor w -text $tag -fill $fgcolor \
 ++                -font $font -tags [list tag.$id text]]
 ++     if {$ntags >= 0} {
 ++         $canv bind $t <1> [list showtag $tag 1]
 ++     } elseif {$nheads >= 0} {
 ++         $canv bind $t <Button-3> [list headmenu %X %Y $id $tag]
 ++     }
 ++    }
 ++    return $xt
 ++}
 ++
 ++proc xcoord {i level ln} {
 ++    global canvx0 xspc1 xspc2
 ++
 ++    set x [expr {$canvx0 + $i * $xspc1($ln)}]
 ++    if {$i > 0 && $i == $level} {
 ++     set x [expr {$x + 0.5 * ($xspc2 - $xspc1($ln))}]
 ++    } elseif {$i > $level} {
 ++     set x [expr {$x + $xspc2 - $xspc1($ln)}]
 ++    }
 ++    return $x
 ++}
 ++
 ++proc show_status {msg} {
 ++    global canv fgcolor
 ++
 ++    clear_display
 ++    $canv create text 3 3 -anchor nw -text $msg -font mainfont \
 ++     -tags text -fill $fgcolor
 ++}
 ++
 ++# Don't change the text pane cursor if it is currently the hand cursor,
 ++# showing that we are over a sha1 ID link.
 ++proc settextcursor {c} {
 ++    global ctext curtextcursor
 ++
 ++    if {[$ctext cget -cursor] == $curtextcursor} {
 ++     $ctext config -cursor $c
 ++    }
 ++    set curtextcursor $c
 ++}
 ++
 ++proc nowbusy {what {name {}}} {
 ++    global isbusy busyname statusw
 ++
 ++    if {[array names isbusy] eq {}} {
 ++     . config -cursor watch
 ++     settextcursor watch
 ++    }
 ++    set isbusy($what) 1
 ++    set busyname($what) $name
 ++    if {$name ne {}} {
 ++     $statusw conf -text $name
 ++    }
 ++}
 ++
 ++proc notbusy {what} {
 ++    global isbusy maincursor textcursor busyname statusw
 ++
 ++    catch {
 ++     unset isbusy($what)
 ++     if {$busyname($what) ne {} &&
 ++         [$statusw cget -text] eq $busyname($what)} {
 ++         $statusw conf -text {}
 ++     }
 ++    }
 ++    if {[array names isbusy] eq {}} {
 ++     . config -cursor $maincursor
 ++     settextcursor $textcursor
 ++    }
 ++}
 ++
 ++proc findmatches {f} {
 ++    global findtype findstring
 ++    if {$findtype == [mc "Regexp"]} {
 ++     set matches [regexp -indices -all -inline $findstring $f]
 ++    } else {
 ++     set fs $findstring
 ++     if {$findtype == [mc "IgnCase"]} {
 ++         set f [string tolower $f]
 ++         set fs [string tolower $fs]
 ++     }
 ++     set matches {}
 ++     set i 0
 ++     set l [string length $fs]
 ++     while {[set j [string first $fs $f $i]] >= 0} {
 ++         lappend matches [list $j [expr {$j+$l-1}]]
 ++         set i [expr {$j + $l}]
 ++     }
 ++    }
 ++    return $matches
 ++}
 ++
 ++proc dofind {{dirn 1} {wrap 1}} {
 ++    global findstring findstartline findcurline selectedline numcommits
 ++    global gdttype filehighlight fh_serial find_dirn findallowwrap
 ++
 ++    if {[info exists find_dirn]} {
 ++     if {$find_dirn == $dirn} return
 ++     stopfinding
 ++    }
 ++    focus .
 ++    if {$findstring eq {} || $numcommits == 0} return
 ++    if {$selectedline eq {}} {
 ++     set findstartline [lindex [visiblerows] [expr {$dirn < 0}]]
 ++    } else {
 ++     set findstartline $selectedline
 ++    }
 ++    set findcurline $findstartline
 ++    nowbusy finding [mc "Searching"]
 ++    if {$gdttype ne [mc "containing:"] && ![info exists filehighlight]} {
 ++     after cancel do_file_hl $fh_serial
 ++     do_file_hl $fh_serial
 ++    }
 ++    set find_dirn $dirn
 ++    set findallowwrap $wrap
 ++    run findmore
 ++}
 ++
 ++proc stopfinding {} {
 ++    global find_dirn findcurline fprogcoord
 ++
 ++    if {[info exists find_dirn]} {
 ++     unset find_dirn
 ++     unset findcurline
 ++     notbusy finding
 ++     set fprogcoord 0
 ++     adjustprogress
 ++    }
 ++}
 ++
 ++proc findmore {} {
 ++    global commitdata commitinfo numcommits findpattern findloc
 ++    global findstartline findcurline findallowwrap
 ++    global find_dirn gdttype fhighlights fprogcoord
 ++    global curview varcorder vrownum varccommits vrowmod
 ++
 ++    if {![info exists find_dirn]} {
 ++     return 0
 ++    }
 ++    set fldtypes [list [mc "Headline"] [mc "Author"] [mc "Date"] [mc "Committer"] [mc "CDate"] [mc "Comments"]]
 ++    set l $findcurline
 ++    set moretodo 0
 ++    if {$find_dirn > 0} {
 ++     incr l
 ++     if {$l >= $numcommits} {
 ++         set l 0
 ++     }
 ++     if {$l <= $findstartline} {
 ++         set lim [expr {$findstartline + 1}]
 ++     } else {
 ++         set lim $numcommits
 ++         set moretodo $findallowwrap
 ++     }
 ++    } else {
 ++     if {$l == 0} {
 ++         set l $numcommits
 ++     }
 ++     incr l -1
 ++     if {$l >= $findstartline} {
 ++         set lim [expr {$findstartline - 1}]
 ++     } else {
 ++         set lim -1
 ++         set moretodo $findallowwrap
 ++     }
 ++    }
 ++    set n [expr {($lim - $l) * $find_dirn}]
 ++    if {$n > 500} {
 ++     set n 500
 ++     set moretodo 1
 ++    }
 ++    if {$l + ($find_dirn > 0? $n: 1) > $vrowmod($curview)} {
 ++     update_arcrows $curview
 ++    }
 ++    set found 0
 ++    set domore 1
 ++    set ai [bsearch $vrownum($curview) $l]
 ++    set a [lindex $varcorder($curview) $ai]
 ++    set arow [lindex $vrownum($curview) $ai]
 ++    set ids [lindex $varccommits($curview,$a)]
 ++    set arowend [expr {$arow + [llength $ids]}]
 ++    if {$gdttype eq [mc "containing:"]} {
 ++     for {} {$n > 0} {incr n -1; incr l $find_dirn} {
 ++         if {$l < $arow || $l >= $arowend} {
 ++             incr ai $find_dirn
 ++             set a [lindex $varcorder($curview) $ai]
 ++             set arow [lindex $vrownum($curview) $ai]
 ++             set ids [lindex $varccommits($curview,$a)]
 ++             set arowend [expr {$arow + [llength $ids]}]
 ++         }
 ++         set id [lindex $ids [expr {$l - $arow}]]
 ++         # shouldn't happen unless git log doesn't give all the commits...
 ++         if {![info exists commitdata($id)] ||
 ++             ![doesmatch $commitdata($id)]} {
 ++             continue
 ++         }
 ++         if {![info exists commitinfo($id)]} {
 ++             getcommit $id
 ++         }
 ++         set info $commitinfo($id)
 ++         foreach f $info ty $fldtypes {
 ++             if {($findloc eq [mc "All fields"] || $findloc eq $ty) &&
 ++                 [doesmatch $f]} {
 ++                 set found 1
 ++                 break
 ++             }
 ++         }
 ++         if {$found} break
 ++     }
 ++    } else {
 ++     for {} {$n > 0} {incr n -1; incr l $find_dirn} {
 ++         if {$l < $arow || $l >= $arowend} {
 ++             incr ai $find_dirn
 ++             set a [lindex $varcorder($curview) $ai]
 ++             set arow [lindex $vrownum($curview) $ai]
 ++             set ids [lindex $varccommits($curview,$a)]
 ++             set arowend [expr {$arow + [llength $ids]}]
 ++         }
 ++         set id [lindex $ids [expr {$l - $arow}]]
 ++         if {![info exists fhighlights($id)]} {
 ++             # this sets fhighlights($id) to -1
 ++             askfilehighlight $l $id
 ++         }
 ++         if {$fhighlights($id) > 0} {
 ++             set found $domore
 ++             break
 ++         }
 ++         if {$fhighlights($id) < 0} {
 ++             if {$domore} {
 ++                 set domore 0
 ++                 set findcurline [expr {$l - $find_dirn}]
 ++             }
 ++         }
 ++     }
 ++    }
 ++    if {$found || ($domore && !$moretodo)} {
 ++     unset findcurline
 ++     unset find_dirn
 ++     notbusy finding
 ++     set fprogcoord 0
 ++     adjustprogress
 ++     if {$found} {
 ++         findselectline $l
 ++     } else {
 ++         bell
 ++     }
 ++     return 0
 ++    }
 ++    if {!$domore} {
 ++     flushhighlights
 ++    } else {
 ++     set findcurline [expr {$l - $find_dirn}]
 ++    }
 ++    set n [expr {($findcurline - $findstartline) * $find_dirn - 1}]
 ++    if {$n < 0} {
 ++     incr n $numcommits
 ++    }
 ++    set fprogcoord [expr {$n * 1.0 / $numcommits}]
 ++    adjustprogress
 ++    return $domore
 ++}
 ++
 ++proc findselectline {l} {
 ++    global findloc commentend ctext findcurline markingmatches gdttype
 ++
 ++    set markingmatches 1
 ++    set findcurline $l
 ++    selectline $l 1
 ++    if {$findloc == [mc "All fields"] || $findloc == [mc "Comments"]} {
 ++     # highlight the matches in the comments
 ++     set f [$ctext get 1.0 $commentend]
 ++     set matches [findmatches $f]
 ++     foreach match $matches {
 ++         set start [lindex $match 0]
 ++         set end [expr {[lindex $match 1] + 1}]
 ++         $ctext tag add found "1.0 + $start c" "1.0 + $end c"
 ++     }
 ++    }
 ++    drawvisible
 ++}
 ++
 ++# mark the bits of a headline or author that match a find string
 ++proc markmatches {canv l str tag matches font row} {
 ++    global selectedline
 ++
 ++    set bbox [$canv bbox $tag]
 ++    set x0 [lindex $bbox 0]
 ++    set y0 [lindex $bbox 1]
 ++    set y1 [lindex $bbox 3]
 ++    foreach match $matches {
 ++     set start [lindex $match 0]
 ++     set end [lindex $match 1]
 ++     if {$start > $end} continue
 ++     set xoff [font measure $font [string range $str 0 [expr {$start-1}]]]
 ++     set xlen [font measure $font [string range $str 0 [expr {$end}]]]
 ++     set t [$canv create rect [expr {$x0+$xoff}] $y0 \
 ++                [expr {$x0+$xlen+2}] $y1 \
 ++                -outline {} -tags [list match$l matches] -fill yellow]
 ++     $canv lower $t
 ++     if {$row == $selectedline} {
 ++         $canv raise $t secsel
 ++     }
 ++    }
 ++}
 ++
 ++proc unmarkmatches {} {
 ++    global markingmatches
 ++
 ++    allcanvs delete matches
 ++    set markingmatches 0
 ++    stopfinding
 ++}
 ++
 ++proc selcanvline {w x y} {
 ++    global canv canvy0 ctext linespc
 ++    global rowtextx
 ++    set ymax [lindex [$canv cget -scrollregion] 3]
 ++    if {$ymax == {}} return
 ++    set yfrac [lindex [$canv yview] 0]
 ++    set y [expr {$y + $yfrac * $ymax}]
 ++    set l [expr {int(($y - $canvy0) / $linespc + 0.5)}]
 ++    if {$l < 0} {
 ++     set l 0
 ++    }
 ++    if {$w eq $canv} {
 ++     set xmax [lindex [$canv cget -scrollregion] 2]
 ++     set xleft [expr {[lindex [$canv xview] 0] * $xmax}]
 ++     if {![info exists rowtextx($l)] || $xleft + $x < $rowtextx($l)} return
 ++    }
 ++    unmarkmatches
 ++    selectline $l 1
 ++}
 ++
 ++proc commit_descriptor {p} {
 ++    global commitinfo
 ++    if {![info exists commitinfo($p)]} {
 ++     getcommit $p
 ++    }
 ++    set l "..."
 ++    if {[llength $commitinfo($p)] > 1} {
 ++     set l [lindex $commitinfo($p) 0]
 ++    }
 ++    return "$p ($l)\n"
 ++}
 ++
 ++# append some text to the ctext widget, and make any SHA1 ID
 ++# that we know about be a clickable link.
 ++proc appendwithlinks {text tags} {
 ++    global ctext linknum curview pendinglinks
 ++
 ++    set start [$ctext index "end - 1c"]
 ++    $ctext insert end $text $tags
 ++    set links [regexp -indices -all -inline {[0-9a-f]{40}} $text]
 ++    foreach l $links {
 ++     set s [lindex $l 0]
 ++     set e [lindex $l 1]
 ++     set linkid [string range $text $s $e]
 ++     incr e
 ++     $ctext tag delete link$linknum
 ++     $ctext tag add link$linknum "$start + $s c" "$start + $e c"
 ++     setlink $linkid link$linknum
 ++     incr linknum
 ++    }
 ++}
 ++
 ++proc setlink {id lk} {
 ++    global curview ctext pendinglinks commitinterest
 ++
 ++    if {[commitinview $id $curview]} {
 ++     $ctext tag conf $lk -foreground blue -underline 1
 ++     $ctext tag bind $lk <1> [list selectline [rowofcommit $id] 1]
 ++     $ctext tag bind $lk <Enter> {linkcursor %W 1}
 ++     $ctext tag bind $lk <Leave> {linkcursor %W -1}
 ++    } else {
 ++     lappend pendinglinks($id) $lk
 ++     lappend commitinterest($id) {makelink %I}
 ++    }
 ++}
 ++
 ++proc makelink {id} {
 ++    global pendinglinks
 ++
 ++    if {![info exists pendinglinks($id)]} return
 ++    foreach lk $pendinglinks($id) {
 ++     setlink $id $lk
 ++    }
 ++    unset pendinglinks($id)
 ++}
 ++
 ++proc linkcursor {w inc} {
 ++    global linkentercount curtextcursor
 ++
 ++    if {[incr linkentercount $inc] > 0} {
 ++     $w configure -cursor hand2
 ++    } else {
 ++     $w configure -cursor $curtextcursor
 ++     if {$linkentercount < 0} {
 ++         set linkentercount 0
 ++     }
 ++    }
 ++}
 ++
 ++proc viewnextline {dir} {
 ++    global canv linespc
 ++
 ++    $canv delete hover
 ++    set ymax [lindex [$canv cget -scrollregion] 3]
 ++    set wnow [$canv yview]
 ++    set wtop [expr {[lindex $wnow 0] * $ymax}]
 ++    set newtop [expr {$wtop + $dir * $linespc}]
 ++    if {$newtop < 0} {
 ++     set newtop 0
 ++    } elseif {$newtop > $ymax} {
 ++     set newtop $ymax
 ++    }
 ++    allcanvs yview moveto [expr {$newtop * 1.0 / $ymax}]
 ++}
 ++
 ++# add a list of tag or branch names at position pos
 ++# returns the number of names inserted
 ++proc appendrefs {pos ids var} {
 ++    global ctext linknum curview $var maxrefs
 ++
 ++    if {[catch {$ctext index $pos}]} {
 ++     return 0
 ++    }
 ++    $ctext conf -state normal
 ++    $ctext delete $pos "$pos lineend"
 ++    set tags {}
 ++    foreach id $ids {
 ++     foreach tag [set $var\($id\)] {
 ++         lappend tags [list $tag $id]
 ++     }
 ++    }
 ++    if {[llength $tags] > $maxrefs} {
 ++     $ctext insert $pos "many ([llength $tags])"
 ++    } else {
 ++     set tags [lsort -index 0 -decreasing $tags]
 ++     set sep {}
 ++     foreach ti $tags {
 ++         set id [lindex $ti 1]
 ++         set lk link$linknum
 ++         incr linknum
 ++         $ctext tag delete $lk
 ++         $ctext insert $pos $sep
 ++         $ctext insert $pos [lindex $ti 0] $lk
 ++         setlink $id $lk
 ++         set sep ", "
 ++     }
 ++    }
 ++    $ctext conf -state disabled
 ++    return [llength $tags]
 ++}
 ++
 ++# called when we have finished computing the nearby tags
 ++proc dispneartags {delay} {
 ++    global selectedline currentid showneartags tagphase
 ++
 ++    if {$selectedline eq {} || !$showneartags} return
 ++    after cancel dispnexttag
 ++    if {$delay} {
 ++     after 200 dispnexttag
 ++     set tagphase -1
 ++    } else {
 ++     after idle dispnexttag
 ++     set tagphase 0
 ++    }
 ++}
 ++
 ++proc dispnexttag {} {
 ++    global selectedline currentid showneartags tagphase ctext
 ++
 ++    if {$selectedline eq {} || !$showneartags} return
 ++    switch -- $tagphase {
 ++     0 {
 ++         set dtags [desctags $currentid]
 ++         if {$dtags ne {}} {
 ++             appendrefs precedes $dtags idtags
 ++         }
 ++     }
 ++     1 {
 ++         set atags [anctags $currentid]
 ++         if {$atags ne {}} {
 ++             appendrefs follows $atags idtags
 ++         }
 ++     }
 ++     2 {
 ++         set dheads [descheads $currentid]
 ++         if {$dheads ne {}} {
 ++             if {[appendrefs branch $dheads idheads] > 1
 ++                 && [$ctext get "branch -3c"] eq "h"} {
 ++                 # turn "Branch" into "Branches"
 ++                 $ctext conf -state normal
 ++                 $ctext insert "branch -2c" "es"
 ++                 $ctext conf -state disabled
 ++             }
 ++         }
 ++     }
 ++    }
 ++    if {[incr tagphase] <= 2} {
 ++     after idle dispnexttag
 ++    }
 ++}
 ++
 ++proc make_secsel {l} {
 ++    global linehtag linentag linedtag canv canv2 canv3
 ++
 ++    if {![info exists linehtag($l)]} return
 ++    $canv delete secsel
 ++    set t [eval $canv create rect [$canv bbox $linehtag($l)] -outline {{}} \
 ++            -tags secsel -fill [$canv cget -selectbackground]]
 ++    $canv lower $t
 ++    $canv2 delete secsel
 ++    set t [eval $canv2 create rect [$canv2 bbox $linentag($l)] -outline {{}} \
 ++            -tags secsel -fill [$canv2 cget -selectbackground]]
 ++    $canv2 lower $t
 ++    $canv3 delete secsel
 ++    set t [eval $canv3 create rect [$canv3 bbox $linedtag($l)] -outline {{}} \
 ++            -tags secsel -fill [$canv3 cget -selectbackground]]
 ++    $canv3 lower $t
 ++}
 ++
 ++proc selectline {l isnew} {
 ++    global canv ctext commitinfo selectedline
 ++    global canvy0 linespc parents children curview
 ++    global currentid sha1entry
 ++    global commentend idtags linknum
 ++    global mergemax numcommits pending_select
 ++    global cmitmode showneartags allcommits
 ++    global targetrow targetid lastscrollrows
 ++    global autoselect
 ++
 ++    catch {unset pending_select}
 ++    $canv delete hover
 ++    normalline
 ++    unsel_reflist
 ++    stopfinding
 ++    if {$l < 0 || $l >= $numcommits} return
 ++    set id [commitonrow $l]
 ++    set targetid $id
 ++    set targetrow $l
 ++    set selectedline $l
 ++    set currentid $id
 ++    if {$lastscrollrows < $numcommits} {
 ++     setcanvscroll
 ++    }
 ++
 ++    set y [expr {$canvy0 + $l * $linespc}]
 ++    set ymax [lindex [$canv cget -scrollregion] 3]
 ++    set ytop [expr {$y - $linespc - 1}]
 ++    set ybot [expr {$y + $linespc + 1}]
 ++    set wnow [$canv yview]
 ++    set wtop [expr {[lindex $wnow 0] * $ymax}]
 ++    set wbot [expr {[lindex $wnow 1] * $ymax}]
 ++    set wh [expr {$wbot - $wtop}]
 ++    set newtop $wtop
 ++    if {$ytop < $wtop} {
 ++     if {$ybot < $wtop} {
 ++         set newtop [expr {$y - $wh / 2.0}]
 ++     } else {
 ++         set newtop $ytop
 ++         if {$newtop > $wtop - $linespc} {
 ++             set newtop [expr {$wtop - $linespc}]
 ++         }
 ++     }
 ++    } elseif {$ybot > $wbot} {
 ++     if {$ytop > $wbot} {
 ++         set newtop [expr {$y - $wh / 2.0}]
 ++     } else {
 ++         set newtop [expr {$ybot - $wh}]
 ++         if {$newtop < $wtop + $linespc} {
 ++             set newtop [expr {$wtop + $linespc}]
 ++         }
 ++     }
 ++    }
 ++    if {$newtop != $wtop} {
 ++     if {$newtop < 0} {
 ++         set newtop 0
 ++     }
 ++     allcanvs yview moveto [expr {$newtop * 1.0 / $ymax}]
 ++     drawvisible
 ++    }
 ++
 ++    make_secsel $l
 ++
 ++    if {$isnew} {
 ++     addtohistory [list selbyid $id]
 ++    }
 ++
 ++    $sha1entry delete 0 end
 ++    $sha1entry insert 0 $id
 ++    if {$autoselect} {
 ++     $sha1entry selection from 0
 ++     $sha1entry selection to end
 ++    }
 ++    rhighlight_sel $id
 ++
 ++    $ctext conf -state normal
 ++    clear_ctext
 ++    set linknum 0
 ++    if {![info exists commitinfo($id)]} {
 ++     getcommit $id
 ++    }
 ++    set info $commitinfo($id)
 ++    set date [formatdate [lindex $info 2]]
 ++    $ctext insert end "[mc "Author"]: [lindex $info 1]  $date\n"
 ++    set date [formatdate [lindex $info 4]]
 ++    $ctext insert end "[mc "Committer"]: [lindex $info 3]  $date\n"
 ++    if {[info exists idtags($id)]} {
 ++     $ctext insert end [mc "Tags:"]
 ++     foreach tag $idtags($id) {
 ++         $ctext insert end " $tag"
 ++     }
 ++     $ctext insert end "\n"
 ++    }
 ++
 ++    set headers {}
 ++    set olds $parents($curview,$id)
 ++    if {[llength $olds] > 1} {
 ++     set np 0
 ++     foreach p $olds {
 ++         if {$np >= $mergemax} {
 ++             set tag mmax
 ++         } else {
 ++             set tag m$np
 ++         }
 ++         $ctext insert end "[mc "Parent"]: " $tag
 ++         appendwithlinks [commit_descriptor $p] {}
 ++         incr np
 ++     }
 ++    } else {
 ++     foreach p $olds {
 ++         append headers "[mc "Parent"]: [commit_descriptor $p]"
 ++     }
 ++    }
 ++
 ++    foreach c $children($curview,$id) {
 ++     append headers "[mc "Child"]:  [commit_descriptor $c]"
 ++    }
 ++
 ++    # make anything that looks like a SHA1 ID be a clickable link
 ++    appendwithlinks $headers {}
 ++    if {$showneartags} {
 ++     if {![info exists allcommits]} {
 ++         getallcommits
 ++     }
 ++     $ctext insert end "[mc "Branch"]: "
 ++     $ctext mark set branch "end -1c"
 ++     $ctext mark gravity branch left
 ++     $ctext insert end "\n[mc "Follows"]: "
 ++     $ctext mark set follows "end -1c"
 ++     $ctext mark gravity follows left
 ++     $ctext insert end "\n[mc "Precedes"]: "
 ++     $ctext mark set precedes "end -1c"
 ++     $ctext mark gravity precedes left
 ++     $ctext insert end "\n"
 ++     dispneartags 1
 ++    }
 ++    $ctext insert end "\n"
 ++    set comment [lindex $info 5]
 ++    if {[string first "\r" $comment] >= 0} {
 ++     set comment [string map {"\r" "\n    "} $comment]
 ++    }
 ++    appendwithlinks $comment {comment}
 ++
 ++    $ctext tag remove found 1.0 end
 ++    $ctext conf -state disabled
 ++    set commentend [$ctext index "end - 1c"]
 ++
 ++    init_flist [mc "Comments"]
 ++    if {$cmitmode eq "tree"} {
 ++     gettree $id
 ++    } elseif {[llength $olds] <= 1} {
 ++     startdiff $id
 ++    } else {
 ++     mergediff $id
 ++    }
 ++}
 ++
 ++proc selfirstline {} {
 ++    unmarkmatches
 ++    selectline 0 1
 ++}
 ++
 ++proc sellastline {} {
 ++    global numcommits
 ++    unmarkmatches
 ++    set l [expr {$numcommits - 1}]
 ++    selectline $l 1
 ++}
 ++
 ++proc selnextline {dir} {
 ++    global selectedline
 ++    focus .
 ++    if {$selectedline eq {}} return
 ++    set l [expr {$selectedline + $dir}]
 ++    unmarkmatches
 ++    selectline $l 1
 ++}
 ++
 ++proc selnextpage {dir} {
 ++    global canv linespc selectedline numcommits
 ++
 ++    set lpp [expr {([winfo height $canv] - 2) / $linespc}]
 ++    if {$lpp < 1} {
 ++     set lpp 1
 ++    }
 ++    allcanvs yview scroll [expr {$dir * $lpp}] units
 ++    drawvisible
 ++    if {$selectedline eq {}} return
 ++    set l [expr {$selectedline + $dir * $lpp}]
 ++    if {$l < 0} {
 ++     set l 0
 ++    } elseif {$l >= $numcommits} {
 ++        set l [expr $numcommits - 1]
 ++    }
 ++    unmarkmatches
 ++    selectline $l 1
 ++}
 ++
 ++proc unselectline {} {
 ++    global selectedline currentid
 ++
 ++    set selectedline {}
 ++    catch {unset currentid}
 ++    allcanvs delete secsel
 ++    rhighlight_none
 ++}
 ++
 ++proc reselectline {} {
 ++    global selectedline
 ++
 ++    if {$selectedline ne {}} {
 ++     selectline $selectedline 0
 ++    }
 ++}
 ++
 ++proc addtohistory {cmd} {
 ++    global history historyindex curview
 ++
 ++    set elt [list $curview $cmd]
 ++    if {$historyindex > 0
 ++     && [lindex $history [expr {$historyindex - 1}]] == $elt} {
 ++     return
 ++    }
 ++
 ++    if {$historyindex < [llength $history]} {
 ++     set history [lreplace $history $historyindex end $elt]
 ++    } else {
 ++     lappend history $elt
 ++    }
 ++    incr historyindex
 ++    if {$historyindex > 1} {
 ++     .tf.bar.leftbut conf -state normal
 ++    } else {
 ++     .tf.bar.leftbut conf -state disabled
 ++    }
 ++    .tf.bar.rightbut conf -state disabled
 ++}
 ++
 ++proc godo {elt} {
 ++    global curview
 ++
 ++    set view [lindex $elt 0]
 ++    set cmd [lindex $elt 1]
 ++    if {$curview != $view} {
 ++     showview $view
 ++    }
 ++    eval $cmd
 ++}
 ++
 ++proc goback {} {
 ++    global history historyindex
 ++    focus .
 ++
 ++    if {$historyindex > 1} {
 ++     incr historyindex -1
 ++     godo [lindex $history [expr {$historyindex - 1}]]
 ++     .tf.bar.rightbut conf -state normal
 ++    }
 ++    if {$historyindex <= 1} {
 ++     .tf.bar.leftbut conf -state disabled
 ++    }
 ++}
 ++
 ++proc goforw {} {
 ++    global history historyindex
 ++    focus .
 ++
 ++    if {$historyindex < [llength $history]} {
 ++     set cmd [lindex $history $historyindex]
 ++     incr historyindex
 ++     godo $cmd
 ++     .tf.bar.leftbut conf -state normal
 ++    }
 ++    if {$historyindex >= [llength $history]} {
 ++     .tf.bar.rightbut conf -state disabled
 ++    }
 ++}
 ++
 ++proc gettree {id} {
 ++    global treefilelist treeidlist diffids diffmergeid treepending
 ++    global nullid nullid2
 ++
 ++    set diffids $id
 ++    catch {unset diffmergeid}
 ++    if {![info exists treefilelist($id)]} {
 ++     if {![info exists treepending]} {
 ++         if {$id eq $nullid} {
 ++             set cmd [list | git ls-files]
 ++         } elseif {$id eq $nullid2} {
 ++             set cmd [list | git ls-files --stage -t]
 ++         } else {
 ++             set cmd [list | git ls-tree -r $id]
 ++         }
 ++         if {[catch {set gtf [open $cmd r]}]} {
 ++             return
 ++         }
 ++         set treepending $id
 ++         set treefilelist($id) {}
 ++         set treeidlist($id) {}
 ++         fconfigure $gtf -blocking 0
 ++         filerun $gtf [list gettreeline $gtf $id]
 ++     }
 ++    } else {
 ++     setfilelist $id
 ++    }
 ++}
 ++
 ++proc gettreeline {gtf id} {
 ++    global treefilelist treeidlist treepending cmitmode diffids nullid nullid2
 ++
 ++    set nl 0
 ++    while {[incr nl] <= 1000 && [gets $gtf line] >= 0} {
 ++     if {$diffids eq $nullid} {
 ++         set fname $line
 ++     } else {
 ++         set i [string first "\t" $line]
 ++         if {$i < 0} continue
 ++         set fname [string range $line [expr {$i+1}] end]
 ++         set line [string range $line 0 [expr {$i-1}]]
 ++         if {$diffids ne $nullid2 && [lindex $line 1] ne "blob"} continue
 ++         set sha1 [lindex $line 2]
 ++         if {[string index $fname 0] eq "\""} {
 ++             set fname [lindex $fname 0]
 ++         }
 ++         lappend treeidlist($id) $sha1
 ++     }
 ++     lappend treefilelist($id) $fname
 ++    }
 ++    if {![eof $gtf]} {
 ++     return [expr {$nl >= 1000? 2: 1}]
 ++    }
 ++    close $gtf
 ++    unset treepending
 ++    if {$cmitmode ne "tree"} {
 ++     if {![info exists diffmergeid]} {
 ++         gettreediffs $diffids
 ++     }
 ++    } elseif {$id ne $diffids} {
 ++     gettree $diffids
 ++    } else {
 ++     setfilelist $id
 ++    }
 ++    return 0
 ++}
 ++
 ++proc showfile {f} {
 ++    global treefilelist treeidlist diffids nullid nullid2
 ++    global ctext commentend
 ++
 ++    set i [lsearch -exact $treefilelist($diffids) $f]
 ++    if {$i < 0} {
 ++     puts "oops, $f not in list for id $diffids"
 ++     return
 ++    }
 ++    if {$diffids eq $nullid} {
 ++     if {[catch {set bf [open $f r]} err]} {
 ++         puts "oops, can't read $f: $err"
 ++         return
 ++     }
 ++    } else {
 ++     set blob [lindex $treeidlist($diffids) $i]
 ++     if {[catch {set bf [open [concat | git cat-file blob $blob] r]} err]} {
 ++         puts "oops, error reading blob $blob: $err"
 ++         return
 ++     }
 ++    }
 ++    fconfigure $bf -blocking 0
 ++    filerun $bf [list getblobline $bf $diffids]
 ++    $ctext config -state normal
 ++    clear_ctext $commentend
 ++    $ctext insert end "\n"
 ++    $ctext insert end "$f\n" filesep
 ++    $ctext config -state disabled
 ++    $ctext yview $commentend
 ++    settabs 0
 ++}
 ++
 ++proc getblobline {bf id} {
 ++    global diffids cmitmode ctext
 ++
 ++    if {$id ne $diffids || $cmitmode ne "tree"} {
 ++     catch {close $bf}
 ++     return 0
 ++    }
 ++    $ctext config -state normal
 ++    set nl 0
 ++    while {[incr nl] <= 1000 && [gets $bf line] >= 0} {
 ++     $ctext insert end "$line\n"
 ++    }
 ++    if {[eof $bf]} {
 ++     # delete last newline
 ++     $ctext delete "end - 2c" "end - 1c"
 ++     close $bf
 ++     return 0
 ++    }
 ++    $ctext config -state disabled
 ++    return [expr {$nl >= 1000? 2: 1}]
 ++}
 ++
 ++proc mergediff {id} {
 ++    global diffmergeid mdifffd
 ++    global diffids
 ++    global parents
 ++    global diffcontext
 ++    global limitdiffs vfilelimit curview
 ++
 ++    set diffmergeid $id
 ++    set diffids $id
 ++    # this doesn't seem to actually affect anything...
 ++    set cmd [concat | git diff-tree --no-commit-id --cc -U$diffcontext $id]
 ++    if {$limitdiffs && $vfilelimit($curview) ne {}} {
 ++     set cmd [concat $cmd -- $vfilelimit($curview)]
 ++    }
 ++    if {[catch {set mdf [open $cmd r]} err]} {
 ++     error_popup "[mc "Error getting merge diffs:"] $err"
 ++     return
 ++    }
 ++    fconfigure $mdf -blocking 0
 ++    set mdifffd($id) $mdf
 ++    set np [llength $parents($curview,$id)]
 ++    settabs $np
 ++    filerun $mdf [list getmergediffline $mdf $id $np]
 ++}
 ++
 ++proc getmergediffline {mdf id np} {
 ++    global diffmergeid ctext cflist mergemax
 ++    global difffilestart mdifffd
 ++
 ++    $ctext conf -state normal
 ++    set nr 0
 ++    while {[incr nr] <= 1000 && [gets $mdf line] >= 0} {
 ++     if {![info exists diffmergeid] || $id != $diffmergeid
 ++         || $mdf != $mdifffd($id)} {
 ++         close $mdf
 ++         return 0
 ++     }
 ++     if {[regexp {^diff --cc (.*)} $line match fname]} {
 ++         # start of a new file
 ++         $ctext insert end "\n"
 ++         set here [$ctext index "end - 1c"]
 ++         lappend difffilestart $here
 ++         add_flist [list $fname]
 ++         set l [expr {(78 - [string length $fname]) / 2}]
 ++         set pad [string range "----------------------------------------" 1 $l]
 ++         $ctext insert end "$pad $fname $pad\n" filesep
 ++     } elseif {[regexp {^@@} $line]} {
 ++         $ctext insert end "$line\n" hunksep
 ++     } elseif {[regexp {^[0-9a-f]{40}$} $line] || [regexp {^index} $line]} {
 ++         # do nothing
 ++     } else {
 ++         # parse the prefix - one ' ', '-' or '+' for each parent
 ++         set spaces {}
 ++         set minuses {}
 ++         set pluses {}
 ++         set isbad 0
 ++         for {set j 0} {$j < $np} {incr j} {
 ++             set c [string range $line $j $j]
 ++             if {$c == " "} {
 ++                 lappend spaces $j
 ++             } elseif {$c == "-"} {
 ++                 lappend minuses $j
 ++             } elseif {$c == "+"} {
 ++                 lappend pluses $j
 ++             } else {
 ++                 set isbad 1
 ++                 break
 ++             }
 ++         }
 ++         set tags {}
 ++         set num {}
 ++         if {!$isbad && $minuses ne {} && $pluses eq {}} {
 ++             # line doesn't appear in result, parents in $minuses have the line
 ++             set num [lindex $minuses 0]
 ++         } elseif {!$isbad && $pluses ne {} && $minuses eq {}} {
 ++             # line appears in result, parents in $pluses don't have the line
 ++             lappend tags mresult
 ++             set num [lindex $spaces 0]
 ++         }
 ++         if {$num ne {}} {
 ++             if {$num >= $mergemax} {
 ++                 set num "max"
 ++             }
 ++             lappend tags m$num
 ++         }
 ++         $ctext insert end "$line\n" $tags
 ++     }
 ++    }
 ++    $ctext conf -state disabled
 ++    if {[eof $mdf]} {
 ++     close $mdf
 ++     return 0
 ++    }
 ++    return [expr {$nr >= 1000? 2: 1}]
 ++}
 ++
 ++proc startdiff {ids} {
 ++    global treediffs diffids treepending diffmergeid nullid nullid2
 ++
 ++    settabs 1
 ++    set diffids $ids
 ++    catch {unset diffmergeid}
 ++    if {![info exists treediffs($ids)] ||
 ++     [lsearch -exact $ids $nullid] >= 0 ||
 ++     [lsearch -exact $ids $nullid2] >= 0} {
 ++     if {![info exists treepending]} {
 ++         gettreediffs $ids
 ++     }
 ++    } else {
 ++     addtocflist $ids
 ++    }
 ++}
 ++
 ++proc path_filter {filter name} {
 ++    foreach p $filter {
 ++     set l [string length $p]
 ++     if {[string index $p end] eq "/"} {
 ++         if {[string compare -length $l $p $name] == 0} {
 ++             return 1
 ++         }
 ++     } else {
 ++         if {[string compare -length $l $p $name] == 0 &&
 ++             ([string length $name] == $l ||
 ++              [string index $name $l] eq "/")} {
 ++             return 1
 ++         }
 ++     }
 ++    }
 ++    return 0
 ++}
 ++
 ++proc addtocflist {ids} {
 ++    global treediffs
 ++
 ++    add_flist $treediffs($ids)
 ++    getblobdiffs $ids
 ++}
 ++
 ++proc diffcmd {ids flags} {
 ++    global nullid nullid2
 ++
 ++    set i [lsearch -exact $ids $nullid]
 ++    set j [lsearch -exact $ids $nullid2]
 ++    if {$i >= 0} {
 ++     if {[llength $ids] > 1 && $j < 0} {
 ++         # comparing working directory with some specific revision
 ++         set cmd [concat | git diff-index $flags]
 ++         if {$i == 0} {
 ++             lappend cmd -R [lindex $ids 1]
 ++         } else {
 ++             lappend cmd [lindex $ids 0]
 ++         }
 ++     } else {
 ++         # comparing working directory with index
 ++         set cmd [concat | git diff-files $flags]
 ++         if {$j == 1} {
 ++             lappend cmd -R
 ++         }
 ++     }
 ++    } elseif {$j >= 0} {
 ++     set cmd [concat | git diff-index --cached $flags]
 ++     if {[llength $ids] > 1} {
 ++         # comparing index with specific revision
 ++         if {$i == 0} {
 ++             lappend cmd -R [lindex $ids 1]
 ++         } else {
 ++             lappend cmd [lindex $ids 0]
 ++         }
 ++     } else {
 ++         # comparing index with HEAD
 ++         lappend cmd HEAD
 ++     }
 ++    } else {
 ++     set cmd [concat | git diff-tree -r $flags $ids]
 ++    }
 ++    return $cmd
 ++}
 ++
 ++proc gettreediffs {ids} {
 ++    global treediff treepending
 ++
 ++    if {[catch {set gdtf [open [diffcmd $ids {--no-commit-id}] r]}]} return
 ++
 ++    set treepending $ids
 ++    set treediff {}
 ++    fconfigure $gdtf -blocking 0
 ++    filerun $gdtf [list gettreediffline $gdtf $ids]
 ++}
 ++
 ++proc gettreediffline {gdtf ids} {
 ++    global treediff treediffs treepending diffids diffmergeid
 ++    global cmitmode vfilelimit curview limitdiffs
 ++
 ++    set nr 0
 ++    while {[incr nr] <= 1000 && [gets $gdtf line] >= 0} {
 ++     set i [string first "\t" $line]
 ++     if {$i >= 0} {
 ++         set file [string range $line [expr {$i+1}] end]
 ++         if {[string index $file 0] eq "\""} {
 ++             set file [lindex $file 0]
 ++         }
 ++         lappend treediff $file
 ++     }
 ++    }
 ++    if {![eof $gdtf]} {
 ++     return [expr {$nr >= 1000? 2: 1}]
 ++    }
 ++    close $gdtf
 ++    if {$limitdiffs && $vfilelimit($curview) ne {}} {
 ++     set flist {}
 ++     foreach f $treediff {
 ++         if {[path_filter $vfilelimit($curview) $f]} {
 ++             lappend flist $f
 ++         }
 ++     }
 ++     set treediffs($ids) $flist
 ++    } else {
 ++     set treediffs($ids) $treediff
 ++    }
 ++    unset treepending
 ++    if {$cmitmode eq "tree"} {
 ++     gettree $diffids
 ++    } elseif {$ids != $diffids} {
 ++     if {![info exists diffmergeid]} {
 ++         gettreediffs $diffids
 ++     }
 ++    } else {
 ++     addtocflist $ids
 ++    }
 ++    return 0
 ++}
 ++
 ++# empty string or positive integer
 ++proc diffcontextvalidate {v} {
 ++    return [regexp {^(|[1-9][0-9]*)$} $v]
 ++}
 ++
 ++proc diffcontextchange {n1 n2 op} {
 ++    global diffcontextstring diffcontext
 ++
 ++    if {[string is integer -strict $diffcontextstring]} {
 ++     if {$diffcontextstring > 0} {
 ++         set diffcontext $diffcontextstring
 ++         reselectline
 ++     }
 ++    }
 ++}
 ++
 ++proc changeignorespace {} {
 ++    reselectline
 ++}
 ++
 ++proc getblobdiffs {ids} {
 ++    global blobdifffd diffids env
 ++    global diffinhdr treediffs
 ++    global diffcontext
 ++    global ignorespace
 ++    global limitdiffs vfilelimit curview
 ++
 ++    set cmd [diffcmd $ids "-p -C --no-commit-id -U$diffcontext"]
 ++    if {$ignorespace} {
 ++     append cmd " -w"
 ++    }
 ++    if {$limitdiffs && $vfilelimit($curview) ne {}} {
 ++     set cmd [concat $cmd -- $vfilelimit($curview)]
 ++    }
 ++    if {[catch {set bdf [open $cmd r]} err]} {
 ++     puts "error getting diffs: $err"
 ++     return
 ++    }
 ++    set diffinhdr 0
 ++    fconfigure $bdf -blocking 0
 ++    set blobdifffd($ids) $bdf
 ++    filerun $bdf [list getblobdiffline $bdf $diffids]
 ++}
 ++
 ++proc setinlist {var i val} {
 ++    global $var
 ++
 ++    while {[llength [set $var]] < $i} {
 ++     lappend $var {}
 ++    }
 ++    if {[llength [set $var]] == $i} {
 ++     lappend $var $val
 ++    } else {
 ++     lset $var $i $val
 ++    }
 ++}
 ++
 ++proc makediffhdr {fname ids} {
 ++    global ctext curdiffstart treediffs
 ++
 ++    set i [lsearch -exact $treediffs($ids) $fname]
 ++    if {$i >= 0} {
 ++     setinlist difffilestart $i $curdiffstart
 ++    }
 ++    set l [expr {(78 - [string length $fname]) / 2}]
 ++    set pad [string range "----------------------------------------" 1 $l]
 ++    $ctext insert $curdiffstart "$pad $fname $pad" filesep
 ++}
 ++
 ++proc getblobdiffline {bdf ids} {
 ++    global diffids blobdifffd ctext curdiffstart
 ++    global diffnexthead diffnextnote difffilestart
 ++    global diffinhdr treediffs
 ++
 ++    set nr 0
 ++    $ctext conf -state normal
 ++    while {[incr nr] <= 1000 && [gets $bdf line] >= 0} {
 ++     if {$ids != $diffids || $bdf != $blobdifffd($ids)} {
 ++         close $bdf
 ++         return 0
 ++     }
 ++     if {![string compare -length 11 "diff --git " $line]} {
 ++         # trim off "diff --git "
 ++         set line [string range $line 11 end]
 ++         set diffinhdr 1
 ++         # start of a new file
 ++         $ctext insert end "\n"
 ++         set curdiffstart [$ctext index "end - 1c"]
 ++         $ctext insert end "\n" filesep
 ++         # If the name hasn't changed the length will be odd,
 ++         # the middle char will be a space, and the two bits either
 ++         # side will be a/name and b/name, or "a/name" and "b/name".
 ++         # If the name has changed we'll get "rename from" and
 ++         # "rename to" or "copy from" and "copy to" lines following this,
 ++         # and we'll use them to get the filenames.
 ++         # This complexity is necessary because spaces in the filename(s)
 ++         # don't get escaped.
 ++         set l [string length $line]
 ++         set i [expr {$l / 2}]
 ++         if {!(($l & 1) && [string index $line $i] eq " " &&
 ++               [string range $line 2 [expr {$i - 1}]] eq \
 ++                   [string range $line [expr {$i + 3}] end])} {
 ++             continue
 ++         }
 ++         # unescape if quoted and chop off the a/ from the front
 ++         if {[string index $line 0] eq "\""} {
 ++             set fname [string range [lindex $line 0] 2 end]
 ++         } else {
 ++             set fname [string range $line 2 [expr {$i - 1}]]
 ++         }
 ++         makediffhdr $fname $ids
 ++
 ++     } elseif {[regexp {^@@ -([0-9]+)(,[0-9]+)? \+([0-9]+)(,[0-9]+)? @@(.*)} \
 ++                    $line match f1l f1c f2l f2c rest]} {
 ++         $ctext insert end "$line\n" hunksep
 ++         set diffinhdr 0
 ++
 ++     } elseif {$diffinhdr} {
 ++         if {![string compare -length 12 "rename from " $line]} {
 ++             set fname [string range $line [expr 6 + [string first " from " $line] ] end]
 ++             if {[string index $fname 0] eq "\""} {
 ++                 set fname [lindex $fname 0]
 ++             }
 ++             set i [lsearch -exact $treediffs($ids) $fname]
 ++             if {$i >= 0} {
 ++                 setinlist difffilestart $i $curdiffstart
 ++             }
 ++         } elseif {![string compare -length 10 $line "rename to "] ||
 ++                   ![string compare -length 8 $line "copy to "]} {
 ++             set fname [string range $line [expr 4 + [string first " to " $line] ] end]
 ++             if {[string index $fname 0] eq "\""} {
 ++                 set fname [lindex $fname 0]
 ++             }
 ++             makediffhdr $fname $ids
 ++         } elseif {[string compare -length 3 $line "---"] == 0} {
 ++             # do nothing
 ++             continue
 ++         } elseif {[string compare -length 3 $line "+++"] == 0} {
 ++             set diffinhdr 0
 ++             continue
 ++         }
 ++         $ctext insert end "$line\n" filesep
 ++
 ++     } else {
 ++         set x [string range $line 0 0]
 ++         if {$x == "-" || $x == "+"} {
 ++             set tag [expr {$x == "+"}]
 ++             $ctext insert end "$line\n" d$tag
 ++         } elseif {$x == " "} {
 ++             $ctext insert end "$line\n"
 ++         } else {
 ++             # "\ No newline at end of file",
 ++             # or something else we don't recognize
 ++             $ctext insert end "$line\n" hunksep
 ++         }
 ++     }
 ++    }
 ++    $ctext conf -state disabled
 ++    if {[eof $bdf]} {
 ++     close $bdf
 ++     return 0
 ++    }
 ++    return [expr {$nr >= 1000? 2: 1}]
 ++}
 ++
 ++proc changediffdisp {} {
 ++    global ctext diffelide
 ++
 ++    $ctext tag conf d0 -elide [lindex $diffelide 0]
 ++    $ctext tag conf d1 -elide [lindex $diffelide 1]
 ++}
 ++
 ++proc highlightfile {loc cline} {
 ++    global ctext cflist cflist_top
 ++
 ++    $ctext yview $loc
 ++    $cflist tag remove highlight $cflist_top.0 "$cflist_top.0 lineend"
 ++    $cflist tag add highlight $cline.0 "$cline.0 lineend"
 ++    $cflist see $cline.0
 ++    set cflist_top $cline
 ++}
 ++
 ++proc prevfile {} {
 ++    global difffilestart ctext cmitmode
 ++
 ++    if {$cmitmode eq "tree"} return
 ++    set prev 0.0
 ++    set prevline 1
 ++    set here [$ctext index @0,0]
 ++    foreach loc $difffilestart {
 ++     if {[$ctext compare $loc >= $here]} {
 ++         highlightfile $prev $prevline
 ++         return
 ++     }
 ++     set prev $loc
 ++     incr prevline
 ++    }
 ++    highlightfile $prev $prevline
 ++}
 ++
 ++proc nextfile {} {
 ++    global difffilestart ctext cmitmode
 ++
 ++    if {$cmitmode eq "tree"} return
 ++    set here [$ctext index @0,0]
 ++    set line 1
 ++    foreach loc $difffilestart {
 ++     incr line
 ++     if {[$ctext compare $loc > $here]} {
 ++         highlightfile $loc $line
 ++         return
 ++     }
 ++    }
 ++}
 ++
 ++proc clear_ctext {{first 1.0}} {
 ++    global ctext smarktop smarkbot
 ++    global pendinglinks
 ++
 ++    set l [lindex [split $first .] 0]
 ++    if {![info exists smarktop] || [$ctext compare $first < $smarktop.0]} {
 ++     set smarktop $l
 ++    }
 ++    if {![info exists smarkbot] || [$ctext compare $first < $smarkbot.0]} {
 ++     set smarkbot $l
 ++    }
 ++    $ctext delete $first end
 ++    if {$first eq "1.0"} {
 ++     catch {unset pendinglinks}
 ++    }
 ++}
 ++
 ++proc settabs {{firstab {}}} {
 ++    global firsttabstop tabstop ctext have_tk85
 ++
 ++    if {$firstab ne {} && $have_tk85} {
 ++     set firsttabstop $firstab
 ++    }
 ++    set w [font measure textfont "0"]
 ++    if {$firsttabstop != 0} {
 ++     $ctext conf -tabs [list [expr {($firsttabstop + $tabstop) * $w}] \
 ++                            [expr {($firsttabstop + 2 * $tabstop) * $w}]]
 ++    } elseif {$have_tk85 || $tabstop != 8} {
 ++     $ctext conf -tabs [expr {$tabstop * $w}]
 ++    } else {
 ++     $ctext conf -tabs {}
 ++    }
 ++}
 ++
 ++proc incrsearch {name ix op} {
 ++    global ctext searchstring searchdirn
 ++
 ++    $ctext tag remove found 1.0 end
 ++    if {[catch {$ctext index anchor}]} {
 ++     # no anchor set, use start of selection, or of visible area
 ++     set sel [$ctext tag ranges sel]
 ++     if {$sel ne {}} {
 ++         $ctext mark set anchor [lindex $sel 0]
 ++     } elseif {$searchdirn eq "-forwards"} {
 ++         $ctext mark set anchor @0,0
 ++     } else {
 ++         $ctext mark set anchor @0,[winfo height $ctext]
 ++     }
 ++    }
 ++    if {$searchstring ne {}} {
 ++     set here [$ctext search $searchdirn -- $searchstring anchor]
 ++     if {$here ne {}} {
 ++         $ctext see $here
 ++     }
 ++     searchmarkvisible 1
 ++    }
 ++}
 ++
 ++proc dosearch {} {
 ++    global sstring ctext searchstring searchdirn
 ++
 ++    focus $sstring
 ++    $sstring icursor end
 ++    set searchdirn -forwards
 ++    if {$searchstring ne {}} {
 ++     set sel [$ctext tag ranges sel]
 ++     if {$sel ne {}} {
 ++         set start "[lindex $sel 0] + 1c"
 ++     } elseif {[catch {set start [$ctext index anchor]}]} {
 ++         set start "@0,0"
 ++     }
 ++     set match [$ctext search -count mlen -- $searchstring $start]
 ++     $ctext tag remove sel 1.0 end
 ++     if {$match eq {}} {
 ++         bell
 ++         return
 ++     }
 ++     $ctext see $match
 ++     set mend "$match + $mlen c"
 ++     $ctext tag add sel $match $mend
 ++     $ctext mark unset anchor
 ++    }
 ++}
 ++
 ++proc dosearchback {} {
 ++    global sstring ctext searchstring searchdirn
 ++
 ++    focus $sstring
 ++    $sstring icursor end
 ++    set searchdirn -backwards
 ++    if {$searchstring ne {}} {
 ++     set sel [$ctext tag ranges sel]
 ++     if {$sel ne {}} {
 ++         set start [lindex $sel 0]
 ++     } elseif {[catch {set start [$ctext index anchor]}]} {
 ++         set start @0,[winfo height $ctext]
 ++     }
 ++     set match [$ctext search -backwards -count ml -- $searchstring $start]
 ++     $ctext tag remove sel 1.0 end
 ++     if {$match eq {}} {
 ++         bell
 ++         return
 ++     }
 ++     $ctext see $match
 ++     set mend "$match + $ml c"
 ++     $ctext tag add sel $match $mend
 ++     $ctext mark unset anchor
 ++    }
 ++}
 ++
 ++proc searchmark {first last} {
 ++    global ctext searchstring
 ++
 ++    set mend $first.0
 ++    while {1} {
 ++     set match [$ctext search -count mlen -- $searchstring $mend $last.end]
 ++     if {$match eq {}} break
 ++     set mend "$match + $mlen c"
 ++     $ctext tag add found $match $mend
 ++    }
 ++}
 ++
 ++proc searchmarkvisible {doall} {
 ++    global ctext smarktop smarkbot
 ++
 ++    set topline [lindex [split [$ctext index @0,0] .] 0]
 ++    set botline [lindex [split [$ctext index @0,[winfo height $ctext]] .] 0]
 ++    if {$doall || $botline < $smarktop || $topline > $smarkbot} {
 ++     # no overlap with previous
 ++     searchmark $topline $botline
 ++     set smarktop $topline
 ++     set smarkbot $botline
 ++    } else {
 ++     if {$topline < $smarktop} {
 ++         searchmark $topline [expr {$smarktop-1}]
 ++         set smarktop $topline
 ++     }
 ++     if {$botline > $smarkbot} {
 ++         searchmark [expr {$smarkbot+1}] $botline
 ++         set smarkbot $botline
 ++     }
 ++    }
 ++}
 ++
 ++proc scrolltext {f0 f1} {
 ++    global searchstring
 ++
 ++    .bleft.bottom.sb set $f0 $f1
 ++    if {$searchstring ne {}} {
 ++     searchmarkvisible 0
 ++    }
 ++}
 ++
 ++proc setcoords {} {
 ++    global linespc charspc canvx0 canvy0
 ++    global xspc1 xspc2 lthickness
 ++
 ++    set linespc [font metrics mainfont -linespace]
 ++    set charspc [font measure mainfont "m"]
 ++    set canvy0 [expr {int(3 + 0.5 * $linespc)}]
 ++    set canvx0 [expr {int(3 + 0.5 * $linespc)}]
 ++    set lthickness [expr {int($linespc / 9) + 1}]
 ++    set xspc1(0) $linespc
 ++    set xspc2 $linespc
 ++}
 ++
 ++proc redisplay {} {
 ++    global canv
 ++    global selectedline
 ++
 ++    set ymax [lindex [$canv cget -scrollregion] 3]
 ++    if {$ymax eq {} || $ymax == 0} return
 ++    set span [$canv yview]
 ++    clear_display
 ++    setcanvscroll
 ++    allcanvs yview moveto [lindex $span 0]
 ++    drawvisible
 ++    if {$selectedline ne {}} {
 ++     selectline $selectedline 0
 ++     allcanvs yview moveto [lindex $span 0]
 ++    }
 ++}
 ++
 ++proc parsefont {f n} {
 ++    global fontattr
 ++
 ++    set fontattr($f,family) [lindex $n 0]
 ++    set s [lindex $n 1]
 ++    if {$s eq {} || $s == 0} {
 ++     set s 10
 ++    } elseif {$s < 0} {
 ++     set s [expr {int(-$s / [winfo fpixels . 1p] + 0.5)}]
 ++    }
 ++    set fontattr($f,size) $s
 ++    set fontattr($f,weight) normal
 ++    set fontattr($f,slant) roman
 ++    foreach style [lrange $n 2 end] {
 ++     switch -- $style {
 ++         "normal" -
 ++         "bold"   {set fontattr($f,weight) $style}
 ++         "roman" -
 ++         "italic" {set fontattr($f,slant) $style}
 ++     }
 ++    }
 ++}
 ++
 ++proc fontflags {f {isbold 0}} {
 ++    global fontattr
 ++
 ++    return [list -family $fontattr($f,family) -size $fontattr($f,size) \
 ++             -weight [expr {$isbold? "bold": $fontattr($f,weight)}] \
 ++             -slant $fontattr($f,slant)]
 ++}
 ++
 ++proc fontname {f} {
 ++    global fontattr
 ++
 ++    set n [list $fontattr($f,family) $fontattr($f,size)]
 ++    if {$fontattr($f,weight) eq "bold"} {
 ++     lappend n "bold"
 ++    }
 ++    if {$fontattr($f,slant) eq "italic"} {
 ++     lappend n "italic"
 ++    }
 ++    return $n
 ++}
 ++
 ++proc incrfont {inc} {
 ++    global mainfont textfont ctext canv cflist showrefstop
 ++    global stopped entries fontattr
 ++
 ++    unmarkmatches
 ++    set s $fontattr(mainfont,size)
 ++    incr s $inc
 ++    if {$s < 1} {
 ++     set s 1
 ++    }
 ++    set fontattr(mainfont,size) $s
 ++    font config mainfont -size $s
 ++    font config mainfontbold -size $s
 ++    set mainfont [fontname mainfont]
 ++    set s $fontattr(textfont,size)
 ++    incr s $inc
 ++    if {$s < 1} {
 ++     set s 1
 ++    }
 ++    set fontattr(textfont,size) $s
 ++    font config textfont -size $s
 ++    font config textfontbold -size $s
 ++    set textfont [fontname textfont]
 ++    setcoords
 ++    settabs
 ++    redisplay
 ++}
 ++
 ++proc clearsha1 {} {
 ++    global sha1entry sha1string
 ++    if {[string length $sha1string] == 40} {
 ++     $sha1entry delete 0 end
 ++    }
 ++}
 ++
 ++proc sha1change {n1 n2 op} {
 ++    global sha1string currentid sha1but
 ++    if {$sha1string == {}
 ++     || ([info exists currentid] && $sha1string == $currentid)} {
 ++     set state disabled
 ++    } else {
 ++     set state normal
 ++    }
 ++    if {[$sha1but cget -state] == $state} return
 ++    if {$state == "normal"} {
 ++     $sha1but conf -state normal -relief raised -text "[mc "Goto:"] "
 ++    } else {
 ++     $sha1but conf -state disabled -relief flat -text "[mc "SHA1 ID:"] "
 ++    }
 ++}
 ++
 ++proc gotocommit {} {
 ++    global sha1string tagids headids curview varcid
 ++
 ++    if {$sha1string == {}
 ++     || ([info exists currentid] && $sha1string == $currentid)} return
 ++    if {[info exists tagids($sha1string)]} {
 ++     set id $tagids($sha1string)
 ++    } elseif {[info exists headids($sha1string)]} {
 ++     set id $headids($sha1string)
 ++    } else {
 ++     set id [string tolower $sha1string]
 ++     if {[regexp {^[0-9a-f]{4,39}$} $id]} {
 ++         set matches [array names varcid "$curview,$id*"]
 ++         if {$matches ne {}} {
 ++             if {[llength $matches] > 1} {
 ++                 error_popup [mc "Short SHA1 id %s is ambiguous" $id]
 ++                 return
 ++             }
 ++             set id [lindex [split [lindex $matches 0] ","] 1]
 ++         }
 ++     }
 ++    }
 ++    if {[commitinview $id $curview]} {
 ++     selectline [rowofcommit $id] 1
 ++     return
 ++    }
 ++    if {[regexp {^[0-9a-fA-F]{4,}$} $sha1string]} {
 ++     set msg [mc "SHA1 id %s is not known" $sha1string]
 ++    } else {
 ++     set msg [mc "Tag/Head %s is not known" $sha1string]
 ++    }
 ++    error_popup $msg
 ++}
 ++
 ++proc lineenter {x y id} {
 ++    global hoverx hovery hoverid hovertimer
 ++    global commitinfo canv
 ++
 ++    if {![info exists commitinfo($id)] && ![getcommit $id]} return
 ++    set hoverx $x
 ++    set hovery $y
 ++    set hoverid $id
 ++    if {[info exists hovertimer]} {
 ++     after cancel $hovertimer
 ++    }
 ++    set hovertimer [after 500 linehover]
 ++    $canv delete hover
 ++}
 ++
 ++proc linemotion {x y id} {
 ++    global hoverx hovery hoverid hovertimer
 ++
 ++    if {[info exists hoverid] && $id == $hoverid} {
 ++     set hoverx $x
 ++     set hovery $y
 ++     if {[info exists hovertimer]} {
 ++         after cancel $hovertimer
 ++     }
 ++     set hovertimer [after 500 linehover]
 ++    }
 ++}
 ++
 ++proc lineleave {id} {
 ++    global hoverid hovertimer canv
 ++
 ++    if {[info exists hoverid] && $id == $hoverid} {
 ++     $canv delete hover
 ++     if {[info exists hovertimer]} {
 ++         after cancel $hovertimer
 ++         unset hovertimer
 ++     }
 ++     unset hoverid
 ++    }
 ++}
 ++
 ++proc linehover {} {
 ++    global hoverx hovery hoverid hovertimer
 ++    global canv linespc lthickness
 ++    global commitinfo
 ++
 ++    set text [lindex $commitinfo($hoverid) 0]
 ++    set ymax [lindex [$canv cget -scrollregion] 3]
 ++    if {$ymax == {}} return
 ++    set yfrac [lindex [$canv yview] 0]
 ++    set x [expr {$hoverx + 2 * $linespc}]
 ++    set y [expr {$hovery + $yfrac * $ymax - $linespc / 2}]
 ++    set x0 [expr {$x - 2 * $lthickness}]
 ++    set y0 [expr {$y - 2 * $lthickness}]
 ++    set x1 [expr {$x + [font measure mainfont $text] + 2 * $lthickness}]
 ++    set y1 [expr {$y + $linespc + 2 * $lthickness}]
 ++    set t [$canv create rectangle $x0 $y0 $x1 $y1 \
 ++            -fill \#ffff80 -outline black -width 1 -tags hover]
 ++    $canv raise $t
 ++    set t [$canv create text $x $y -anchor nw -text $text -tags hover \
 ++            -font mainfont]
 ++    $canv raise $t
 ++}
 ++
 ++proc clickisonarrow {id y} {
 ++    global lthickness
 ++
 ++    set ranges [rowranges $id]
 ++    set thresh [expr {2 * $lthickness + 6}]
 ++    set n [expr {[llength $ranges] - 1}]
 ++    for {set i 1} {$i < $n} {incr i} {
 ++     set row [lindex $ranges $i]
 ++     if {abs([yc $row] - $y) < $thresh} {
 ++         return $i
 ++     }
 ++    }
 ++    return {}
 ++}
 ++
 ++proc arrowjump {id n y} {
 ++    global canv
 ++
 ++    # 1 <-> 2, 3 <-> 4, etc...
 ++    set n [expr {(($n - 1) ^ 1) + 1}]
 ++    set row [lindex [rowranges $id] $n]
 ++    set yt [yc $row]
 ++    set ymax [lindex [$canv cget -scrollregion] 3]
 ++    if {$ymax eq {} || $ymax <= 0} return
 ++    set view [$canv yview]
 ++    set yspan [expr {[lindex $view 1] - [lindex $view 0]}]
 ++    set yfrac [expr {$yt / $ymax - $yspan / 2}]
 ++    if {$yfrac < 0} {
 ++     set yfrac 0
 ++    }
 ++    allcanvs yview moveto $yfrac
 ++}
 ++
 ++proc lineclick {x y id isnew} {
 ++    global ctext commitinfo children canv thickerline curview
 ++
 ++    if {![info exists commitinfo($id)] && ![getcommit $id]} return
 ++    unmarkmatches
 ++    unselectline
 ++    normalline
 ++    $canv delete hover
 ++    # draw this line thicker than normal
 ++    set thickerline $id
 ++    drawlines $id
 ++    if {$isnew} {
 ++     set ymax [lindex [$canv cget -scrollregion] 3]
 ++     if {$ymax eq {}} return
 ++     set yfrac [lindex [$canv yview] 0]
 ++     set y [expr {$y + $yfrac * $ymax}]
 ++    }
 ++    set dirn [clickisonarrow $id $y]
 ++    if {$dirn ne {}} {
 ++     arrowjump $id $dirn $y
 ++     return
 ++    }
 ++
 ++    if {$isnew} {
 ++     addtohistory [list lineclick $x $y $id 0]
 ++    }
 ++    # fill the details pane with info about this line
 ++    $ctext conf -state normal
 ++    clear_ctext
 ++    settabs 0
 ++    $ctext insert end "[mc "Parent"]:\t"
 ++    $ctext insert end $id link0
 ++    setlink $id link0
 ++    set info $commitinfo($id)
 ++    $ctext insert end "\n\t[lindex $info 0]\n"
 ++    $ctext insert end "\t[mc "Author"]:\t[lindex $info 1]\n"
 ++    set date [formatdate [lindex $info 2]]
 ++    $ctext insert end "\t[mc "Date"]:\t$date\n"
 ++    set kids $children($curview,$id)
 ++    if {$kids ne {}} {
 ++     $ctext insert end "\n[mc "Children"]:"
 ++     set i 0
 ++     foreach child $kids {
 ++         incr i
 ++         if {![info exists commitinfo($child)] && ![getcommit $child]} continue
 ++         set info $commitinfo($child)
 ++         $ctext insert end "\n\t"
 ++         $ctext insert end $child link$i
 ++         setlink $child link$i
 ++         $ctext insert end "\n\t[lindex $info 0]"
 ++         $ctext insert end "\n\t[mc "Author"]:\t[lindex $info 1]"
 ++         set date [formatdate [lindex $info 2]]
 ++         $ctext insert end "\n\t[mc "Date"]:\t$date\n"
 ++     }
 ++    }
 ++    $ctext conf -state disabled
 ++    init_flist {}
 ++}
 ++
 ++proc normalline {} {
 ++    global thickerline
 ++    if {[info exists thickerline]} {
 ++     set id $thickerline
 ++     unset thickerline
 ++     drawlines $id
 ++    }
 ++}
 ++
 ++proc selbyid {id} {
 ++    global curview
 ++    if {[commitinview $id $curview]} {
 ++     selectline [rowofcommit $id] 1
 ++    }
 ++}
 ++
 ++proc mstime {} {
 ++    global startmstime
 ++    if {![info exists startmstime]} {
 ++     set startmstime [clock clicks -milliseconds]
 ++    }
 ++    return [format "%.3f" [expr {([clock click -milliseconds] - $startmstime) / 1000.0}]]
 ++}
 ++
 ++proc rowmenu {x y id} {
 ++    global rowctxmenu selectedline rowmenuid curview
 ++    global nullid nullid2 fakerowmenu mainhead
 ++
 ++    stopfinding
 ++    set rowmenuid $id
 ++    if {$selectedline eq {} || [rowofcommit $id] eq $selectedline} {
 ++     set state disabled
 ++    } else {
 ++     set state normal
 ++    }
 ++    if {$id ne $nullid && $id ne $nullid2} {
 ++     set menu $rowctxmenu
 ++     if {$mainhead ne {}} {
 ++         $menu entryconfigure 7 -label [mc "Reset %s branch to here" $mainhead]
 ++     } else {
 ++         $menu entryconfigure 7 -label [mc "Detached head: can't reset" $mainhead] -state disabled
 ++     }
 ++    } else {
 ++     set menu $fakerowmenu
 ++    }
 ++    $menu entryconfigure [mc "Diff this -> selected"] -state $state
 ++    $menu entryconfigure [mc "Diff selected -> this"] -state $state
 ++    $menu entryconfigure [mc "Make patch"] -state $state
 ++    tk_popup $menu $x $y
 ++}
 ++
 ++proc diffvssel {dirn} {
 ++    global rowmenuid selectedline
 ++
 ++    if {$selectedline eq {}} return
 ++    if {$dirn} {
 ++     set oldid [commitonrow $selectedline]
 ++     set newid $rowmenuid
 ++    } else {
 ++     set oldid $rowmenuid
 ++     set newid [commitonrow $selectedline]
 ++    }
 ++    addtohistory [list doseldiff $oldid $newid]
 ++    doseldiff $oldid $newid
 ++}
 ++
 ++proc doseldiff {oldid newid} {
 ++    global ctext
 ++    global commitinfo
 ++
 ++    $ctext conf -state normal
 ++    clear_ctext
 ++    init_flist [mc "Top"]
 ++    $ctext insert end "[mc "From"] "
 ++    $ctext insert end $oldid link0
 ++    setlink $oldid link0
 ++    $ctext insert end "\n     "
 ++    $ctext insert end [lindex $commitinfo($oldid) 0]
 ++    $ctext insert end "\n\n[mc "To"]   "
 ++    $ctext insert end $newid link1
 ++    setlink $newid link1
 ++    $ctext insert end "\n     "
 ++    $ctext insert end [lindex $commitinfo($newid) 0]
 ++    $ctext insert end "\n"
 ++    $ctext conf -state disabled
 ++    $ctext tag remove found 1.0 end
 ++    startdiff [list $oldid $newid]
 ++}
 ++
 ++proc mkpatch {} {
 ++    global rowmenuid currentid commitinfo patchtop patchnum
 ++
 ++    if {![info exists currentid]} return
 ++    set oldid $currentid
 ++    set oldhead [lindex $commitinfo($oldid) 0]
 ++    set newid $rowmenuid
 ++    set newhead [lindex $commitinfo($newid) 0]
 ++    set top .patch
 ++    set patchtop $top
 ++    catch {destroy $top}
 ++    toplevel $top
 ++    label $top.title -text [mc "Generate patch"]
 ++    grid $top.title - -pady 10
 ++    label $top.from -text [mc "From:"]
 ++    entry $top.fromsha1 -width 40 -relief flat
 ++    $top.fromsha1 insert 0 $oldid
 ++    $top.fromsha1 conf -state readonly
 ++    grid $top.from $top.fromsha1 -sticky w
 ++    entry $top.fromhead -width 60 -relief flat
 ++    $top.fromhead insert 0 $oldhead
 ++    $top.fromhead conf -state readonly
 ++    grid x $top.fromhead -sticky w
 ++    label $top.to -text [mc "To:"]
 ++    entry $top.tosha1 -width 40 -relief flat
 ++    $top.tosha1 insert 0 $newid
 ++    $top.tosha1 conf -state readonly
 ++    grid $top.to $top.tosha1 -sticky w
 ++    entry $top.tohead -width 60 -relief flat
 ++    $top.tohead insert 0 $newhead
 ++    $top.tohead conf -state readonly
 ++    grid x $top.tohead -sticky w
 ++    button $top.rev -text [mc "Reverse"] -command mkpatchrev -padx 5
 ++    grid $top.rev x -pady 10
 ++    label $top.flab -text [mc "Output file:"]
 ++    entry $top.fname -width 60
 ++    $top.fname insert 0 [file normalize "patch$patchnum.patch"]
 ++    incr patchnum
 ++    grid $top.flab $top.fname -sticky w
 ++    frame $top.buts
 ++    button $top.buts.gen -text [mc "Generate"] -command mkpatchgo
 ++    button $top.buts.can -text [mc "Cancel"] -command mkpatchcan
 ++    grid $top.buts.gen $top.buts.can
 ++    grid columnconfigure $top.buts 0 -weight 1 -uniform a
 ++    grid columnconfigure $top.buts 1 -weight 1 -uniform a
 ++    grid $top.buts - -pady 10 -sticky ew
 ++    focus $top.fname
 ++}
 ++
 ++proc mkpatchrev {} {
 ++    global patchtop
 ++
 ++    set oldid [$patchtop.fromsha1 get]
 ++    set oldhead [$patchtop.fromhead get]
 ++    set newid [$patchtop.tosha1 get]
 ++    set newhead [$patchtop.tohead get]
 ++    foreach e [list fromsha1 fromhead tosha1 tohead] \
 ++         v [list $newid $newhead $oldid $oldhead] {
 ++     $patchtop.$e conf -state normal
 ++     $patchtop.$e delete 0 end
 ++     $patchtop.$e insert 0 $v
 ++     $patchtop.$e conf -state readonly
 ++    }
 ++}
 ++
 ++proc mkpatchgo {} {
 ++    global patchtop nullid nullid2
 ++
 ++    set oldid [$patchtop.fromsha1 get]
 ++    set newid [$patchtop.tosha1 get]
 ++    set fname [$patchtop.fname get]
 ++    set cmd [diffcmd [list $oldid $newid] -p]
 ++    # trim off the initial "|"
 ++    set cmd [lrange $cmd 1 end]
 ++    lappend cmd >$fname &
 ++    if {[catch {eval exec $cmd} err]} {
 ++     error_popup "[mc "Error creating patch:"] $err"
 ++    }
 ++    catch {destroy $patchtop}
 ++    unset patchtop
 ++}
 ++
 ++proc mkpatchcan {} {
 ++    global patchtop
 ++
 ++    catch {destroy $patchtop}
 ++    unset patchtop
 ++}
 ++
 ++proc mktag {} {
 ++    global rowmenuid mktagtop commitinfo
 ++
 ++    set top .maketag
 ++    set mktagtop $top
 ++    catch {destroy $top}
 ++    toplevel $top
 ++    label $top.title -text [mc "Create tag"]
 ++    grid $top.title - -pady 10
 ++    label $top.id -text [mc "ID:"]
 ++    entry $top.sha1 -width 40 -relief flat
 ++    $top.sha1 insert 0 $rowmenuid
 ++    $top.sha1 conf -state readonly
 ++    grid $top.id $top.sha1 -sticky w
 ++    entry $top.head -width 60 -relief flat
 ++    $top.head insert 0 [lindex $commitinfo($rowmenuid) 0]
 ++    $top.head conf -state readonly
 ++    grid x $top.head -sticky w
 ++    label $top.tlab -text [mc "Tag name:"]
 ++    entry $top.tag -width 60
 ++    grid $top.tlab $top.tag -sticky w
 ++    frame $top.buts
 ++    button $top.buts.gen -text [mc "Create"] -command mktaggo
 ++    button $top.buts.can -text [mc "Cancel"] -command mktagcan
 ++    grid $top.buts.gen $top.buts.can
 ++    grid columnconfigure $top.buts 0 -weight 1 -uniform a
 ++    grid columnconfigure $top.buts 1 -weight 1 -uniform a
 ++    grid $top.buts - -pady 10 -sticky ew
 ++    focus $top.tag
 ++}
 ++
 ++proc domktag {} {
 ++    global mktagtop env tagids idtags
 ++
 ++    set id [$mktagtop.sha1 get]
 ++    set tag [$mktagtop.tag get]
 ++    if {$tag == {}} {
 ++     error_popup [mc "No tag name specified"]
 ++     return
 ++    }
 ++    if {[info exists tagids($tag)]} {
 ++     error_popup [mc "Tag \"%s\" already exists" $tag]
 ++     return
 ++    }
 ++    if {[catch {
 ++     exec git tag $tag $id
 ++    } err]} {
 ++     error_popup "[mc "Error creating tag:"] $err"
 ++     return
 ++    }
 ++
 ++    set tagids($tag) $id
 ++    lappend idtags($id) $tag
 ++    redrawtags $id
 ++    addedtag $id
 ++    dispneartags 0
 ++    run refill_reflist
 ++}
 ++
 ++proc redrawtags {id} {
 ++    global canv linehtag idpos currentid curview cmitlisted
 ++    global canvxmax iddrawn circleitem mainheadid circlecolors
 ++
 ++    if {![commitinview $id $curview]} return
 ++    if {![info exists iddrawn($id)]} return
 ++    set row [rowofcommit $id]
 ++    if {$id eq $mainheadid} {
 ++     set ofill yellow
 ++    } else {
 ++     set ofill [lindex $circlecolors $cmitlisted($curview,$id)]
 ++    }
 ++    $canv itemconf $circleitem($row) -fill $ofill
 ++    $canv delete tag.$id
 ++    set xt [eval drawtags $id $idpos($id)]
 ++    $canv coords $linehtag($row) $xt [lindex $idpos($id) 2]
 ++    set text [$canv itemcget $linehtag($row) -text]
 ++    set font [$canv itemcget $linehtag($row) -font]
 ++    set xr [expr {$xt + [font measure $font $text]}]
 ++    if {$xr > $canvxmax} {
 ++     set canvxmax $xr
 ++     setcanvscroll
 ++    }
 ++    if {[info exists currentid] && $currentid == $id} {
 ++     make_secsel $row
 ++    }
 ++}
 ++
 ++proc mktagcan {} {
 ++    global mktagtop
 ++
 ++    catch {destroy $mktagtop}
 ++    unset mktagtop
 ++}
 ++
 ++proc mktaggo {} {
 ++    domktag
 ++    mktagcan
 ++}
 ++
 ++proc writecommit {} {
 ++    global rowmenuid wrcomtop commitinfo wrcomcmd
 ++
 ++    set top .writecommit
 ++    set wrcomtop $top
 ++    catch {destroy $top}
 ++    toplevel $top
 ++    label $top.title -text [mc "Write commit to file"]
 ++    grid $top.title - -pady 10
 ++    label $top.id -text [mc "ID:"]
 ++    entry $top.sha1 -width 40 -relief flat
 ++    $top.sha1 insert 0 $rowmenuid
 ++    $top.sha1 conf -state readonly
 ++    grid $top.id $top.sha1 -sticky w
 ++    entry $top.head -width 60 -relief flat
 ++    $top.head insert 0 [lindex $commitinfo($rowmenuid) 0]
 ++    $top.head conf -state readonly
 ++    grid x $top.head -sticky w
 ++    label $top.clab -text [mc "Command:"]
 ++    entry $top.cmd -width 60 -textvariable wrcomcmd
 ++    grid $top.clab $top.cmd -sticky w -pady 10
 ++    label $top.flab -text [mc "Output file:"]
 ++    entry $top.fname -width 60
 ++    $top.fname insert 0 [file normalize "commit-[string range $rowmenuid 0 6]"]
 ++    grid $top.flab $top.fname -sticky w
 ++    frame $top.buts
 ++    button $top.buts.gen -text [mc "Write"] -command wrcomgo
 ++    button $top.buts.can -text [mc "Cancel"] -command wrcomcan
 ++    grid $top.buts.gen $top.buts.can
 ++    grid columnconfigure $top.buts 0 -weight 1 -uniform a
 ++    grid columnconfigure $top.buts 1 -weight 1 -uniform a
 ++    grid $top.buts - -pady 10 -sticky ew
 ++    focus $top.fname
 ++}
 ++
 ++proc wrcomgo {} {
 ++    global wrcomtop
 ++
 ++    set id [$wrcomtop.sha1 get]
 ++    set cmd "echo $id | [$wrcomtop.cmd get]"
 ++    set fname [$wrcomtop.fname get]
 ++    if {[catch {exec sh -c $cmd >$fname &} err]} {
 ++     error_popup "[mc "Error writing commit:"] $err"
 ++    }
 ++    catch {destroy $wrcomtop}
 ++    unset wrcomtop
 ++}
 ++
 ++proc wrcomcan {} {
 ++    global wrcomtop
 ++
 ++    catch {destroy $wrcomtop}
 ++    unset wrcomtop
 ++}
 ++
 ++proc mkbranch {} {
 ++    global rowmenuid mkbrtop
 ++
 ++    set top .makebranch
 ++    catch {destroy $top}
 ++    toplevel $top
 ++    label $top.title -text [mc "Create new branch"]
 ++    grid $top.title - -pady 10
 ++    label $top.id -text [mc "ID:"]
 ++    entry $top.sha1 -width 40 -relief flat
 ++    $top.sha1 insert 0 $rowmenuid
 ++    $top.sha1 conf -state readonly
 ++    grid $top.id $top.sha1 -sticky w
 ++    label $top.nlab -text [mc "Name:"]
 ++    entry $top.name -width 40
 ++    grid $top.nlab $top.name -sticky w
 ++    frame $top.buts
 ++    button $top.buts.go -text [mc "Create"] -command [list mkbrgo $top]
 ++    button $top.buts.can -text [mc "Cancel"] -command "catch {destroy $top}"
 ++    grid $top.buts.go $top.buts.can
 ++    grid columnconfigure $top.buts 0 -weight 1 -uniform a
 ++    grid columnconfigure $top.buts 1 -weight 1 -uniform a
 ++    grid $top.buts - -pady 10 -sticky ew
 ++    focus $top.name
 ++}
 ++
 ++proc mkbrgo {top} {
 ++    global headids idheads
 ++
 ++    set name [$top.name get]
 ++    set id [$top.sha1 get]
 ++    if {$name eq {}} {
 ++     error_popup [mc "Please specify a name for the new branch"]
 ++     return
 ++    }
 ++    catch {destroy $top}
 ++    nowbusy newbranch
 ++    update
 ++    if {[catch {
 ++     exec git branch $name $id
 ++    } err]} {
 ++     notbusy newbranch
 ++     error_popup $err
 ++    } else {
 ++     set headids($name) $id
 ++     lappend idheads($id) $name
 ++     addedhead $id $name
 ++     notbusy newbranch
 ++     redrawtags $id
 ++     dispneartags 0
 ++     run refill_reflist
 ++    }
 ++}
 ++
 ++proc cherrypick {} {
 ++    global rowmenuid curview
 ++    global mainhead mainheadid
 ++
 ++    set oldhead [exec git rev-parse HEAD]
 ++    set dheads [descheads $rowmenuid]
 ++    if {$dheads ne {} && [lsearch -exact $dheads $oldhead] >= 0} {
 ++     set ok [confirm_popup [mc "Commit %s is already\
 ++             included in branch %s -- really re-apply it?" \
 ++                                [string range $rowmenuid 0 7] $mainhead]]
 ++     if {!$ok} return
 ++    }
 ++    nowbusy cherrypick [mc "Cherry-picking"]
 ++    update
 ++    # Unfortunately git-cherry-pick writes stuff to stderr even when
 ++    # no error occurs, and exec takes that as an indication of error...
 ++    if {[catch {exec sh -c "git cherry-pick -r $rowmenuid 2>&1"} err]} {
 ++     notbusy cherrypick
 ++     error_popup $err
 ++     return
 ++    }
 ++    set newhead [exec git rev-parse HEAD]
 ++    if {$newhead eq $oldhead} {
 ++     notbusy cherrypick
 ++     error_popup [mc "No changes committed"]
 ++     return
 ++    }
 ++    addnewchild $newhead $oldhead
 ++    if {[commitinview $oldhead $curview]} {
 ++     insertrow $newhead $oldhead $curview
 ++     if {$mainhead ne {}} {
 ++         movehead $newhead $mainhead
 ++         movedhead $newhead $mainhead
 ++     }
 ++     set mainheadid $newhead
 ++     redrawtags $oldhead
 ++     redrawtags $newhead
 ++     selbyid $newhead
 ++    }
 ++    notbusy cherrypick
 ++}
 ++
 ++proc resethead {} {
 ++    global mainhead rowmenuid confirm_ok resettype
 ++
 ++    set confirm_ok 0
 ++    set w ".confirmreset"
 ++    toplevel $w
 ++    wm transient $w .
 ++    wm title $w [mc "Confirm reset"]
 ++    message $w.m -text \
 ++     [mc "Reset branch %s to %s?" $mainhead [string range $rowmenuid 0 7]] \
 ++     -justify center -aspect 1000
 ++    pack $w.m -side top -fill x -padx 20 -pady 20
 ++    frame $w.f -relief sunken -border 2
 ++    message $w.f.rt -text [mc "Reset type:"] -aspect 1000
 ++    grid $w.f.rt -sticky w
 ++    set resettype mixed
 ++    radiobutton $w.f.soft -value soft -variable resettype -justify left \
 ++     -text [mc "Soft: Leave working tree and index untouched"]
 ++    grid $w.f.soft -sticky w
 ++    radiobutton $w.f.mixed -value mixed -variable resettype -justify left \
 ++     -text [mc "Mixed: Leave working tree untouched, reset index"]
 ++    grid $w.f.mixed -sticky w
 ++    radiobutton $w.f.hard -value hard -variable resettype -justify left \
 ++     -text [mc "Hard: Reset working tree and index\n(discard ALL local changes)"]
 ++    grid $w.f.hard -sticky w
 ++    pack $w.f -side top -fill x
 ++    button $w.ok -text [mc OK] -command "set confirm_ok 1; destroy $w"
 ++    pack $w.ok -side left -fill x -padx 20 -pady 20
 ++    button $w.cancel -text [mc Cancel] -command "destroy $w"
 ++    pack $w.cancel -side right -fill x -padx 20 -pady 20
 ++    bind $w <Visibility> "grab $w; focus $w"
 ++    tkwait window $w
 ++    if {!$confirm_ok} return
 ++    if {[catch {set fd [open \
 ++         [list | git reset --$resettype $rowmenuid 2>@1] r]} err]} {
 ++     error_popup $err
 ++    } else {
 ++     dohidelocalchanges
 ++     filerun $fd [list readresetstat $fd]
 ++     nowbusy reset [mc "Resetting"]
 ++     selbyid $rowmenuid
 ++    }
 ++}
 ++
 ++proc readresetstat {fd} {
 ++    global mainhead mainheadid showlocalchanges rprogcoord
 ++
 ++    if {[gets $fd line] >= 0} {
 ++     if {[regexp {([0-9]+)% \(([0-9]+)/([0-9]+)\)} $line match p m n]} {
 ++         set rprogcoord [expr {1.0 * $m / $n}]
 ++         adjustprogress
 ++     }
 ++     return 1
 ++    }
 ++    set rprogcoord 0
 ++    adjustprogress
 ++    notbusy reset
 ++    if {[catch {close $fd} err]} {
 ++     error_popup $err
 ++    }
 ++    set oldhead $mainheadid
 ++    set newhead [exec git rev-parse HEAD]
 ++    if {$newhead ne $oldhead} {
 ++     movehead $newhead $mainhead
 ++     movedhead $newhead $mainhead
 ++     set mainheadid $newhead
 ++     redrawtags $oldhead
 ++     redrawtags $newhead
 ++    }
 ++    if {$showlocalchanges} {
 ++     doshowlocalchanges
 ++    }
 ++    return 0
 ++}
 ++
 ++# context menu for a head
 ++proc headmenu {x y id head} {
 ++    global headmenuid headmenuhead headctxmenu mainhead
 ++
 ++    stopfinding
 ++    set headmenuid $id
 ++    set headmenuhead $head
 ++    set state normal
 ++    if {$head eq $mainhead} {
 ++     set state disabled
 ++    }
 ++    $headctxmenu entryconfigure 0 -state $state
 ++    $headctxmenu entryconfigure 1 -state $state
 ++    tk_popup $headctxmenu $x $y
 ++}
 ++
 ++proc cobranch {} {
 ++    global headmenuid headmenuhead headids
 ++    global showlocalchanges mainheadid
 ++
 ++    # check the tree is clean first??
 ++    nowbusy checkout [mc "Checking out"]
 ++    update
 ++    dohidelocalchanges
 ++    if {[catch {
 ++     set fd [open [list | git checkout $headmenuhead 2>@1] r]
 ++    } err]} {
 ++     notbusy checkout
 ++     error_popup $err
 ++     if {$showlocalchanges} {
 ++         dodiffindex
 ++     }
 ++    } else {
 ++     filerun $fd [list readcheckoutstat $fd $headmenuhead $headmenuid]
 ++    }
 ++}
 ++
 ++proc readcheckoutstat {fd newhead newheadid} {
 ++    global mainhead mainheadid headids showlocalchanges progresscoords
 ++
 ++    if {[gets $fd line] >= 0} {
 ++     if {[regexp {([0-9]+)% \(([0-9]+)/([0-9]+)\)} $line match p m n]} {
 ++         set progresscoords [list 0 [expr {1.0 * $m / $n}]]
 ++         adjustprogress
 ++     }
 ++     return 1
 ++    }
 ++    set progresscoords {0 0}
 ++    adjustprogress
 ++    notbusy checkout
 ++    if {[catch {close $fd} err]} {
 ++     error_popup $err
 ++    }
 ++    set oldmainid $mainheadid
 ++    set mainhead $newhead
 ++    set mainheadid $newheadid
 ++    redrawtags $oldmainid
 ++    redrawtags $newheadid
 ++    selbyid $newheadid
 ++    if {$showlocalchanges} {
 ++     dodiffindex
 ++    }
 ++}
 ++
 ++proc rmbranch {} {
 ++    global headmenuid headmenuhead mainhead
 ++    global idheads
 ++
 ++    set head $headmenuhead
 ++    set id $headmenuid
 ++    # this check shouldn't be needed any more...
 ++    if {$head eq $mainhead} {
 ++     error_popup [mc "Cannot delete the currently checked-out branch"]
 ++     return
 ++    }
 ++    set dheads [descheads $id]
 ++    if {[llength $dheads] == 1 && $idheads($dheads) eq $head} {
 ++     # the stuff on this branch isn't on any other branch
 ++     if {![confirm_popup [mc "The commits on branch %s aren't on any other\
 ++                     branch.\nReally delete branch %s?" $head $head]]} return
 ++    }
 ++    nowbusy rmbranch
 ++    update
 ++    if {[catch {exec git branch -D $head} err]} {
 ++     notbusy rmbranch
 ++     error_popup $err
 ++     return
 ++    }
 ++    removehead $id $head
 ++    removedhead $id $head
 ++    redrawtags $id
 ++    notbusy rmbranch
 ++    dispneartags 0
 ++    run refill_reflist
 ++}
 ++
 ++# Display a list of tags and heads
 ++proc showrefs {} {
 ++    global showrefstop bgcolor fgcolor selectbgcolor
 ++    global bglist fglist reflistfilter reflist maincursor
 ++
 ++    set top .showrefs
 ++    set showrefstop $top
 ++    if {[winfo exists $top]} {
 ++     raise $top
 ++     refill_reflist
 ++     return
 ++    }
 ++    toplevel $top
 ++    wm title $top [mc "Tags and heads: %s" [file tail [pwd]]]
 ++    text $top.list -background $bgcolor -foreground $fgcolor \
 ++     -selectbackground $selectbgcolor -font mainfont \
 ++     -xscrollcommand "$top.xsb set" -yscrollcommand "$top.ysb set" \
 ++     -width 30 -height 20 -cursor $maincursor \
 ++     -spacing1 1 -spacing3 1 -state disabled
 ++    $top.list tag configure highlight -background $selectbgcolor
 ++    lappend bglist $top.list
 ++    lappend fglist $top.list
 ++    scrollbar $top.ysb -command "$top.list yview" -orient vertical
 ++    scrollbar $top.xsb -command "$top.list xview" -orient horizontal
 ++    grid $top.list $top.ysb -sticky nsew
 ++    grid $top.xsb x -sticky ew
 ++    frame $top.f
 ++    label $top.f.l -text "[mc "Filter"]: "
 ++    entry $top.f.e -width 20 -textvariable reflistfilter
 ++    set reflistfilter "*"
 ++    trace add variable reflistfilter write reflistfilter_change
 ++    pack $top.f.e -side right -fill x -expand 1
 ++    pack $top.f.l -side left
 ++    grid $top.f - -sticky ew -pady 2
 ++    button $top.close -command [list destroy $top] -text [mc "Close"]
 ++    grid $top.close -
 ++    grid columnconfigure $top 0 -weight 1
 ++    grid rowconfigure $top 0 -weight 1
 ++    bind $top.list <1> {break}
 ++    bind $top.list <B1-Motion> {break}
 ++    bind $top.list <ButtonRelease-1> {sel_reflist %W %x %y; break}
 ++    set reflist {}
 ++    refill_reflist
 ++}
 ++
 ++proc sel_reflist {w x y} {
 ++    global showrefstop reflist headids tagids otherrefids
 ++
 ++    if {![winfo exists $showrefstop]} return
 ++    set l [lindex [split [$w index "@$x,$y"] "."] 0]
 ++    set ref [lindex $reflist [expr {$l-1}]]
 ++    set n [lindex $ref 0]
 ++    switch -- [lindex $ref 1] {
 ++     "H" {selbyid $headids($n)}
 ++     "T" {selbyid $tagids($n)}
 ++     "o" {selbyid $otherrefids($n)}
 ++    }
 ++    $showrefstop.list tag add highlight $l.0 "$l.0 lineend"
 ++}
 ++
 ++proc unsel_reflist {} {
 ++    global showrefstop
 ++
 ++    if {![info exists showrefstop] || ![winfo exists $showrefstop]} return
 ++    $showrefstop.list tag remove highlight 0.0 end
 ++}
 ++
 ++proc reflistfilter_change {n1 n2 op} {
 ++    global reflistfilter
 ++
 ++    after cancel refill_reflist
 ++    after 200 refill_reflist
 ++}
 ++
 ++proc refill_reflist {} {
 ++    global reflist reflistfilter showrefstop headids tagids otherrefids
 ++    global curview commitinterest
 ++
 ++    if {![info exists showrefstop] || ![winfo exists $showrefstop]} return
 ++    set refs {}
 ++    foreach n [array names headids] {
 ++     if {[string match $reflistfilter $n]} {
 ++         if {[commitinview $headids($n) $curview]} {
 ++             lappend refs [list $n H]
 ++         } else {
 ++             set commitinterest($headids($n)) {run refill_reflist}
 ++         }
 ++     }
 ++    }
 ++    foreach n [array names tagids] {
 ++     if {[string match $reflistfilter $n]} {
 ++         if {[commitinview $tagids($n) $curview]} {
 ++             lappend refs [list $n T]
 ++         } else {
 ++             set commitinterest($tagids($n)) {run refill_reflist}
 ++         }
 ++     }
 ++    }
 ++    foreach n [array names otherrefids] {
 ++     if {[string match $reflistfilter $n]} {
 ++         if {[commitinview $otherrefids($n) $curview]} {
 ++             lappend refs [list $n o]
 ++         } else {
 ++             set commitinterest($otherrefids($n)) {run refill_reflist}
 ++         }
 ++     }
 ++    }
 ++    set refs [lsort -index 0 $refs]
 ++    if {$refs eq $reflist} return
 ++
 ++    # Update the contents of $showrefstop.list according to the
 ++    # differences between $reflist (old) and $refs (new)
 ++    $showrefstop.list conf -state normal
 ++    $showrefstop.list insert end "\n"
 ++    set i 0
 ++    set j 0
 ++    while {$i < [llength $reflist] || $j < [llength $refs]} {
 ++     if {$i < [llength $reflist]} {
 ++         if {$j < [llength $refs]} {
 ++             set cmp [string compare [lindex $reflist $i 0] \
 ++                          [lindex $refs $j 0]]
 ++             if {$cmp == 0} {
 ++                 set cmp [string compare [lindex $reflist $i 1] \
 ++                              [lindex $refs $j 1]]
 ++             }
 ++         } else {
 ++             set cmp -1
 ++         }
 ++     } else {
 ++         set cmp 1
 ++     }
 ++     switch -- $cmp {
 ++         -1 {
 ++             $showrefstop.list delete "[expr {$j+1}].0" "[expr {$j+2}].0"
 ++             incr i
 ++         }
 ++         0 {
 ++             incr i
 ++             incr j
 ++         }
 ++         1 {
 ++             set l [expr {$j + 1}]
 ++             $showrefstop.list image create $l.0 -align baseline \
 ++                 -image reficon-[lindex $refs $j 1] -padx 2
 ++             $showrefstop.list insert $l.1 "[lindex $refs $j 0]\n"
 ++             incr j
 ++         }
 ++     }
 ++    }
 ++    set reflist $refs
 ++    # delete last newline
 ++    $showrefstop.list delete end-2c end-1c
 ++    $showrefstop.list conf -state disabled
 ++}
 ++
 ++# Stuff for finding nearby tags
 ++proc getallcommits {} {
 ++    global allcommits nextarc seeds allccache allcwait cachedarcs allcupdate
 ++    global idheads idtags idotherrefs allparents tagobjid
 ++
 ++    if {![info exists allcommits]} {
 ++     set nextarc 0
 ++     set allcommits 0
 ++     set seeds {}
 ++     set allcwait 0
 ++     set cachedarcs 0
 ++     set allccache [file join [gitdir] "gitk.cache"]
 ++     if {![catch {
 ++         set f [open $allccache r]
 ++         set allcwait 1
 ++         getcache $f
 ++     }]} return
 ++    }
 ++
 ++    if {$allcwait} {
 ++     return
 ++    }
 ++    set cmd [list | git rev-list --parents]
 ++    set allcupdate [expr {$seeds ne {}}]
 ++    if {!$allcupdate} {
 ++     set ids "--all"
 ++    } else {
 ++     set refs [concat [array names idheads] [array names idtags] \
 ++                   [array names idotherrefs]]
 ++     set ids {}
 ++     set tagobjs {}
 ++     foreach name [array names tagobjid] {
 ++         lappend tagobjs $tagobjid($name)
 ++     }
 ++     foreach id [lsort -unique $refs] {
 ++         if {![info exists allparents($id)] &&
 ++             [lsearch -exact $tagobjs $id] < 0} {
 ++             lappend ids $id
 ++         }
 ++     }
 ++     if {$ids ne {}} {
 ++         foreach id $seeds {
 ++             lappend ids "^$id"
 ++         }
 ++     }
 ++    }
 ++    if {$ids ne {}} {
 ++     set fd [open [concat $cmd $ids] r]
 ++     fconfigure $fd -blocking 0
 ++     incr allcommits
 ++     nowbusy allcommits
 ++     filerun $fd [list getallclines $fd]
 ++    } else {
 ++     dispneartags 0
 ++    }
 ++}
 ++
 ++# Since most commits have 1 parent and 1 child, we group strings of
 ++# such commits into "arcs" joining branch/merge points (BMPs), which
 ++# are commits that either don't have 1 parent or don't have 1 child.
 ++#
 ++# arcnos(id) - incoming arcs for BMP, arc we're on for other nodes
 ++# arcout(id) - outgoing arcs for BMP
 ++# arcids(a) - list of IDs on arc including end but not start
 ++# arcstart(a) - BMP ID at start of arc
 ++# arcend(a) - BMP ID at end of arc
 ++# growing(a) - arc a is still growing
 ++# arctags(a) - IDs out of arcids (excluding end) that have tags
 ++# archeads(a) - IDs out of arcids (excluding end) that have heads
 ++# The start of an arc is at the descendent end, so "incoming" means
 ++# coming from descendents, and "outgoing" means going towards ancestors.
 ++
 ++proc getallclines {fd} {
 ++    global allparents allchildren idtags idheads nextarc
 ++    global arcnos arcids arctags arcout arcend arcstart archeads growing
 ++    global seeds allcommits cachedarcs allcupdate
 ++    
 ++    set nid 0
 ++    while {[incr nid] <= 1000 && [gets $fd line] >= 0} {
 ++     set id [lindex $line 0]
 ++     if {[info exists allparents($id)]} {
 ++         # seen it already
 ++         continue
 ++     }
 ++     set cachedarcs 0
 ++     set olds [lrange $line 1 end]
 ++     set allparents($id) $olds
 ++     if {![info exists allchildren($id)]} {
 ++         set allchildren($id) {}
 ++         set arcnos($id) {}
 ++         lappend seeds $id
 ++     } else {
 ++         set a $arcnos($id)
 ++         if {[llength $olds] == 1 && [llength $a] == 1} {
 ++             lappend arcids($a) $id
 ++             if {[info exists idtags($id)]} {
 ++                 lappend arctags($a) $id
 ++             }
 ++             if {[info exists idheads($id)]} {
 ++                 lappend archeads($a) $id
 ++             }
 ++             if {[info exists allparents($olds)]} {
 ++                 # seen parent already
 ++                 if {![info exists arcout($olds)]} {
 ++                     splitarc $olds
 ++                 }
 ++                 lappend arcids($a) $olds
 ++                 set arcend($a) $olds
 ++                 unset growing($a)
 ++             }
 ++             lappend allchildren($olds) $id
 ++             lappend arcnos($olds) $a
 ++             continue
 ++         }
 ++     }
 ++     foreach a $arcnos($id) {
 ++         lappend arcids($a) $id
 ++         set arcend($a) $id
 ++         unset growing($a)
 ++     }
 ++
 ++     set ao {}
 ++     foreach p $olds {
 ++         lappend allchildren($p) $id
 ++         set a [incr nextarc]
 ++         set arcstart($a) $id
 ++         set archeads($a) {}
 ++         set arctags($a) {}
 ++         set archeads($a) {}
 ++         set arcids($a) {}
 ++         lappend ao $a
 ++         set growing($a) 1
 ++         if {[info exists allparents($p)]} {
 ++             # seen it already, may need to make a new branch
 ++             if {![info exists arcout($p)]} {
 ++                 splitarc $p
 ++             }
 ++             lappend arcids($a) $p
 ++             set arcend($a) $p
 ++             unset growing($a)
 ++         }
 ++         lappend arcnos($p) $a
 ++     }
 ++     set arcout($id) $ao
 ++    }
 ++    if {$nid > 0} {
 ++     global cached_dheads cached_dtags cached_atags
 ++     catch {unset cached_dheads}
 ++     catch {unset cached_dtags}
 ++     catch {unset cached_atags}
 ++    }
 ++    if {![eof $fd]} {
 ++     return [expr {$nid >= 1000? 2: 1}]
 ++    }
 ++    set cacheok 1
 ++    if {[catch {
 ++     fconfigure $fd -blocking 1
 ++     close $fd
 ++    } err]} {
 ++     # got an error reading the list of commits
 ++     # if we were updating, try rereading the whole thing again
 ++     if {$allcupdate} {
 ++         incr allcommits -1
 ++         dropcache $err
 ++         return
 ++     }
 ++     error_popup "[mc "Error reading commit topology information;\
 ++             branch and preceding/following tag information\
 ++             will be incomplete."]\n($err)"
 ++     set cacheok 0
 ++    }
 ++    if {[incr allcommits -1] == 0} {
 ++     notbusy allcommits
 ++     if {$cacheok} {
 ++         run savecache
 ++     }
 ++    }
 ++    dispneartags 0
 ++    return 0
 ++}
 ++
 ++proc recalcarc {a} {
 ++    global arctags archeads arcids idtags idheads
 ++
 ++    set at {}
 ++    set ah {}
 ++    foreach id [lrange $arcids($a) 0 end-1] {
 ++     if {[info exists idtags($id)]} {
 ++         lappend at $id
 ++     }
 ++     if {[info exists idheads($id)]} {
 ++         lappend ah $id
 ++     }
 ++    }
 ++    set arctags($a) $at
 ++    set archeads($a) $ah
 ++}
 ++
 ++proc splitarc {p} {
 ++    global arcnos arcids nextarc arctags archeads idtags idheads
 ++    global arcstart arcend arcout allparents growing
 ++
 ++    set a $arcnos($p)
 ++    if {[llength $a] != 1} {
 ++     puts "oops splitarc called but [llength $a] arcs already"
 ++     return
 ++    }
 ++    set a [lindex $a 0]
 ++    set i [lsearch -exact $arcids($a) $p]
 ++    if {$i < 0} {
 ++     puts "oops splitarc $p not in arc $a"
 ++     return
 ++    }
 ++    set na [incr nextarc]
 ++    if {[info exists arcend($a)]} {
 ++     set arcend($na) $arcend($a)
 ++    } else {
 ++     set l [lindex $allparents([lindex $arcids($a) end]) 0]
 ++     set j [lsearch -exact $arcnos($l) $a]
 ++     set arcnos($l) [lreplace $arcnos($l) $j $j $na]
 ++    }
 ++    set tail [lrange $arcids($a) [expr {$i+1}] end]
 ++    set arcids($a) [lrange $arcids($a) 0 $i]
 ++    set arcend($a) $p
 ++    set arcstart($na) $p
 ++    set arcout($p) $na
 ++    set arcids($na) $tail
 ++    if {[info exists growing($a)]} {
 ++     set growing($na) 1
 ++     unset growing($a)
 ++    }
 ++
 ++    foreach id $tail {
 ++     if {[llength $arcnos($id)] == 1} {
 ++         set arcnos($id) $na
 ++     } else {
 ++         set j [lsearch -exact $arcnos($id) $a]
 ++         set arcnos($id) [lreplace $arcnos($id) $j $j $na]
 ++     }
 ++    }
 ++
 ++    # reconstruct tags and heads lists
 ++    if {$arctags($a) ne {} || $archeads($a) ne {}} {
 ++     recalcarc $a
 ++     recalcarc $na
 ++    } else {
 ++     set arctags($na) {}
 ++     set archeads($na) {}
 ++    }
 ++}
 ++
 ++# Update things for a new commit added that is a child of one
 ++# existing commit.  Used when cherry-picking.
 ++proc addnewchild {id p} {
 ++    global allparents allchildren idtags nextarc
 ++    global arcnos arcids arctags arcout arcend arcstart archeads growing
 ++    global seeds allcommits
 ++
 ++    if {![info exists allcommits] || ![info exists arcnos($p)]} return
 ++    set allparents($id) [list $p]
 ++    set allchildren($id) {}
 ++    set arcnos($id) {}
 ++    lappend seeds $id
 ++    lappend allchildren($p) $id
 ++    set a [incr nextarc]
 ++    set arcstart($a) $id
 ++    set archeads($a) {}
 ++    set arctags($a) {}
 ++    set arcids($a) [list $p]
 ++    set arcend($a) $p
 ++    if {![info exists arcout($p)]} {
 ++     splitarc $p
 ++    }
 ++    lappend arcnos($p) $a
 ++    set arcout($id) [list $a]
 ++}
 ++
 ++# This implements a cache for the topology information.
 ++# The cache saves, for each arc, the start and end of the arc,
 ++# the ids on the arc, and the outgoing arcs from the end.
 ++proc readcache {f} {
 ++    global arcnos arcids arcout arcstart arcend arctags archeads nextarc
 ++    global idtags idheads allparents cachedarcs possible_seeds seeds growing
 ++    global allcwait
 ++
 ++    set a $nextarc
 ++    set lim $cachedarcs
 ++    if {$lim - $a > 500} {
 ++     set lim [expr {$a + 500}]
 ++    }
 ++    if {[catch {
 ++     if {$a == $lim} {
 ++         # finish reading the cache and setting up arctags, etc.
 ++         set line [gets $f]
 ++         if {$line ne "1"} {error "bad final version"}
 ++         close $f
 ++         foreach id [array names idtags] {
 ++             if {[info exists arcnos($id)] && [llength $arcnos($id)] == 1 &&
 ++                 [llength $allparents($id)] == 1} {
 ++                 set a [lindex $arcnos($id) 0]
 ++                 if {$arctags($a) eq {}} {
 ++                     recalcarc $a
 ++                 }
 ++             }
 ++         }
 ++         foreach id [array names idheads] {
 ++             if {[info exists arcnos($id)] && [llength $arcnos($id)] == 1 &&
 ++                 [llength $allparents($id)] == 1} {
 ++                 set a [lindex $arcnos($id) 0]
 ++                 if {$archeads($a) eq {}} {
 ++                     recalcarc $a
 ++                 }
 ++             }
 ++         }
 ++         foreach id [lsort -unique $possible_seeds] {
 ++             if {$arcnos($id) eq {}} {
 ++                 lappend seeds $id
 ++             }
 ++         }
 ++         set allcwait 0
 ++     } else {
 ++         while {[incr a] <= $lim} {
 ++             set line [gets $f]
 ++             if {[llength $line] != 3} {error "bad line"}
 ++             set s [lindex $line 0]
 ++             set arcstart($a) $s
 ++             lappend arcout($s) $a
 ++             if {![info exists arcnos($s)]} {
 ++                 lappend possible_seeds $s
 ++                 set arcnos($s) {}
 ++             }
 ++             set e [lindex $line 1]
 ++             if {$e eq {}} {
 ++                 set growing($a) 1
 ++             } else {
 ++                 set arcend($a) $e
 ++                 if {![info exists arcout($e)]} {
 ++                     set arcout($e) {}
 ++                 }
 ++             }
 ++             set arcids($a) [lindex $line 2]
 ++             foreach id $arcids($a) {
 ++                 lappend allparents($s) $id
 ++                 set s $id
 ++                 lappend arcnos($id) $a
 ++             }
 ++             if {![info exists allparents($s)]} {
 ++                 set allparents($s) {}
 ++             }
 ++             set arctags($a) {}
 ++             set archeads($a) {}
 ++         }
 ++         set nextarc [expr {$a - 1}]
 ++     }
 ++    } err]} {
 ++     dropcache $err
 ++     return 0
 ++    }
 ++    if {!$allcwait} {
 ++     getallcommits
 ++    }
 ++    return $allcwait
 ++}
 ++
 ++proc getcache {f} {
 ++    global nextarc cachedarcs possible_seeds
 ++
 ++    if {[catch {
 ++     set line [gets $f]
 ++     if {[llength $line] != 2 || [lindex $line 0] ne "1"} {error "bad version"}
 ++     # make sure it's an integer
 ++     set cachedarcs [expr {int([lindex $line 1])}]
 ++     if {$cachedarcs < 0} {error "bad number of arcs"}
 ++     set nextarc 0
 ++     set possible_seeds {}
 ++     run readcache $f
 ++    } err]} {
 ++     dropcache $err
 ++    }
 ++    return 0
 ++}
 ++
 ++proc dropcache {err} {
 ++    global allcwait nextarc cachedarcs seeds
 ++
 ++    #puts "dropping cache ($err)"
 ++    foreach v {arcnos arcout arcids arcstart arcend growing \
 ++                arctags archeads allparents allchildren} {
 ++     global $v
 ++     catch {unset $v}
 ++    }
 ++    set allcwait 0
 ++    set nextarc 0
 ++    set cachedarcs 0
 ++    set seeds {}
 ++    getallcommits
 ++}
 ++
 ++proc writecache {f} {
 ++    global cachearc cachedarcs allccache
 ++    global arcstart arcend arcnos arcids arcout
 ++
 ++    set a $cachearc
 ++    set lim $cachedarcs
 ++    if {$lim - $a > 1000} {
 ++     set lim [expr {$a + 1000}]
 ++    }
 ++    if {[catch {
 ++     while {[incr a] <= $lim} {
 ++         if {[info exists arcend($a)]} {
 ++             puts $f [list $arcstart($a) $arcend($a) $arcids($a)]
 ++         } else {
 ++             puts $f [list $arcstart($a) {} $arcids($a)]
 ++         }
 ++     }
 ++    } err]} {
 ++     catch {close $f}
 ++     catch {file delete $allccache}
 ++     #puts "writing cache failed ($err)"
 ++     return 0
 ++    }
 ++    set cachearc [expr {$a - 1}]
 ++    if {$a > $cachedarcs} {
 ++     puts $f "1"
 ++     close $f
 ++     return 0
 ++    }
 ++    return 1
 ++}
 ++
 ++proc savecache {} {
 ++    global nextarc cachedarcs cachearc allccache
 ++
 ++    if {$nextarc == $cachedarcs} return
 ++    set cachearc 0
 ++    set cachedarcs $nextarc
 ++    catch {
 ++     set f [open $allccache w]
 ++     puts $f [list 1 $cachedarcs]
 ++     run writecache $f
 ++    }
 ++}
 ++
 ++# Returns 1 if a is an ancestor of b, -1 if b is an ancestor of a,
 ++# or 0 if neither is true.
 ++proc anc_or_desc {a b} {
 ++    global arcout arcstart arcend arcnos cached_isanc
 ++
 ++    if {$arcnos($a) eq $arcnos($b)} {
 ++     # Both are on the same arc(s); either both are the same BMP,
 ++     # or if one is not a BMP, the other is also not a BMP or is
 ++     # the BMP at end of the arc (and it only has 1 incoming arc).
 ++     # Or both can be BMPs with no incoming arcs.
 ++     if {$a eq $b || $arcnos($a) eq {}} {
 ++         return 0
 ++     }
 ++     # assert {[llength $arcnos($a)] == 1}
 ++     set arc [lindex $arcnos($a) 0]
 ++     set i [lsearch -exact $arcids($arc) $a]
 ++     set j [lsearch -exact $arcids($arc) $b]
 ++     if {$i < 0 || $i > $j} {
 ++         return 1
 ++     } else {
 ++         return -1
 ++     }
 ++    }
 ++
 ++    if {![info exists arcout($a)]} {
 ++     set arc [lindex $arcnos($a) 0]
 ++     if {[info exists arcend($arc)]} {
 ++         set aend $arcend($arc)
 ++     } else {
 ++         set aend {}
 ++     }
 ++     set a $arcstart($arc)
 ++    } else {
 ++     set aend $a
 ++    }
 ++    if {![info exists arcout($b)]} {
 ++     set arc [lindex $arcnos($b) 0]
 ++     if {[info exists arcend($arc)]} {
 ++         set bend $arcend($arc)
 ++     } else {
 ++         set bend {}
 ++     }
 ++     set b $arcstart($arc)
 ++    } else {
 ++     set bend $b
 ++    }
 ++    if {$a eq $bend} {
 ++     return 1
 ++    }
 ++    if {$b eq $aend} {
 ++     return -1
 ++    }
 ++    if {[info exists cached_isanc($a,$bend)]} {
 ++     if {$cached_isanc($a,$bend)} {
 ++         return 1
 ++     }
 ++    }
 ++    if {[info exists cached_isanc($b,$aend)]} {
 ++     if {$cached_isanc($b,$aend)} {
 ++         return -1
 ++     }
 ++     if {[info exists cached_isanc($a,$bend)]} {
 ++         return 0
 ++     }
 ++    }
 ++
 ++    set todo [list $a $b]
 ++    set anc($a) a
 ++    set anc($b) b
 ++    for {set i 0} {$i < [llength $todo]} {incr i} {
 ++     set x [lindex $todo $i]
 ++     if {$anc($x) eq {}} {
 ++         continue
 ++     }
 ++     foreach arc $arcnos($x) {
 ++         set xd $arcstart($arc)
 ++         if {$xd eq $bend} {
 ++             set cached_isanc($a,$bend) 1
 ++             set cached_isanc($b,$aend) 0
 ++             return 1
 ++         } elseif {$xd eq $aend} {
 ++             set cached_isanc($b,$aend) 1
 ++             set cached_isanc($a,$bend) 0
 ++             return -1
 ++         }
 ++         if {![info exists anc($xd)]} {
 ++             set anc($xd) $anc($x)
 ++             lappend todo $xd
 ++         } elseif {$anc($xd) ne $anc($x)} {
 ++             set anc($xd) {}
 ++         }
 ++     }
 ++    }
 ++    set cached_isanc($a,$bend) 0
 ++    set cached_isanc($b,$aend) 0
 ++    return 0
 ++}
 ++
 ++# This identifies whether $desc has an ancestor that is
 ++# a growing tip of the graph and which is not an ancestor of $anc
 ++# and returns 0 if so and 1 if not.
 ++# If we subsequently discover a tag on such a growing tip, and that
 ++# turns out to be a descendent of $anc (which it could, since we
 ++# don't necessarily see children before parents), then $desc
 ++# isn't a good choice to display as a descendent tag of
 ++# $anc (since it is the descendent of another tag which is
 ++# a descendent of $anc).  Similarly, $anc isn't a good choice to
 ++# display as a ancestor tag of $desc.
 ++#
 ++proc is_certain {desc anc} {
 ++    global arcnos arcout arcstart arcend growing problems
 ++
 ++    set certain {}
 ++    if {[llength $arcnos($anc)] == 1} {
 ++     # tags on the same arc are certain
 ++     if {$arcnos($desc) eq $arcnos($anc)} {
 ++         return 1
 ++     }
 ++     if {![info exists arcout($anc)]} {
 ++         # if $anc is partway along an arc, use the start of the arc instead
 ++         set a [lindex $arcnos($anc) 0]
 ++         set anc $arcstart($a)
 ++     }
 ++    }
 ++    if {[llength $arcnos($desc)] > 1 || [info exists arcout($desc)]} {
 ++     set x $desc
 ++    } else {
 ++     set a [lindex $arcnos($desc) 0]
 ++     set x $arcend($a)
 ++    }
 ++    if {$x == $anc} {
 ++     return 1
 ++    }
 ++    set anclist [list $x]
 ++    set dl($x) 1
 ++    set nnh 1
 ++    set ngrowanc 0
 ++    for {set i 0} {$i < [llength $anclist] && ($nnh > 0 || $ngrowanc > 0)} {incr i} {
 ++     set x [lindex $anclist $i]
 ++     if {$dl($x)} {
 ++         incr nnh -1
 ++     }
 ++     set done($x) 1
 ++     foreach a $arcout($x) {
 ++         if {[info exists growing($a)]} {
 ++             if {![info exists growanc($x)] && $dl($x)} {
 ++                 set growanc($x) 1
 ++                 incr ngrowanc
 ++             }
 ++         } else {
 ++             set y $arcend($a)
 ++             if {[info exists dl($y)]} {
 ++                 if {$dl($y)} {
 ++                     if {!$dl($x)} {
 ++                         set dl($y) 0
 ++                         if {![info exists done($y)]} {
 ++                             incr nnh -1
 ++                         }
 ++                         if {[info exists growanc($x)]} {
 ++                             incr ngrowanc -1
 ++                         }
 ++                         set xl [list $y]
 ++                         for {set k 0} {$k < [llength $xl]} {incr k} {
 ++                             set z [lindex $xl $k]
 ++                             foreach c $arcout($z) {
 ++                                 if {[info exists arcend($c)]} {
 ++                                     set v $arcend($c)
 ++                                     if {[info exists dl($v)] && $dl($v)} {
 ++                                         set dl($v) 0
 ++                                         if {![info exists done($v)]} {
 ++                                             incr nnh -1
 ++                                         }
 ++                                         if {[info exists growanc($v)]} {
 ++                                             incr ngrowanc -1
 ++                                         }
 ++                                         lappend xl $v
 ++                                     }
 ++                                 }
 ++                             }
 ++                         }
 ++                     }
 ++                 }
 ++             } elseif {$y eq $anc || !$dl($x)} {
 ++                 set dl($y) 0
 ++                 lappend anclist $y
 ++             } else {
 ++                 set dl($y) 1
 ++                 lappend anclist $y
 ++                 incr nnh
 ++             }
 ++         }
 ++     }
 ++    }
 ++    foreach x [array names growanc] {
 ++     if {$dl($x)} {
 ++         return 0
 ++     }
 ++     return 0
 ++    }
 ++    return 1
 ++}
 ++
 ++proc validate_arctags {a} {
 ++    global arctags idtags
 ++
 ++    set i -1
 ++    set na $arctags($a)
 ++    foreach id $arctags($a) {
 ++     incr i
 ++     if {![info exists idtags($id)]} {
 ++         set na [lreplace $na $i $i]
 ++         incr i -1
 ++     }
 ++    }
 ++    set arctags($a) $na
 ++}
 ++
 ++proc validate_archeads {a} {
 ++    global archeads idheads
 ++
 ++    set i -1
 ++    set na $archeads($a)
 ++    foreach id $archeads($a) {
 ++     incr i
 ++     if {![info exists idheads($id)]} {
 ++         set na [lreplace $na $i $i]
 ++         incr i -1
 ++     }
 ++    }
 ++    set archeads($a) $na
 ++}
 ++
 ++# Return the list of IDs that have tags that are descendents of id,
 ++# ignoring IDs that are descendents of IDs already reported.
 ++proc desctags {id} {
 ++    global arcnos arcstart arcids arctags idtags allparents
 ++    global growing cached_dtags
 ++
 ++    if {![info exists allparents($id)]} {
 ++     return {}
 ++    }
 ++    set t1 [clock clicks -milliseconds]
 ++    set argid $id
 ++    if {[llength $arcnos($id)] == 1 && [llength $allparents($id)] == 1} {
 ++     # part-way along an arc; check that arc first
 ++     set a [lindex $arcnos($id) 0]
 ++     if {$arctags($a) ne {}} {
 ++         validate_arctags $a
 ++         set i [lsearch -exact $arcids($a) $id]
 ++         set tid {}
 ++         foreach t $arctags($a) {
 ++             set j [lsearch -exact $arcids($a) $t]
 ++             if {$j >= $i} break
 ++             set tid $t
 ++         }
 ++         if {$tid ne {}} {
 ++             return $tid
 ++         }
 ++     }
 ++     set id $arcstart($a)
 ++     if {[info exists idtags($id)]} {
 ++         return $id
 ++     }
 ++    }
 ++    if {[info exists cached_dtags($id)]} {
 ++     return $cached_dtags($id)
 ++    }
 ++
 ++    set origid $id
 ++    set todo [list $id]
 ++    set queued($id) 1
 ++    set nc 1
 ++    for {set i 0} {$i < [llength $todo] && $nc > 0} {incr i} {
 ++     set id [lindex $todo $i]
 ++     set done($id) 1
 ++     set ta [info exists hastaggedancestor($id)]
 ++     if {!$ta} {
 ++         incr nc -1
 ++     }
 ++     # ignore tags on starting node
 ++     if {!$ta && $i > 0} {
 ++         if {[info exists idtags($id)]} {
 ++             set tagloc($id) $id
 ++             set ta 1
 ++         } elseif {[info exists cached_dtags($id)]} {
 ++             set tagloc($id) $cached_dtags($id)
 ++             set ta 1
 ++         }
 ++     }
 ++     foreach a $arcnos($id) {
 ++         set d $arcstart($a)
 ++         if {!$ta && $arctags($a) ne {}} {
 ++             validate_arctags $a
 ++             if {$arctags($a) ne {}} {
 ++                 lappend tagloc($id) [lindex $arctags($a) end]
 ++             }
 ++         }
 ++         if {$ta || $arctags($a) ne {}} {
 ++             set tomark [list $d]
 ++             for {set j 0} {$j < [llength $tomark]} {incr j} {
 ++                 set dd [lindex $tomark $j]
 ++                 if {![info exists hastaggedancestor($dd)]} {
 ++                     if {[info exists done($dd)]} {
 ++                         foreach b $arcnos($dd) {
 ++                             lappend tomark $arcstart($b)
 ++                         }
 ++                         if {[info exists tagloc($dd)]} {
 ++                             unset tagloc($dd)
 ++                         }
 ++                     } elseif {[info exists queued($dd)]} {
 ++                         incr nc -1
 ++                     }
 ++                     set hastaggedancestor($dd) 1
 ++                 }
 ++             }
 ++         }
 ++         if {![info exists queued($d)]} {
 ++             lappend todo $d
 ++             set queued($d) 1
 ++             if {![info exists hastaggedancestor($d)]} {
 ++                 incr nc
 ++             }
 ++         }
 ++     }
 ++    }
 ++    set tags {}
 ++    foreach id [array names tagloc] {
 ++     if {![info exists hastaggedancestor($id)]} {
 ++         foreach t $tagloc($id) {
 ++             if {[lsearch -exact $tags $t] < 0} {
 ++                 lappend tags $t
 ++             }
 ++         }
 ++     }
 ++    }
 ++    set t2 [clock clicks -milliseconds]
 ++    set loopix $i
 ++
 ++    # remove tags that are descendents of other tags
 ++    for {set i 0} {$i < [llength $tags]} {incr i} {
 ++     set a [lindex $tags $i]
 ++     for {set j 0} {$j < $i} {incr j} {
 ++         set b [lindex $tags $j]
 ++         set r [anc_or_desc $a $b]
 ++         if {$r == 1} {
 ++             set tags [lreplace $tags $j $j]
 ++             incr j -1
 ++             incr i -1
 ++         } elseif {$r == -1} {
 ++             set tags [lreplace $tags $i $i]
 ++             incr i -1
 ++             break
 ++         }
 ++     }
 ++    }
 ++
 ++    if {[array names growing] ne {}} {
 ++     # graph isn't finished, need to check if any tag could get
 ++     # eclipsed by another tag coming later.  Simply ignore any
 ++     # tags that could later get eclipsed.
 ++     set ctags {}
 ++     foreach t $tags {
 ++         if {[is_certain $t $origid]} {
 ++             lappend ctags $t
 ++         }
 ++     }
 ++     if {$tags eq $ctags} {
 ++         set cached_dtags($origid) $tags
 ++     } else {
 ++         set tags $ctags
 ++     }
 ++    } else {
 ++     set cached_dtags($origid) $tags
 ++    }
 ++    set t3 [clock clicks -milliseconds]
 ++    if {0 && $t3 - $t1 >= 100} {
 ++     puts "iterating descendents ($loopix/[llength $todo] nodes) took\
 ++         [expr {$t2-$t1}]+[expr {$t3-$t2}]ms, $nc candidates left"
 ++    }
 ++    return $tags
 ++}
 ++
 ++proc anctags {id} {
 ++    global arcnos arcids arcout arcend arctags idtags allparents
 ++    global growing cached_atags
 ++
 ++    if {![info exists allparents($id)]} {
 ++     return {}
 ++    }
 ++    set t1 [clock clicks -milliseconds]
 ++    set argid $id
 ++    if {[llength $arcnos($id)] == 1 && [llength $allparents($id)] == 1} {
 ++     # part-way along an arc; check that arc first
 ++     set a [lindex $arcnos($id) 0]
 ++     if {$arctags($a) ne {}} {
 ++         validate_arctags $a
 ++         set i [lsearch -exact $arcids($a) $id]
 ++         foreach t $arctags($a) {
 ++             set j [lsearch -exact $arcids($a) $t]
 ++             if {$j > $i} {
 ++                 return $t
 ++             }
 ++         }
 ++     }
 ++     if {![info exists arcend($a)]} {
 ++         return {}
 ++     }
 ++     set id $arcend($a)
 ++     if {[info exists idtags($id)]} {
 ++         return $id
 ++     }
 ++    }
 ++    if {[info exists cached_atags($id)]} {
 ++     return $cached_atags($id)
 ++    }
 ++
 ++    set origid $id
 ++    set todo [list $id]
 ++    set queued($id) 1
 ++    set taglist {}
 ++    set nc 1
 ++    for {set i 0} {$i < [llength $todo] && $nc > 0} {incr i} {
 ++     set id [lindex $todo $i]
 ++     set done($id) 1
 ++     set td [info exists hastaggeddescendent($id)]
 ++     if {!$td} {
 ++         incr nc -1
 ++     }
 ++     # ignore tags on starting node
 ++     if {!$td && $i > 0} {
 ++         if {[info exists idtags($id)]} {
 ++             set tagloc($id) $id
 ++             set td 1
 ++         } elseif {[info exists cached_atags($id)]} {
 ++             set tagloc($id) $cached_atags($id)
 ++             set td 1
 ++         }
 ++     }
 ++     foreach a $arcout($id) {
 ++         if {!$td && $arctags($a) ne {}} {
 ++             validate_arctags $a
 ++             if {$arctags($a) ne {}} {
 ++                 lappend tagloc($id) [lindex $arctags($a) 0]
 ++             }
 ++         }
 ++         if {![info exists arcend($a)]} continue
 ++         set d $arcend($a)
 ++         if {$td || $arctags($a) ne {}} {
 ++             set tomark [list $d]
 ++             for {set j 0} {$j < [llength $tomark]} {incr j} {
 ++                 set dd [lindex $tomark $j]
 ++                 if {![info exists hastaggeddescendent($dd)]} {
 ++                     if {[info exists done($dd)]} {
 ++                         foreach b $arcout($dd) {
 ++                             if {[info exists arcend($b)]} {
 ++                                 lappend tomark $arcend($b)
 ++                             }
 ++                         }
 ++                         if {[info exists tagloc($dd)]} {
 ++                             unset tagloc($dd)
 ++                         }
 ++                     } elseif {[info exists queued($dd)]} {
 ++                         incr nc -1
 ++                     }
 ++                     set hastaggeddescendent($dd) 1
 ++                 }
 ++             }
 ++         }
 ++         if {![info exists queued($d)]} {
 ++             lappend todo $d
 ++             set queued($d) 1
 ++             if {![info exists hastaggeddescendent($d)]} {
 ++                 incr nc
 ++             }
 ++         }
 ++     }
 ++    }
 ++    set t2 [clock clicks -milliseconds]
 ++    set loopix $i
 ++    set tags {}
 ++    foreach id [array names tagloc] {
 ++     if {![info exists hastaggeddescendent($id)]} {
 ++         foreach t $tagloc($id) {
 ++             if {[lsearch -exact $tags $t] < 0} {
 ++                 lappend tags $t
 ++             }
 ++         }
 ++     }
 ++    }
 ++
 ++    # remove tags that are ancestors of other tags
 ++    for {set i 0} {$i < [llength $tags]} {incr i} {
 ++     set a [lindex $tags $i]
 ++     for {set j 0} {$j < $i} {incr j} {
 ++         set b [lindex $tags $j]
 ++         set r [anc_or_desc $a $b]
 ++         if {$r == -1} {
 ++             set tags [lreplace $tags $j $j]
 ++             incr j -1
 ++             incr i -1
 ++         } elseif {$r == 1} {
 ++             set tags [lreplace $tags $i $i]
 ++             incr i -1
 ++             break
 ++         }
 ++     }
 ++    }
 ++
 ++    if {[array names growing] ne {}} {
 ++     # graph isn't finished, need to check if any tag could get
 ++     # eclipsed by another tag coming later.  Simply ignore any
 ++     # tags that could later get eclipsed.
 ++     set ctags {}
 ++     foreach t $tags {
 ++         if {[is_certain $origid $t]} {
 ++             lappend ctags $t
 ++         }
 ++     }
 ++     if {$tags eq $ctags} {
 ++         set cached_atags($origid) $tags
 ++     } else {
 ++         set tags $ctags
 ++     }
 ++    } else {
 ++     set cached_atags($origid) $tags
 ++    }
 ++    set t3 [clock clicks -milliseconds]
 ++    if {0 && $t3 - $t1 >= 100} {
 ++     puts "iterating ancestors ($loopix/[llength $todo] nodes) took\
 ++         [expr {$t2-$t1}]+[expr {$t3-$t2}]ms, $nc candidates left"
 ++    }
 ++    return $tags
 ++}
 ++
 ++# Return the list of IDs that have heads that are descendents of id,
 ++# including id itself if it has a head.
 ++proc descheads {id} {
 ++    global arcnos arcstart arcids archeads idheads cached_dheads
 ++    global allparents
 ++
 ++    if {![info exists allparents($id)]} {
 ++     return {}
 ++    }
 ++    set aret {}
 ++    if {[llength $arcnos($id)] == 1 && [llength $allparents($id)] == 1} {
 ++     # part-way along an arc; check it first
 ++     set a [lindex $arcnos($id) 0]
 ++     if {$archeads($a) ne {}} {
 ++         validate_archeads $a
 ++         set i [lsearch -exact $arcids($a) $id]
 ++         foreach t $archeads($a) {
 ++             set j [lsearch -exact $arcids($a) $t]
 ++             if {$j > $i} break
 ++             lappend aret $t
 ++         }
 ++     }
 ++     set id $arcstart($a)
 ++    }
 ++    set origid $id
 ++    set todo [list $id]
 ++    set seen($id) 1
 ++    set ret {}
 ++    for {set i 0} {$i < [llength $todo]} {incr i} {
 ++     set id [lindex $todo $i]
 ++     if {[info exists cached_dheads($id)]} {
 ++         set ret [concat $ret $cached_dheads($id)]
 ++     } else {
 ++         if {[info exists idheads($id)]} {
 ++             lappend ret $id
 ++         }
 ++         foreach a $arcnos($id) {
 ++             if {$archeads($a) ne {}} {
 ++                 validate_archeads $a
 ++                 if {$archeads($a) ne {}} {
 ++                     set ret [concat $ret $archeads($a)]
 ++                 }
 ++             }
 ++             set d $arcstart($a)
 ++             if {![info exists seen($d)]} {
 ++                 lappend todo $d
 ++                 set seen($d) 1
 ++             }
 ++         }
 ++     }
 ++    }
 ++    set ret [lsort -unique $ret]
 ++    set cached_dheads($origid) $ret
 ++    return [concat $ret $aret]
 ++}
 ++
 ++proc addedtag {id} {
 ++    global arcnos arcout cached_dtags cached_atags
 ++
 ++    if {![info exists arcnos($id)]} return
 ++    if {![info exists arcout($id)]} {
 ++     recalcarc [lindex $arcnos($id) 0]
 ++    }
 ++    catch {unset cached_dtags}
 ++    catch {unset cached_atags}
 ++}
 ++
 ++proc addedhead {hid head} {
 ++    global arcnos arcout cached_dheads
 ++
 ++    if {![info exists arcnos($hid)]} return
 ++    if {![info exists arcout($hid)]} {
 ++     recalcarc [lindex $arcnos($hid) 0]
 ++    }
 ++    catch {unset cached_dheads}
 ++}
 ++
 ++proc removedhead {hid head} {
 ++    global cached_dheads
 ++
 ++    catch {unset cached_dheads}
 ++}
 ++
 ++proc movedhead {hid head} {
 ++    global arcnos arcout cached_dheads
 ++
 ++    if {![info exists arcnos($hid)]} return
 ++    if {![info exists arcout($hid)]} {
 ++     recalcarc [lindex $arcnos($hid) 0]
 ++    }
 ++    catch {unset cached_dheads}
 ++}
 ++
 ++proc changedrefs {} {
 ++    global cached_dheads cached_dtags cached_atags
 ++    global arctags archeads arcnos arcout idheads idtags
 ++
 ++    foreach id [concat [array names idheads] [array names idtags]] {
 ++     if {[info exists arcnos($id)] && ![info exists arcout($id)]} {
 ++         set a [lindex $arcnos($id) 0]
 ++         if {![info exists donearc($a)]} {
 ++             recalcarc $a
 ++             set donearc($a) 1
 ++         }
 ++     }
 ++    }
 ++    catch {unset cached_dtags}
 ++    catch {unset cached_atags}
 ++    catch {unset cached_dheads}
 ++}
 ++
 ++proc rereadrefs {} {
 ++    global idtags idheads idotherrefs mainheadid
 ++
 ++    set refids [concat [array names idtags] \
 ++                 [array names idheads] [array names idotherrefs]]
 ++    foreach id $refids {
 ++     if {![info exists ref($id)]} {
 ++         set ref($id) [listrefs $id]
 ++     }
 ++    }
 ++    set oldmainhead $mainheadid
 ++    readrefs
 ++    changedrefs
 ++    set refids [lsort -unique [concat $refids [array names idtags] \
 ++                     [array names idheads] [array names idotherrefs]]]
 ++    foreach id $refids {
 ++     set v [listrefs $id]
 ++     if {![info exists ref($id)] || $ref($id) != $v} {
 ++         redrawtags $id
 ++     }
 ++    }
 ++    if {$oldmainhead ne $mainheadid} {
 ++     redrawtags $oldmainhead
 ++     redrawtags $mainheadid
 ++    }
 ++    run refill_reflist
 ++}
 ++
 ++proc listrefs {id} {
 ++    global idtags idheads idotherrefs
 ++
 ++    set x {}
 ++    if {[info exists idtags($id)]} {
 ++     set x $idtags($id)
 ++    }
 ++    set y {}
 ++    if {[info exists idheads($id)]} {
 ++     set y $idheads($id)
 ++    }
 ++    set z {}
 ++    if {[info exists idotherrefs($id)]} {
 ++     set z $idotherrefs($id)
 ++    }
 ++    return [list $x $y $z]
 ++}
 ++
 ++proc showtag {tag isnew} {
 ++    global ctext tagcontents tagids linknum tagobjid
 ++
 ++    if {$isnew} {
 ++     addtohistory [list showtag $tag 0]
 ++    }
 ++    $ctext conf -state normal
 ++    clear_ctext
 ++    settabs 0
 ++    set linknum 0
 ++    if {![info exists tagcontents($tag)]} {
 ++     catch {
 ++         set tagcontents($tag) [exec git cat-file tag $tagobjid($tag)]
 ++     }
 ++    }
 ++    if {[info exists tagcontents($tag)]} {
 ++     set text $tagcontents($tag)
 ++    } else {
 ++     set text "[mc "Tag"]: $tag\n[mc "Id"]:  $tagids($tag)"
 ++    }
 ++    appendwithlinks $text {}
 ++    $ctext conf -state disabled
 ++    init_flist {}
 ++}
 ++
 ++proc doquit {} {
 ++    global stopped
 ++    global gitktmpdir
 ++
 ++    set stopped 100
 ++    savestuff .
 ++    destroy .
 ++
 ++    if {[info exists gitktmpdir]} {
 ++     catch {file delete -force $gitktmpdir}
 ++    }
 ++}
 ++
 ++proc mkfontdisp {font top which} {
 ++    global fontattr fontpref $font
 ++
 ++    set fontpref($font) [set $font]
 ++    button $top.${font}but -text $which -font optionfont \
 ++     -command [list choosefont $font $which]
 ++    label $top.$font -relief flat -font $font \
 ++     -text $fontattr($font,family) -justify left
 ++    grid x $top.${font}but $top.$font -sticky w
 ++}
 ++
 ++proc choosefont {font which} {
 ++    global fontparam fontlist fonttop fontattr
 ++
 ++    set fontparam(which) $which
 ++    set fontparam(font) $font
 ++    set fontparam(family) [font actual $font -family]
 ++    set fontparam(size) $fontattr($font,size)
 ++    set fontparam(weight) $fontattr($font,weight)
 ++    set fontparam(slant) $fontattr($font,slant)
 ++    set top .gitkfont
 ++    set fonttop $top
 ++    if {![winfo exists $top]} {
 ++     font create sample
 ++     eval font config sample [font actual $font]
 ++     toplevel $top
 ++     wm title $top [mc "Gitk font chooser"]
 ++     label $top.l -textvariable fontparam(which)
 ++     pack $top.l -side top
 ++     set fontlist [lsort [font families]]
 ++     frame $top.f
 ++     listbox $top.f.fam -listvariable fontlist \
 ++         -yscrollcommand [list $top.f.sb set]
 ++     bind $top.f.fam <<ListboxSelect>> selfontfam
 ++     scrollbar $top.f.sb -command [list $top.f.fam yview]
 ++     pack $top.f.sb -side right -fill y
 ++     pack $top.f.fam -side left -fill both -expand 1
 ++     pack $top.f -side top -fill both -expand 1
 ++     frame $top.g
 ++     spinbox $top.g.size -from 4 -to 40 -width 4 \
 ++         -textvariable fontparam(size) \
 ++         -validatecommand {string is integer -strict %s}
 ++     checkbutton $top.g.bold -padx 5 \
 ++         -font {{Times New Roman} 12 bold} -text [mc "B"] -indicatoron 0 \
 ++         -variable fontparam(weight) -onvalue bold -offvalue normal
 ++     checkbutton $top.g.ital -padx 5 \
 ++         -font {{Times New Roman} 12 italic} -text [mc "I"] -indicatoron 0  \
 ++         -variable fontparam(slant) -onvalue italic -offvalue roman
 ++     pack $top.g.size $top.g.bold $top.g.ital -side left
 ++     pack $top.g -side top
 ++     canvas $top.c -width 150 -height 50 -border 2 -relief sunk \
 ++         -background white
 ++     $top.c create text 100 25 -anchor center -text $which -font sample \
 ++         -fill black -tags text
 ++     bind $top.c <Configure> [list centertext $top.c]
 ++     pack $top.c -side top -fill x
 ++     frame $top.buts
 ++     button $top.buts.ok -text [mc "OK"] -command fontok -default active
 ++     button $top.buts.can -text [mc "Cancel"] -command fontcan -default normal
 ++     grid $top.buts.ok $top.buts.can
 ++     grid columnconfigure $top.buts 0 -weight 1 -uniform a
 ++     grid columnconfigure $top.buts 1 -weight 1 -uniform a
 ++     pack $top.buts -side bottom -fill x
 ++     trace add variable fontparam write chg_fontparam
 ++    } else {
 ++     raise $top
 ++     $top.c itemconf text -text $which
 ++    }
 ++    set i [lsearch -exact $fontlist $fontparam(family)]
 ++    if {$i >= 0} {
 ++     $top.f.fam selection set $i
 ++     $top.f.fam see $i
 ++    }
 ++}
 ++
 ++proc centertext {w} {
 ++    $w coords text [expr {[winfo width $w] / 2}] [expr {[winfo height $w] / 2}]
 ++}
 ++
 ++proc fontok {} {
 ++    global fontparam fontpref prefstop
 ++
 ++    set f $fontparam(font)
 ++    set fontpref($f) [list $fontparam(family) $fontparam(size)]
 ++    if {$fontparam(weight) eq "bold"} {
 ++     lappend fontpref($f) "bold"
 ++    }
 ++    if {$fontparam(slant) eq "italic"} {
 ++     lappend fontpref($f) "italic"
 ++    }
 ++    set w $prefstop.$f
 ++    $w conf -text $fontparam(family) -font $fontpref($f)
 ++     
 ++    fontcan
 ++}
 ++
 ++proc fontcan {} {
 ++    global fonttop fontparam
 ++
 ++    if {[info exists fonttop]} {
 ++     catch {destroy $fonttop}
 ++     catch {font delete sample}
 ++     unset fonttop
 ++     unset fontparam
 ++    }
 ++}
 ++
 ++proc selfontfam {} {
 ++    global fonttop fontparam
 ++
 ++    set i [$fonttop.f.fam curselection]
 ++    if {$i ne {}} {
 ++     set fontparam(family) [$fonttop.f.fam get $i]
 ++    }
 ++}
 ++
 ++proc chg_fontparam {v sub op} {
 ++    global fontparam
 ++
 ++    font config sample -$sub $fontparam($sub)
 ++}
 ++
 ++proc doprefs {} {
 ++    global maxwidth maxgraphpct
 ++    global oldprefs prefstop showneartags showlocalchanges
 ++    global bgcolor fgcolor ctext diffcolors selectbgcolor
 ++    global tabstop limitdiffs autoselect extdifftool
 ++
 ++    set top .gitkprefs
 ++    set prefstop $top
 ++    if {[winfo exists $top]} {
 ++     raise $top
 ++     return
 ++    }
 ++    foreach v {maxwidth maxgraphpct showneartags showlocalchanges \
 ++                limitdiffs tabstop} {
 ++     set oldprefs($v) [set $v]
 ++    }
 ++    toplevel $top
 ++    wm title $top [mc "Gitk preferences"]
 ++    label $top.ldisp -text [mc "Commit list display options"]
 ++    grid $top.ldisp - -sticky w -pady 10
 ++    label $top.spacer -text " "
 ++    label $top.maxwidthl -text [mc "Maximum graph width (lines)"] \
 ++     -font optionfont
 ++    spinbox $top.maxwidth -from 0 -to 100 -width 4 -textvariable maxwidth
 ++    grid $top.spacer $top.maxwidthl $top.maxwidth -sticky w
 ++    label $top.maxpctl -text [mc "Maximum graph width (% of pane)"] \
 ++     -font optionfont
 ++    spinbox $top.maxpct -from 1 -to 100 -width 4 -textvariable maxgraphpct
 ++    grid x $top.maxpctl $top.maxpct -sticky w
 ++    frame $top.showlocal
 ++    label $top.showlocal.l -text [mc "Show local changes"] -font optionfont
 ++    checkbutton $top.showlocal.b -variable showlocalchanges
 ++    pack $top.showlocal.b $top.showlocal.l -side left
 ++    grid x $top.showlocal -sticky w
 ++    frame $top.autoselect
 ++    label $top.autoselect.l -text [mc "Auto-select SHA1"] -font optionfont
 ++    checkbutton $top.autoselect.b -variable autoselect
 ++    pack $top.autoselect.b $top.autoselect.l -side left
 ++    grid x $top.autoselect -sticky w
 ++
 ++    label $top.ddisp -text [mc "Diff display options"]
 ++    grid $top.ddisp - -sticky w -pady 10
 ++    label $top.tabstopl -text [mc "Tab spacing"] -font optionfont
 ++    spinbox $top.tabstop -from 1 -to 20 -width 4 -textvariable tabstop
 ++    grid x $top.tabstopl $top.tabstop -sticky w
 ++    frame $top.ntag
 ++    label $top.ntag.l -text [mc "Display nearby tags"] -font optionfont
 ++    checkbutton $top.ntag.b -variable showneartags
 ++    pack $top.ntag.b $top.ntag.l -side left
 ++    grid x $top.ntag -sticky w
 ++    frame $top.ldiff
 ++    label $top.ldiff.l -text [mc "Limit diffs to listed paths"] -font optionfont
 ++    checkbutton $top.ldiff.b -variable limitdiffs
 ++    pack $top.ldiff.b $top.ldiff.l -side left
 ++    grid x $top.ldiff -sticky w
 ++
 ++    entry $top.extdifft -textvariable extdifftool
 ++    frame $top.extdifff
 ++    label $top.extdifff.l -text [mc "External diff tool" ] -font optionfont \
 ++     -padx 10
 ++    button $top.extdifff.b -text [mc "Choose..."] -font optionfont \
 ++     -command choose_extdiff
 ++    pack $top.extdifff.l $top.extdifff.b -side left
 ++    grid x $top.extdifff $top.extdifft -sticky w
 ++
 ++    label $top.cdisp -text [mc "Colors: press to choose"]
 ++    grid $top.cdisp - -sticky w -pady 10
 ++    label $top.bg -padx 40 -relief sunk -background $bgcolor
 ++    button $top.bgbut -text [mc "Background"] -font optionfont \
 ++     -command [list choosecolor bgcolor {} $top.bg background setbg]
 ++    grid x $top.bgbut $top.bg -sticky w
 ++    label $top.fg -padx 40 -relief sunk -background $fgcolor
 ++    button $top.fgbut -text [mc "Foreground"] -font optionfont \
 ++     -command [list choosecolor fgcolor {} $top.fg foreground setfg]
 ++    grid x $top.fgbut $top.fg -sticky w
 ++    label $top.diffold -padx 40 -relief sunk -background [lindex $diffcolors 0]
 ++    button $top.diffoldbut -text [mc "Diff: old lines"] -font optionfont \
 ++     -command [list choosecolor diffcolors 0 $top.diffold "diff old lines" \
 ++                   [list $ctext tag conf d0 -foreground]]
 ++    grid x $top.diffoldbut $top.diffold -sticky w
 ++    label $top.diffnew -padx 40 -relief sunk -background [lindex $diffcolors 1]
 ++    button $top.diffnewbut -text [mc "Diff: new lines"] -font optionfont \
 ++     -command [list choosecolor diffcolors 1 $top.diffnew "diff new lines" \
 ++                   [list $ctext tag conf d1 -foreground]]
 ++    grid x $top.diffnewbut $top.diffnew -sticky w
 ++    label $top.hunksep -padx 40 -relief sunk -background [lindex $diffcolors 2]
 ++    button $top.hunksepbut -text [mc "Diff: hunk header"] -font optionfont \
 ++     -command [list choosecolor diffcolors 2 $top.hunksep \
 ++                   "diff hunk header" \
 ++                   [list $ctext tag conf hunksep -foreground]]
 ++    grid x $top.hunksepbut $top.hunksep -sticky w
 ++    label $top.selbgsep -padx 40 -relief sunk -background $selectbgcolor
 ++    button $top.selbgbut -text [mc "Select bg"] -font optionfont \
 ++     -command [list choosecolor selectbgcolor {} $top.selbgsep background setselbg]
 ++    grid x $top.selbgbut $top.selbgsep -sticky w
 ++
 ++    label $top.cfont -text [mc "Fonts: press to choose"]
 ++    grid $top.cfont - -sticky w -pady 10
 ++    mkfontdisp mainfont $top [mc "Main font"]
 ++    mkfontdisp textfont $top [mc "Diff display font"]
 ++    mkfontdisp uifont $top [mc "User interface font"]
 ++
 ++    frame $top.buts
 ++    button $top.buts.ok -text [mc "OK"] -command prefsok -default active
 ++    button $top.buts.can -text [mc "Cancel"] -command prefscan -default normal
 ++    grid $top.buts.ok $top.buts.can
 ++    grid columnconfigure $top.buts 0 -weight 1 -uniform a
 ++    grid columnconfigure $top.buts 1 -weight 1 -uniform a
 ++    grid $top.buts - - -pady 10 -sticky ew
 ++    bind $top <Visibility> "focus $top.buts.ok"
 ++}
 ++
 ++proc choose_extdiff {} {
 ++    global extdifftool
 ++
 ++    set prog [tk_getOpenFile -title "External diff tool" -multiple false]
 ++    if {$prog ne {}} {
 ++     set extdifftool $prog
 ++    }
 ++}
 ++
 ++proc choosecolor {v vi w x cmd} {
 ++    global $v
 ++
 ++    set c [tk_chooseColor -initialcolor [lindex [set $v] $vi] \
 ++            -title [mc "Gitk: choose color for %s" $x]]
 ++    if {$c eq {}} return
 ++    $w conf -background $c
 ++    lset $v $vi $c
 ++    eval $cmd $c
 ++}
 ++
 ++proc setselbg {c} {
 ++    global bglist cflist
 ++    foreach w $bglist {
 ++     $w configure -selectbackground $c
 ++    }
 ++    $cflist tag configure highlight \
 ++     -background [$cflist cget -selectbackground]
 ++    allcanvs itemconf secsel -fill $c
 ++}
 ++
 ++proc setbg {c} {
 ++    global bglist
 ++
 ++    foreach w $bglist {
 ++     $w conf -background $c
 ++    }
 ++}
 ++
 ++proc setfg {c} {
 ++    global fglist canv
 ++
 ++    foreach w $fglist {
 ++     $w conf -foreground $c
 ++    }
 ++    allcanvs itemconf text -fill $c
 ++    $canv itemconf circle -outline $c
 ++}
 ++
 ++proc prefscan {} {
 ++    global oldprefs prefstop
 ++
 ++    foreach v {maxwidth maxgraphpct showneartags showlocalchanges \
 ++                limitdiffs tabstop} {
 ++     global $v
 ++     set $v $oldprefs($v)
 ++    }
 ++    catch {destroy $prefstop}
 ++    unset prefstop
 ++    fontcan
 ++}
 ++
 ++proc prefsok {} {
 ++    global maxwidth maxgraphpct
 ++    global oldprefs prefstop showneartags showlocalchanges
 ++    global fontpref mainfont textfont uifont
 ++    global limitdiffs treediffs
 ++
 ++    catch {destroy $prefstop}
 ++    unset prefstop
 ++    fontcan
 ++    set fontchanged 0
 ++    if {$mainfont ne $fontpref(mainfont)} {
 ++     set mainfont $fontpref(mainfont)
 ++     parsefont mainfont $mainfont
 ++     eval font configure mainfont [fontflags mainfont]
 ++     eval font configure mainfontbold [fontflags mainfont 1]
 ++     setcoords
 ++     set fontchanged 1
 ++    }
 ++    if {$textfont ne $fontpref(textfont)} {
 ++     set textfont $fontpref(textfont)
 ++     parsefont textfont $textfont
 ++     eval font configure textfont [fontflags textfont]
 ++     eval font configure textfontbold [fontflags textfont 1]
 ++    }
 ++    if {$uifont ne $fontpref(uifont)} {
 ++     set uifont $fontpref(uifont)
 ++     parsefont uifont $uifont
 ++     eval font configure uifont [fontflags uifont]
 ++    }
 ++    settabs
 ++    if {$showlocalchanges != $oldprefs(showlocalchanges)} {
 ++     if {$showlocalchanges} {
 ++         doshowlocalchanges
 ++     } else {
 ++         dohidelocalchanges
 ++     }
 ++    }
 ++    if {$limitdiffs != $oldprefs(limitdiffs)} {
 ++     # treediffs elements are limited by path
 ++     catch {unset treediffs}
 ++    }
 ++    if {$fontchanged || $maxwidth != $oldprefs(maxwidth)
 ++     || $maxgraphpct != $oldprefs(maxgraphpct)} {
 ++     redisplay
 ++    } elseif {$showneartags != $oldprefs(showneartags) ||
 ++       $limitdiffs != $oldprefs(limitdiffs)} {
 ++     reselectline
 ++    }
 ++}
 ++
 ++proc formatdate {d} {
 ++    global datetimeformat
 ++    if {$d ne {}} {
 ++     set d [clock format $d -format $datetimeformat]
 ++    }
 ++    return $d
 ++}
 ++
 ++# This list of encoding names and aliases is distilled from
 ++# http://www.iana.org/assignments/character-sets.
 ++# Not all of them are supported by Tcl.
 ++set encoding_aliases {
 ++    { ANSI_X3.4-1968 iso-ir-6 ANSI_X3.4-1986 ISO_646.irv:1991 ASCII
 ++      ISO646-US US-ASCII us IBM367 cp367 csASCII }
 ++    { ISO-10646-UTF-1 csISO10646UTF1 }
 ++    { ISO_646.basic:1983 ref csISO646basic1983 }
 ++    { INVARIANT csINVARIANT }
 ++    { ISO_646.irv:1983 iso-ir-2 irv csISO2IntlRefVersion }
 ++    { BS_4730 iso-ir-4 ISO646-GB gb uk csISO4UnitedKingdom }
 ++    { NATS-SEFI iso-ir-8-1 csNATSSEFI }
 ++    { NATS-SEFI-ADD iso-ir-8-2 csNATSSEFIADD }
 ++    { NATS-DANO iso-ir-9-1 csNATSDANO }
 ++    { NATS-DANO-ADD iso-ir-9-2 csNATSDANOADD }
 ++    { SEN_850200_B iso-ir-10 FI ISO646-FI ISO646-SE se csISO10Swedish }
 ++    { SEN_850200_C iso-ir-11 ISO646-SE2 se2 csISO11SwedishForNames }
 ++    { KS_C_5601-1987 iso-ir-149 KS_C_5601-1989 KSC_5601 korean csKSC56011987 }
 ++    { ISO-2022-KR csISO2022KR }
 ++    { EUC-KR csEUCKR }
 ++    { ISO-2022-JP csISO2022JP }
 ++    { ISO-2022-JP-2 csISO2022JP2 }
 ++    { JIS_C6220-1969-jp JIS_C6220-1969 iso-ir-13 katakana x0201-7
 ++      csISO13JISC6220jp }
 ++    { JIS_C6220-1969-ro iso-ir-14 jp ISO646-JP csISO14JISC6220ro }
 ++    { IT iso-ir-15 ISO646-IT csISO15Italian }
 ++    { PT iso-ir-16 ISO646-PT csISO16Portuguese }
 ++    { ES iso-ir-17 ISO646-ES csISO17Spanish }
 ++    { greek7-old iso-ir-18 csISO18Greek7Old }
 ++    { latin-greek iso-ir-19 csISO19LatinGreek }
 ++    { DIN_66003 iso-ir-21 de ISO646-DE csISO21German }
 ++    { NF_Z_62-010_(1973) iso-ir-25 ISO646-FR1 csISO25French }
 ++    { Latin-greek-1 iso-ir-27 csISO27LatinGreek1 }
 ++    { ISO_5427 iso-ir-37 csISO5427Cyrillic }
 ++    { JIS_C6226-1978 iso-ir-42 csISO42JISC62261978 }
 ++    { BS_viewdata iso-ir-47 csISO47BSViewdata }
 ++    { INIS iso-ir-49 csISO49INIS }
 ++    { INIS-8 iso-ir-50 csISO50INIS8 }
 ++    { INIS-cyrillic iso-ir-51 csISO51INISCyrillic }
 ++    { ISO_5427:1981 iso-ir-54 ISO5427Cyrillic1981 }
 ++    { ISO_5428:1980 iso-ir-55 csISO5428Greek }
 ++    { GB_1988-80 iso-ir-57 cn ISO646-CN csISO57GB1988 }
 ++    { GB_2312-80 iso-ir-58 chinese csISO58GB231280 }
 ++    { NS_4551-1 iso-ir-60 ISO646-NO no csISO60DanishNorwegian
 ++      csISO60Norwegian1 }
 ++    { NS_4551-2 ISO646-NO2 iso-ir-61 no2 csISO61Norwegian2 }
 ++    { NF_Z_62-010 iso-ir-69 ISO646-FR fr csISO69French }
 ++    { videotex-suppl iso-ir-70 csISO70VideotexSupp1 }
 ++    { PT2 iso-ir-84 ISO646-PT2 csISO84Portuguese2 }
 ++    { ES2 iso-ir-85 ISO646-ES2 csISO85Spanish2 }
 ++    { MSZ_7795.3 iso-ir-86 ISO646-HU hu csISO86Hungarian }
 ++    { JIS_C6226-1983 iso-ir-87 x0208 JIS_X0208-1983 csISO87JISX0208 }
 ++    { greek7 iso-ir-88 csISO88Greek7 }
 ++    { ASMO_449 ISO_9036 arabic7 iso-ir-89 csISO89ASMO449 }
 ++    { iso-ir-90 csISO90 }
 ++    { JIS_C6229-1984-a iso-ir-91 jp-ocr-a csISO91JISC62291984a }
 ++    { JIS_C6229-1984-b iso-ir-92 ISO646-JP-OCR-B jp-ocr-b
 ++      csISO92JISC62991984b }
 ++    { JIS_C6229-1984-b-add iso-ir-93 jp-ocr-b-add csISO93JIS62291984badd }
 ++    { JIS_C6229-1984-hand iso-ir-94 jp-ocr-hand csISO94JIS62291984hand }
 ++    { JIS_C6229-1984-hand-add iso-ir-95 jp-ocr-hand-add
 ++      csISO95JIS62291984handadd }
 ++    { JIS_C6229-1984-kana iso-ir-96 csISO96JISC62291984kana }
 ++    { ISO_2033-1983 iso-ir-98 e13b csISO2033 }
 ++    { ANSI_X3.110-1983 iso-ir-99 CSA_T500-1983 NAPLPS csISO99NAPLPS }
 ++    { ISO_8859-1:1987 iso-ir-100 ISO_8859-1 ISO-8859-1 latin1 l1 IBM819
 ++      CP819 csISOLatin1 }
 ++    { ISO_8859-2:1987 iso-ir-101 ISO_8859-2 ISO-8859-2 latin2 l2 csISOLatin2 }
 ++    { T.61-7bit iso-ir-102 csISO102T617bit }
 ++    { T.61-8bit T.61 iso-ir-103 csISO103T618bit }
 ++    { ISO_8859-3:1988 iso-ir-109 ISO_8859-3 ISO-8859-3 latin3 l3 csISOLatin3 }
 ++    { ISO_8859-4:1988 iso-ir-110 ISO_8859-4 ISO-8859-4 latin4 l4 csISOLatin4 }
 ++    { ECMA-cyrillic iso-ir-111 KOI8-E csISO111ECMACyrillic }
 ++    { CSA_Z243.4-1985-1 iso-ir-121 ISO646-CA csa7-1 ca csISO121Canadian1 }
 ++    { CSA_Z243.4-1985-2 iso-ir-122 ISO646-CA2 csa7-2 csISO122Canadian2 }
 ++    { CSA_Z243.4-1985-gr iso-ir-123 csISO123CSAZ24341985gr }
 ++    { ISO_8859-6:1987 iso-ir-127 ISO_8859-6 ISO-8859-6 ECMA-114 ASMO-708
 ++      arabic csISOLatinArabic }
 ++    { ISO_8859-6-E csISO88596E ISO-8859-6-E }
 ++    { ISO_8859-6-I csISO88596I ISO-8859-6-I }
 ++    { ISO_8859-7:1987 iso-ir-126 ISO_8859-7 ISO-8859-7 ELOT_928 ECMA-118
 ++      greek greek8 csISOLatinGreek }
 ++    { T.101-G2 iso-ir-128 csISO128T101G2 }
 ++    { ISO_8859-8:1988 iso-ir-138 ISO_8859-8 ISO-8859-8 hebrew
 ++      csISOLatinHebrew }
 ++    { ISO_8859-8-E csISO88598E ISO-8859-8-E }
 ++    { ISO_8859-8-I csISO88598I ISO-8859-8-I }
 ++    { CSN_369103 iso-ir-139 csISO139CSN369103 }
 ++    { JUS_I.B1.002 iso-ir-141 ISO646-YU js yu csISO141JUSIB1002 }
 ++    { ISO_6937-2-add iso-ir-142 csISOTextComm }
 ++    { IEC_P27-1 iso-ir-143 csISO143IECP271 }
 ++    { ISO_8859-5:1988 iso-ir-144 ISO_8859-5 ISO-8859-5 cyrillic
 ++      csISOLatinCyrillic }
 ++    { JUS_I.B1.003-serb iso-ir-146 serbian csISO146Serbian }
 ++    { JUS_I.B1.003-mac macedonian iso-ir-147 csISO147Macedonian }
 ++    { ISO_8859-9:1989 iso-ir-148 ISO_8859-9 ISO-8859-9 latin5 l5 csISOLatin5 }
 ++    { greek-ccitt iso-ir-150 csISO150 csISO150GreekCCITT }
 ++    { NC_NC00-10:81 cuba iso-ir-151 ISO646-CU csISO151Cuba }
 ++    { ISO_6937-2-25 iso-ir-152 csISO6937Add }
 ++    { GOST_19768-74 ST_SEV_358-88 iso-ir-153 csISO153GOST1976874 }
 ++    { ISO_8859-supp iso-ir-154 latin1-2-5 csISO8859Supp }
 ++    { ISO_10367-box iso-ir-155 csISO10367Box }
 ++    { ISO-8859-10 iso-ir-157 l6 ISO_8859-10:1992 csISOLatin6 latin6 }
 ++    { latin-lap lap iso-ir-158 csISO158Lap }
 ++    { JIS_X0212-1990 x0212 iso-ir-159 csISO159JISX02121990 }
 ++    { DS_2089 DS2089 ISO646-DK dk csISO646Danish }
 ++    { us-dk csUSDK }
 ++    { dk-us csDKUS }
 ++    { JIS_X0201 X0201 csHalfWidthKatakana }
 ++    { KSC5636 ISO646-KR csKSC5636 }
 ++    { ISO-10646-UCS-2 csUnicode }
 ++    { ISO-10646-UCS-4 csUCS4 }
 ++    { DEC-MCS dec csDECMCS }
 ++    { hp-roman8 roman8 r8 csHPRoman8 }
 ++    { macintosh mac csMacintosh }
 ++    { IBM037 cp037 ebcdic-cp-us ebcdic-cp-ca ebcdic-cp-wt ebcdic-cp-nl
 ++      csIBM037 }
 ++    { IBM038 EBCDIC-INT cp038 csIBM038 }
 ++    { IBM273 CP273 csIBM273 }
 ++    { IBM274 EBCDIC-BE CP274 csIBM274 }
 ++    { IBM275 EBCDIC-BR cp275 csIBM275 }
 ++    { IBM277 EBCDIC-CP-DK EBCDIC-CP-NO csIBM277 }
 ++    { IBM278 CP278 ebcdic-cp-fi ebcdic-cp-se csIBM278 }
 ++    { IBM280 CP280 ebcdic-cp-it csIBM280 }
 ++    { IBM281 EBCDIC-JP-E cp281 csIBM281 }
 ++    { IBM284 CP284 ebcdic-cp-es csIBM284 }
 ++    { IBM285 CP285 ebcdic-cp-gb csIBM285 }
 ++    { IBM290 cp290 EBCDIC-JP-kana csIBM290 }
 ++    { IBM297 cp297 ebcdic-cp-fr csIBM297 }
 ++    { IBM420 cp420 ebcdic-cp-ar1 csIBM420 }
 ++    { IBM423 cp423 ebcdic-cp-gr csIBM423 }
 ++    { IBM424 cp424 ebcdic-cp-he csIBM424 }
 ++    { IBM437 cp437 437 csPC8CodePage437 }
 ++    { IBM500 CP500 ebcdic-cp-be ebcdic-cp-ch csIBM500 }
 ++    { IBM775 cp775 csPC775Baltic }
 ++    { IBM850 cp850 850 csPC850Multilingual }
 ++    { IBM851 cp851 851 csIBM851 }
 ++    { IBM852 cp852 852 csPCp852 }
 ++    { IBM855 cp855 855 csIBM855 }
 ++    { IBM857 cp857 857 csIBM857 }
 ++    { IBM860 cp860 860 csIBM860 }
 ++    { IBM861 cp861 861 cp-is csIBM861 }
 ++    { IBM862 cp862 862 csPC862LatinHebrew }
 ++    { IBM863 cp863 863 csIBM863 }
 ++    { IBM864 cp864 csIBM864 }
 ++    { IBM865 cp865 865 csIBM865 }
 ++    { IBM866 cp866 866 csIBM866 }
 ++    { IBM868 CP868 cp-ar csIBM868 }
 ++    { IBM869 cp869 869 cp-gr csIBM869 }
 ++    { IBM870 CP870 ebcdic-cp-roece ebcdic-cp-yu csIBM870 }
 ++    { IBM871 CP871 ebcdic-cp-is csIBM871 }
 ++    { IBM880 cp880 EBCDIC-Cyrillic csIBM880 }
 ++    { IBM891 cp891 csIBM891 }
 ++    { IBM903 cp903 csIBM903 }
 ++    { IBM904 cp904 904 csIBBM904 }
 ++    { IBM905 CP905 ebcdic-cp-tr csIBM905 }
 ++    { IBM918 CP918 ebcdic-cp-ar2 csIBM918 }
 ++    { IBM1026 CP1026 csIBM1026 }
 ++    { EBCDIC-AT-DE csIBMEBCDICATDE }
 ++    { EBCDIC-AT-DE-A csEBCDICATDEA }
 ++    { EBCDIC-CA-FR csEBCDICCAFR }
 ++    { EBCDIC-DK-NO csEBCDICDKNO }
 ++    { EBCDIC-DK-NO-A csEBCDICDKNOA }
 ++    { EBCDIC-FI-SE csEBCDICFISE }
 ++    { EBCDIC-FI-SE-A csEBCDICFISEA }
 ++    { EBCDIC-FR csEBCDICFR }
 ++    { EBCDIC-IT csEBCDICIT }
 ++    { EBCDIC-PT csEBCDICPT }
 ++    { EBCDIC-ES csEBCDICES }
 ++    { EBCDIC-ES-A csEBCDICESA }
 ++    { EBCDIC-ES-S csEBCDICESS }
 ++    { EBCDIC-UK csEBCDICUK }
 ++    { EBCDIC-US csEBCDICUS }
 ++    { UNKNOWN-8BIT csUnknown8BiT }
 ++    { MNEMONIC csMnemonic }
 ++    { MNEM csMnem }
 ++    { VISCII csVISCII }
 ++    { VIQR csVIQR }
 ++    { KOI8-R csKOI8R }
 ++    { IBM00858 CCSID00858 CP00858 PC-Multilingual-850+euro }
 ++    { IBM00924 CCSID00924 CP00924 ebcdic-Latin9--euro }
 ++    { IBM01140 CCSID01140 CP01140 ebcdic-us-37+euro }
 ++    { IBM01141 CCSID01141 CP01141 ebcdic-de-273+euro }
 ++    { IBM01142 CCSID01142 CP01142 ebcdic-dk-277+euro ebcdic-no-277+euro }
 ++    { IBM01143 CCSID01143 CP01143 ebcdic-fi-278+euro ebcdic-se-278+euro }
 ++    { IBM01144 CCSID01144 CP01144 ebcdic-it-280+euro }
 ++    { IBM01145 CCSID01145 CP01145 ebcdic-es-284+euro }
 ++    { IBM01146 CCSID01146 CP01146 ebcdic-gb-285+euro }
 ++    { IBM01147 CCSID01147 CP01147 ebcdic-fr-297+euro }
 ++    { IBM01148 CCSID01148 CP01148 ebcdic-international-500+euro }
 ++    { IBM01149 CCSID01149 CP01149 ebcdic-is-871+euro }
 ++    { IBM1047 IBM-1047 }
 ++    { PTCP154 csPTCP154 PT154 CP154 Cyrillic-Asian }
 ++    { Amiga-1251 Ami1251 Amiga1251 Ami-1251 }
 ++    { UNICODE-1-1 csUnicode11 }
 ++    { CESU-8 csCESU-8 }
 ++    { BOCU-1 csBOCU-1 }
 ++    { UNICODE-1-1-UTF-7 csUnicode11UTF7 }
 ++    { ISO-8859-14 iso-ir-199 ISO_8859-14:1998 ISO_8859-14 latin8 iso-celtic
 ++      l8 }
 ++    { ISO-8859-15 ISO_8859-15 Latin-9 }
 ++    { ISO-8859-16 iso-ir-226 ISO_8859-16:2001 ISO_8859-16 latin10 l10 }
 ++    { GBK CP936 MS936 windows-936 }
 ++    { JIS_Encoding csJISEncoding }
 ++    { Shift_JIS MS_Kanji csShiftJIS }
 ++    { Extended_UNIX_Code_Packed_Format_for_Japanese csEUCPkdFmtJapanese
 ++      EUC-JP }
 ++    { Extended_UNIX_Code_Fixed_Width_for_Japanese csEUCFixWidJapanese }
 ++    { ISO-10646-UCS-Basic csUnicodeASCII }
 ++    { ISO-10646-Unicode-Latin1 csUnicodeLatin1 ISO-10646 }
 ++    { ISO-Unicode-IBM-1261 csUnicodeIBM1261 }
 ++    { ISO-Unicode-IBM-1268 csUnicodeIBM1268 }
 ++    { ISO-Unicode-IBM-1276 csUnicodeIBM1276 }
 ++    { ISO-Unicode-IBM-1264 csUnicodeIBM1264 }
 ++    { ISO-Unicode-IBM-1265 csUnicodeIBM1265 }
 ++    { ISO-8859-1-Windows-3.0-Latin-1 csWindows30Latin1 }
 ++    { ISO-8859-1-Windows-3.1-Latin-1 csWindows31Latin1 }
 ++    { ISO-8859-2-Windows-Latin-2 csWindows31Latin2 }
 ++    { ISO-8859-9-Windows-Latin-5 csWindows31Latin5 }
 ++    { Adobe-Standard-Encoding csAdobeStandardEncoding }
 ++    { Ventura-US csVenturaUS }
 ++    { Ventura-International csVenturaInternational }
 ++    { PC8-Danish-Norwegian csPC8DanishNorwegian }
 ++    { PC8-Turkish csPC8Turkish }
 ++    { IBM-Symbols csIBMSymbols }
 ++    { IBM-Thai csIBMThai }
 ++    { HP-Legal csHPLegal }
 ++    { HP-Pi-font csHPPiFont }
 ++    { HP-Math8 csHPMath8 }
 ++    { Adobe-Symbol-Encoding csHPPSMath }
 ++    { HP-DeskTop csHPDesktop }
 ++    { Ventura-Math csVenturaMath }
 ++    { Microsoft-Publishing csMicrosoftPublishing }
 ++    { Windows-31J csWindows31J }
 ++    { GB2312 csGB2312 }
 ++    { Big5 csBig5 }
 ++}
 ++
 ++proc tcl_encoding {enc} {
 ++    global encoding_aliases
 ++    set names [encoding names]
 ++    set lcnames [string tolower $names]
 ++    set enc [string tolower $enc]
 ++    set i [lsearch -exact $lcnames $enc]
 ++    if {$i < 0} {
 ++     # look for "isonnn" instead of "iso-nnn" or "iso_nnn"
 ++     if {[regsub {^iso[-_]} $enc iso encx]} {
 ++         set i [lsearch -exact $lcnames $encx]
 ++     }
 ++    }
 ++    if {$i < 0} {
 ++     foreach l $encoding_aliases {
 ++         set ll [string tolower $l]
 ++         if {[lsearch -exact $ll $enc] < 0} continue
 ++         # look through the aliases for one that tcl knows about
 ++         foreach e $ll {
 ++             set i [lsearch -exact $lcnames $e]
 ++             if {$i < 0} {
 ++                 if {[regsub {^iso[-_]} $e iso ex]} {
 ++                     set i [lsearch -exact $lcnames $ex]
 ++                 }
 ++             }
 ++             if {$i >= 0} break
 ++         }
 ++         break
 ++     }
 ++    }
 ++    if {$i >= 0} {
 ++     return [lindex $names $i]
 ++    }
 ++    return {}
 ++}
 ++
 ++# First check that Tcl/Tk is recent enough
 ++if {[catch {package require Tk 8.4} err]} {
 ++    show_error {} . [mc "Sorry, gitk cannot run with this version of Tcl/Tk.\n\
 ++                  Gitk requires at least Tcl/Tk 8.4."]
 ++    exit 1
 ++}
 ++
 ++# defaults...
 ++set wrcomcmd "git diff-tree --stdin -p --pretty"
 ++
 ++set gitencoding {}
 ++catch {
 ++    set gitencoding [exec git config --get i18n.commitencoding]
 ++}
 ++if {$gitencoding == ""} {
 ++    set gitencoding "utf-8"
 ++}
 ++set tclencoding [tcl_encoding $gitencoding]
 ++if {$tclencoding == {}} {
 ++    puts stderr "Warning: encoding $gitencoding is not supported by Tcl/Tk"
 ++}
 ++
 ++set mainfont {Helvetica 9}
 ++set textfont {Courier 9}
 ++set uifont {Helvetica 9 bold}
 ++set tabstop 8
 ++set findmergefiles 0
 ++set maxgraphpct 50
 ++set maxwidth 16
 ++set revlistorder 0
 ++set fastdate 0
 ++set uparrowlen 5
 ++set downarrowlen 5
 ++set mingaplen 100
 ++set cmitmode "patch"
 ++set wrapcomment "none"
 ++set showneartags 1
 ++set maxrefs 20
 ++set maxlinelen 200
 ++set showlocalchanges 1
 ++set limitdiffs 1
 ++set datetimeformat "%Y-%m-%d %H:%M:%S"
 ++set autoselect 1
 ++
 ++set extdifftool "meld"
 ++
 ++set colors {green red blue magenta darkgrey brown orange}
 ++set bgcolor white
 ++set fgcolor black
 ++set diffcolors {red "#00a000" blue}
 ++set diffcontext 3
 ++set ignorespace 0
 ++set selectbgcolor gray85
 ++
 ++set circlecolors {white blue gray blue blue}
 ++
 ++## For msgcat loading, first locate the installation location.
 ++if { [info exists ::env(GITK_MSGSDIR)] } {
 ++    ## Msgsdir was manually set in the environment.
 ++    set gitk_msgsdir $::env(GITK_MSGSDIR)
 ++} else {
 ++    ## Let's guess the prefix from argv0.
 ++    set gitk_prefix [file dirname [file dirname [file normalize $argv0]]]
 ++    set gitk_libdir [file join $gitk_prefix share gitk lib]
 ++    set gitk_msgsdir [file join $gitk_libdir msgs]
 ++    unset gitk_prefix
 ++}
 ++
 ++## Internationalization (i18n) through msgcat and gettext. See
 ++## http://www.gnu.org/software/gettext/manual/html_node/Tcl.html
 ++package require msgcat
 ++namespace import ::msgcat::mc
 ++## And eventually load the actual message catalog
 ++::msgcat::mcload $gitk_msgsdir
 ++
 ++catch {source ~/.gitk}
 ++
 ++font create optionfont -family sans-serif -size -12
 ++
 ++parsefont mainfont $mainfont
 ++eval font create mainfont [fontflags mainfont]
 ++eval font create mainfontbold [fontflags mainfont 1]
 ++
 ++parsefont textfont $textfont
 ++eval font create textfont [fontflags textfont]
 ++eval font create textfontbold [fontflags textfont 1]
 ++
 ++parsefont uifont $uifont
 ++eval font create uifont [fontflags uifont]
 ++
 ++setoptions
 ++
 ++# check that we can find a .git directory somewhere...
 ++if {[catch {set gitdir [gitdir]}]} {
 ++    show_error {} . [mc "Cannot find a git repository here."]
 ++    exit 1
 ++}
 ++if {![file isdirectory $gitdir]} {
 ++    show_error {} . [mc "Cannot find the git directory \"%s\"." $gitdir]
 ++    exit 1
 ++}
 ++
+++set selecthead {}
+++set selectheadid {}
+++
 ++set revtreeargs {}
 ++set cmdline_files {}
 ++set i 0
 ++set revtreeargscmd {}
 ++foreach arg $argv {
 ++    switch -glob -- $arg {
 ++     "" { }
 ++     "--" {
 ++         set cmdline_files [lrange $argv [expr {$i + 1}] end]
 ++         break
 ++     }
+++     "--select-commit=*" {
+++         set selecthead [string range $arg 16 end]
+++     }
 ++     "--argscmd=*" {
 ++         set revtreeargscmd [string range $arg 10 end]
 ++     }
 ++     default {
 ++         lappend revtreeargs $arg
 ++     }
 ++    }
 ++    incr i
 ++}
 ++
+++if {$selecthead eq "HEAD"} {
+++    set selecthead {}
+++}
+++
 ++if {$i >= [llength $argv] && $revtreeargs ne {}} {
 ++    # no -- on command line, but some arguments (other than --argscmd)
 ++    if {[catch {
 ++     set f [eval exec git rev-parse --no-revs --no-flags $revtreeargs]
 ++     set cmdline_files [split $f "\n"]
 ++     set n [llength $cmdline_files]
 ++     set revtreeargs [lrange $revtreeargs 0 end-$n]
 ++     # Unfortunately git rev-parse doesn't produce an error when
 ++     # something is both a revision and a filename.  To be consistent
 ++     # with git log and git rev-list, check revtreeargs for filenames.
 ++     foreach arg $revtreeargs {
 ++         if {[file exists $arg]} {
 ++             show_error {} . [mc "Ambiguous argument '%s': both revision\
 ++                              and filename" $arg]
 ++             exit 1
 ++         }
 ++     }
 ++    } err]} {
 ++     # unfortunately we get both stdout and stderr in $err,
 ++     # so look for "fatal:".
 ++     set i [string first "fatal:" $err]
 ++     if {$i > 0} {
 ++         set err [string range $err [expr {$i + 6}] end]
 ++     }
 ++     show_error {} . "[mc "Bad arguments to gitk:"]\n$err"
 ++     exit 1
 ++    }
 ++}
 ++
 ++set nullid "0000000000000000000000000000000000000000"
 ++set nullid2 "0000000000000000000000000000000000000001"
 ++set nullfile "/dev/null"
 ++
 ++set have_tk85 [expr {[package vcompare $tk_version "8.5"] >= 0}]
 ++
 ++set runq {}
 ++set history {}
 ++set historyindex 0
 ++set fh_serial 0
 ++set nhl_names {}
 ++set highlight_paths {}
 ++set findpattern {}
 ++set searchdirn -forwards
 ++set boldrows {}
 ++set boldnamerows {}
 ++set diffelide {0 0}
 ++set markingmatches 0
 ++set linkentercount 0
 ++set need_redisplay 0
 ++set nrows_drawn 0
 ++set firsttabstop 0
 ++
 ++set nextviewnum 1
 ++set curview 0
 ++set selectedview 0
 ++set selectedhlview [mc "None"]
 ++set highlight_related [mc "None"]
 ++set highlight_files {}
 ++set viewfiles(0) {}
 ++set viewperm(0) 0
 ++set viewargs(0) {}
 ++set viewargscmd(0) {}
 ++
 ++set selectedline {}
 ++set numcommits 0
 ++set loginstance 0
 ++set cmdlineok 0
 ++set stopped 0
 ++set stuffsaved 0
 ++set patchnum 0
 ++set lserial 0
 ++set isworktree [expr {[exec git rev-parse --is-inside-work-tree] == "true"}]
 ++setcoords
 ++makewindow
 ++# wait for the window to become visible
 ++tkwait visibility .
 ++wm title . "[file tail $argv0]: [file tail [pwd]]"
 ++readrefs
 ++
 ++if {$cmdline_files ne {} || $revtreeargs ne {} || $revtreeargscmd ne {}} {
 ++    # create a view for the files/dirs specified on the command line
 ++    set curview 1
 ++    set selectedview 1
 ++    set nextviewnum 2
 ++    set viewname(1) [mc "Command line"]
 ++    set viewfiles(1) $cmdline_files
 ++    set viewargs(1) $revtreeargs
 ++    set viewargscmd(1) $revtreeargscmd
 ++    set viewperm(1) 0
 ++    set vdatemode(1) 0
 ++    addviewmenu 1
 ++    .bar.view entryconf [mc "Edit view..."] -state normal
 ++    .bar.view entryconf [mc "Delete view"] -state normal
 ++}
 ++
 ++if {[info exists permviews]} {
 ++    foreach v $permviews {
 ++     set n $nextviewnum
 ++     incr nextviewnum
 ++     set viewname($n) [lindex $v 0]
 ++     set viewfiles($n) [lindex $v 1]
 ++     set viewargs($n) [lindex $v 2]
 ++     set viewargscmd($n) [lindex $v 3]
 ++     set viewperm($n) 1
 ++     addviewmenu $n
 ++    }
 ++}
 ++getcommits {}