Merge branch 'master' into new
authorPaul Mackerras <paulus@samba.org>
Tue, 30 May 2006 11:35:07 +0000 (21:35 +1000)
committerPaul Mackerras <paulus@samba.org>
Tue, 30 May 2006 11:35:07 +0000 (21:35 +1000)
Conflicts:

gitk

1  2 
gitk
diff --combined gitk
index 7c8ad595c04537369b4c13e94fa0eadb93ea05be,101cf9bd9ff8c3b8180ab50e6ceb4655576f9fbf..25c86b70af6284eacd079e508f97f16d03f3f88d
--- 1/gitk
--- 2/gitk
+++ b/gitk
@@@ -34,10 -34,10 +34,10 @@@ proc start_rev_list {view} 
        set order "--date-order"
      }
      if {[catch {
-       set fd [open [concat | git-rev-list --header $order \
+       set fd [open [concat | git rev-list --header $order \
                          --parents --boundary --default HEAD $args] r]
      } err]} {
-       puts stderr "Error executing git-rev-list: $err"
+       puts stderr "Error executing git rev-list: $err"
        exit 1
      }
      set commfd($view) $fd
@@@ -94,10 -94,10 +94,10 @@@ proc getcommitlines {fd view}  
            }
            if {[string range $err 0 4] == "usage"} {
                set err "Gitk: error reading commits$fv:\
-                       bad arguments to git-rev-list."
+                       bad arguments to git rev-list."
                if {$viewname($view) eq "Command line"} {
                    append err \
-                       "  (Note: arguments to gitk are passed to git-rev-list\
+                       "  (Note: arguments to gitk are passed to git rev-list\
                         to allow selection of commits to be displayed.)"
                }
            } else {
            if {[string length $shortcmit] > 80} {
                set shortcmit "[string range $shortcmit 0 80]..."
            }
-           error_popup "Can't parse git-rev-list output: {$shortcmit}"
+           error_popup "Can't parse git rev-list output: {$shortcmit}"
            exit 1
        }
        set id [lindex $ids 0]
        if {$view == $curview} {
            layoutmore
        } elseif {[info exists hlview] && $view == $hlview} {
 -          highlightmore
 +          vhighlightmore
        }
      }
      if {[clock clicks -milliseconds] >= $nextupdate} {
@@@ -217,13 -217,13 +217,13 @@@ proc doupdate {} 
  }
  
  proc readcommit {id} {
-     if {[catch {set contents [exec git-cat-file commit $id]}]} return
+     if {[catch {set contents [exec git cat-file commit $id]}]} return
      parsecommit $id $contents 0
  }
  
  proc updatecommits {} {
      global viewdata curview phase displayorder
 -    global children commitrow
 +    global children commitrow selectedline thickerline
  
      if {$phase ne {}} {
        stop_rev_list
        catch {unset commitrow($n,$id)}
      }
      set curview -1
 +    catch {unset selectedline}
 +    catch {unset thickerline}
      catch {unset viewdata($n)}
      readrefs
      showview $n
@@@ -278,8 -276,8 +278,8 @@@ proc parsecommit {id contents listed} 
        set headline $comment
      }
      if {!$listed} {
-       # git-rev-list indents the comment by 4 spaces;
-       # if we got this via git-cat-file, add the indentation
+       # git rev-list 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 "    "
@@@ -339,14 -337,14 +339,14 @@@ proc readrefs {} 
            set type {}
            set tag {}
            catch {
-               set commit [exec git-rev-parse "$id^0"]
+               set commit [exec git rev-parse "$id^0"]
                if {"$commit" != "$id"} {
                    set tagids($name) $commit
                    lappend idtags($commit) $name
                }
            }           
            catch {
-               set tagcontents($name) [exec git-cat-file tag "$id"]
+               set tagcontents($name) [exec git cat-file tag "$id"]
            }
        } elseif { $type == "heads" } {
            set headids($name) $id
      close $refd
  }
  
- proc show_error {w msg} {
+ 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 OK -command "destroy $w"
+     button $w.ok -text OK -command "destroy $top"
      pack $w.ok -side bottom -fill x
-     bind $w <Visibility> "grab $w; focus $w"
-     bind $w <Key-Return> "destroy $w"
-     tkwait window $w
+     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 $msg
+     show_error $w $w $msg
  }
  
  proc makewindow {} {
      global findtype findtypemenu findloc findstring fstring geometry
      global entries sha1entry sha1string sha1but
      global maincursor textcursor curtextcursor
-     global rowctxmenu mergemax
+     global rowctxmenu mergemax wrapcomment
 +    global highlight_files gdttype
 +    global searchstring sstring
  
      menu .bar
      .bar add cascade -label "File" -menu .bar.file
      .bar.edit configure -font $uifont
  
      menu .bar.view -font $uifont
 -    menu .bar.view.hl -font $uifont -tearoff 0
      .bar add cascade -label "View" -menu .bar.view
      .bar.view add command -label "New view..." -command {newview 0}
      .bar.view add command -label "Edit view..." -command editview \
        -state disabled
      .bar.view add command -label "Delete view" -command delview -state disabled
 -    .bar.view add cascade -label "Highlight" -menu .bar.view.hl
      .bar.view add separator
      .bar.view add radiobutton -label "All files" -command {showview 0} \
        -variable selectedview -value 0
 -    .bar.view.hl add command -label "New view..." -command {newview 1}
 -    .bar.view.hl add command -label "Remove" -command delhighlight \
 -      -state disabled
 -    .bar.view.hl add separator
      
      menu .bar.help
      .bar add cascade -label "Help" -menu .bar.help
      }
      frame .ctop.top
      frame .ctop.top.bar
 +    frame .ctop.top.lbar
 +    pack .ctop.top.lbar -side bottom -fill x
      pack .ctop.top.bar -side bottom -fill x
      set cscroll .ctop.top.csb
      scrollbar $cscroll -command {allcanvs yview} -highlightthickness 0
      set findstring {}
      set fstring .ctop.top.bar.findstring
      lappend entries $fstring
 -    entry $fstring -width 30 -font $textfont -textvariable findstring -font $textfont
 +    entry $fstring -width 30 -font $textfont -textvariable findstring
 +    trace add variable findstring write find_change
      pack $fstring -side left -expand 1 -fill x
      set findtype Exact
      set findtypemenu [tk_optionMenu .ctop.top.bar.findtype \
                          findtype Exact IgnCase Regexp]
 +    trace add variable findtype write find_change
      .ctop.top.bar.findtype configure -font $uifont
      .ctop.top.bar.findtype.menu configure -font $uifont
      set findloc "All fields"
      tk_optionMenu .ctop.top.bar.findloc findloc "All fields" Headline \
 -      Comments Author Committer Files Pickaxe
 +      Comments Author Committer
 +    trace add variable findloc write find_change
      .ctop.top.bar.findloc configure -font $uifont
      .ctop.top.bar.findloc.menu configure -font $uifont
 -
      pack .ctop.top.bar.findloc -side right
      pack .ctop.top.bar.findtype -side right
 -    # for making sure type==Exact whenever loc==Pickaxe
 -    trace add variable findloc write findlocchange
 +
 +    label .ctop.top.lbar.flabel -text "Highlight:  Commits " \
 +      -font $uifont
 +    pack .ctop.top.lbar.flabel -side left -fill y
 +    set gdttype "touching paths:"
 +    set gm [tk_optionMenu .ctop.top.lbar.gdttype gdttype "touching paths:" \
 +              "adding/removing string:"]
 +    trace add variable gdttype write hfiles_change
 +    $gm conf -font $uifont
 +    .ctop.top.lbar.gdttype conf -font $uifont
 +    pack .ctop.top.lbar.gdttype -side left -fill y
 +    entry .ctop.top.lbar.fent -width 25 -font $textfont \
 +      -textvariable highlight_files
 +    trace add variable highlight_files write hfiles_change
 +    lappend entries .ctop.top.lbar.fent
 +    pack .ctop.top.lbar.fent -side left -fill x -expand 1
 +    label .ctop.top.lbar.vlabel -text " OR in view" -font $uifont
 +    pack .ctop.top.lbar.vlabel -side left -fill y
 +    global viewhlmenu selectedhlview
 +    set viewhlmenu [tk_optionMenu .ctop.top.lbar.vhl selectedhlview None]
 +    $viewhlmenu entryconf 0 -command delvhighlight
 +    $viewhlmenu conf -font $uifont
 +    .ctop.top.lbar.vhl conf -font $uifont
 +    pack .ctop.top.lbar.vhl -side left -fill y
 +    label .ctop.top.lbar.rlabel -text " OR " -font $uifont
 +    pack .ctop.top.lbar.rlabel -side left -fill y
 +    global highlight_related
 +    set m [tk_optionMenu .ctop.top.lbar.relm highlight_related None \
 +             "Descendent" "Not descendent" "Ancestor" "Not ancestor"]
 +    $m conf -font $uifont
 +    .ctop.top.lbar.relm conf -font $uifont
 +    trace add variable highlight_related write vrel_change
 +    pack .ctop.top.lbar.relm -side left -fill y
  
      panedwindow .ctop.cdet -orient horizontal
      .ctop add .ctop.cdet
      frame .ctop.cdet.left
 +    frame .ctop.cdet.left.bot
 +    pack .ctop.cdet.left.bot -side bottom -fill x
 +    button .ctop.cdet.left.bot.search -text "Search" -command dosearch \
 +      -font $uifont
 +    pack .ctop.cdet.left.bot.search -side left -padx 5
 +    set sstring .ctop.cdet.left.bot.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
      set ctext .ctop.cdet.left.ctext
      text $ctext -bg white -state disabled -font $textfont \
        -width $geometry(ctextw) -height $geometry(ctexth) \
 -      -yscrollcommand {.ctop.cdet.left.sb set} -wrap none
 +      -yscrollcommand scrolltext -wrap none
      scrollbar .ctop.cdet.left.sb -command "$ctext yview"
      pack .ctop.cdet.left.sb -side right -fill y
      pack $ctext -side left -fill both -expand 1
      .ctop.cdet add .ctop.cdet.left
  
+     $ctext tag conf comment -wrap $wrapcomment
      $ctext tag conf filesep -font [concat $textfont bold] -back "#aaaaaa"
      $ctext tag conf hunksep -fore blue
      $ctext tag conf d0 -fore red
      pack $cflist -side left -fill both -expand 1
      $cflist tag configure highlight \
        -background [$cflist cget -selectbackground]
 +    $cflist tag configure bold -font [concat $mainfont bold]
      .ctop.cdet add .ctop.cdet.right
      bind .ctop.cdet <Configure> {resizecdetpanes %W %w}
  
      bindkey <End> sellastline
      bind . <Key-Up> "selnextline -1"
      bind . <Key-Down> "selnextline 1"
 +    bind . <Shift-Key-Up> "next_highlight -1"
 +    bind . <Shift-Key-Down> "next_highlight 1"
      bindkey <Key-Right> "goforw"
      bindkey <Key-Left> "goback"
      bind . <Key-Prior> "selnextpage -1"
      bind . <Control-q> doquit
      bind . <Control-f> dofind
      bind . <Control-g> {findnext 0}
 -    bind . <Control-r> findprev
 +    bind . <Control-r> dosearchback
 +    bind . <Control-s> dosearch
      bind . <Control-equal> {incrfont 1}
      bind . <Control-KP_Add> {incrfont 1}
      bind . <Control-minus> {incrfont -1}
@@@ -711,7 -665,6 +712,7 @@@ proc canvscan {op w x y} 
  proc scrollcanv {cscroll f0 f1} {
      $cscroll set $f0 $f1
      drawfrac $f0 $f1
 +    flushhighlights
  }
  
  # when we make a key binding for the toplevel, make sure
@@@ -744,7 -697,7 +745,7 @@@ proc savestuff {w} 
      global stuffsaved findmergefiles maxgraphpct
      global maxwidth
      global viewname viewfiles viewargs viewperm nextviewnum
-     global cmitmode
+     global cmitmode wrapcomment
  
      if {$stuffsaved} return
      if {![winfo viewable .]} return
        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 "set geometry(width) [winfo width .ctop]"
        puts $f "set geometry(height) [winfo height .ctop]"
        puts $f "set geometry(canv1) [expr {[winfo width $canv]-2}]"
@@@ -894,8 -848,6 +896,8 @@@ Gitk key bindings
  <Ctrl-Down>   Scroll commit list down one line
  <Ctrl-PageUp> Scroll commit list up one page
  <Ctrl-PageDown>       Scroll commit list down one page
 +<Shift-Up>    Move to previous highlighted line
 +<Shift-Down>  Move to next highlighted line
  <Delete>, b   Scroll diff view up one page
  <Backspace>   Scroll diff view up one page
  <Space>               Scroll diff view down one page
@@@ -903,12 -855,11 +905,12 @@@ u               Scroll diff view up 18 line
  d             Scroll diff view down 18 lines
  <Ctrl-F>              Find
  <Ctrl-G>              Move to next find hit
 -<Ctrl-R>              Move to previous find hit
  <Return>      Move to next find hit
  /             Move to next find hit, or redo find
  ?             Move to previous find hit
  f             Scroll diff view to next file
 +<Ctrl-S>              Search for next hit in diff view
 +<Ctrl-R>              Search for previous hit in diff view
  <Ctrl-KP+>    Increase font size
  <Ctrl-plus>   Increase font size
  <Ctrl-KP->    Decrease font size
@@@ -975,7 -926,7 +977,7 @@@ proc treeview {w l openlevs} 
                $w insert end $str
                $w image create end -align center -image $bm -padx 1 \
                    -name a:$ix
 -              $w insert end $d
 +              $w insert end $d [highlight_tag $prefix]
                $w mark set s:$ix "end -1c"
                $w mark gravity s:$ix left
            }
                set str "\n"
                for {set i 0} {$i < $lev} {incr i} {append str "\t"}
                $w insert end $str
 -              $w insert end $tail
 +              $w insert end $tail [highlight_tag $f]
            }
            lappend treecontents($prefix) $tail
        }
@@@ -1024,22 -975,6 +1026,22 @@@ proc linetoelt {l} 
      }
  }
  
 +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
  
@@@ -1073,8 -1008,8 +1075,8 @@@ proc treeopendir {w dir} 
        incr treeheight($x) $n
      }
      foreach e $treecontents($dir) {
 +      set de $dir$e
        if {[string index $e end] eq "/"} {
 -          set de $dir$e
            set iy $treeindex($de)
            $w mark set d:$iy e:$ix
            $w mark gravity d:$iy left
            set treediropen($de) 0
            $w image create e:$ix -align center -image tri-rt -padx 1 \
                -name a:$iy
 -          $w insert e:$ix $e
 +          $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
 +          $w insert e:$ix $e [highlight_tag $de]
        }
      }
      $w mark gravity e:$ix left
@@@ -1184,56 -1119,20 +1186,56 @@@ proc init_flist {first} 
      set difffilestart {}
  }
  
 -proc add_flist {fl} {
 -    global flistmode cflist
 +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 {$flistmode eq "flat"} {
 -      foreach f $fl {
 -          $cflist insert end "\n$f"
 +    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 flistmode ctext difffilestart cflist cflist_top cmitmode
 +    global ctext difffilestart cflist cflist_top cmitmode
  
      if {$cmitmode eq "tree"} return
      if {![info exists cflist_top]} return
@@@ -1388,7 -1287,7 +1390,7 @@@ proc vieweditor {top n title} 
      checkbutton $top.perm -text "Remember this view" -variable newviewperm($n)
      grid $top.perm - -pady 5 -sticky w
      message $top.al -aspect 1000 -font $uifont \
-       -text "Commits to include (arguments to git-rev-list):"
+       -text "Commits to include (arguments to git rev-list):"
      grid $top.al - -sticky w -pady 5
      entry $top.args -width 50 -textvariable newviewargs($n) \
        -background white
      focus $top.t
  }
  
 -proc doviewmenu {m first cmd op args} {
 +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 $args
 +          eval $m $op $i $argv
            break
        }
      }
  }
  
  proc allviewmenus {n op args} {
 +    global viewhlmenu
 +
      doviewmenu .bar.view 7 [list showview $n] $op $args
 -    doviewmenu .bar.view.hl 3 [list addhighlight $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
 +    global viewargs newviewargs viewhlmenu
  
      if {[catch {
        set newargs [shellsplit $newviewargs($n)]
        if {!$newishighlight} {
            after idle showview $n
        } else {
 -          after idle addhighlight $n
 +          after idle addvhighlight $n
        }
      } else {
        # editing an existing view
        set viewperm($n) $newviewperm($n)
        if {$newviewname($n) ne $viewname($n)} {
            set viewname($n) $newviewname($n)
 -          allviewmenus $n entryconf -label $viewname($n)
 +          doviewmenu .bar.view 7 [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)} {
            set viewfiles($n) $files
  }
  
  proc delview {} {
 -    global curview viewdata viewperm
 +    global curview viewdata viewperm hlview selectedhlview
  
      if {$curview == 0} return
 +    if {[info exists hlview] && $hlview == $curview} {
 +      set selectedhlview None
 +      unset hlview
 +    }
      allviewmenus $curview delete
      set viewdata($curview) {}
      set viewperm($curview) 0
  }
  
  proc addviewmenu {n} {
 -    global viewname
 +    global viewname viewhlmenu
  
      .bar.view add radiobutton -label $viewname($n) \
        -command [list showview $n] -variable selectedview -value $n
 -    .bar.view.hl add radiobutton -label $viewname($n) \
 -      -command [list addhighlight $n] -variable selectedhlview -value $n
 +    $viewhlmenu add radiobutton -label $viewname($n) \
 +      -command [list addvhighlight $n] -variable selectedhlview
  }
  
  proc flatten {var} {
@@@ -1539,9 -1429,8 +1541,9 @@@ proc showview {n} 
      global pending_select phase
      global commitidx rowlaidout rowoptim linesegends
      global commfd nextupdate
 -    global selectedview hlview selectedhlview
 +    global selectedview
      global vparentlist vchildlist vdisporder vcmitlisted
 +    global hlview selectedhlview
  
      if {$n == $curview} return
      set selid {}
      catch {unset matchinglines}
      catch {unset treediffs}
      clear_display
 +    if {[info exists hlview] && $hlview == $n} {
 +      unset hlview
 +      set selectedhlview None
 +    }
  
      set curview $n
      set selectedview $n
 -    set selectedhlview -1
      .bar.view entryconf 2 -state [expr {$n == 0? "disabled": "normal"}]
      .bar.view entryconf 3 -state [expr {$n == 0? "disabled": "normal"}]
 -    catch {unset hlview}
 -    .bar.view.hl entryconf 1 -state disabled
  
      if {![info exists viewdata($n)]} {
        set pending_select $selid
      }
  }
  
 -proc addhighlight {n} {
 -    global hlview curview viewdata highlighted highlightedrows
 -    global selectedhlview
 +# Stuff relating to the highlighting facility
 +
 +proc ishighlighted {row} {
 +    global vhighlights fhighlights nhighlights rhighlights
 +
 +    if {[info exists nhighlights($row)] && $nhighlights($row) > 0} {
 +      return $nhighlights($row)
 +    }
 +    if {[info exists vhighlights($row)] && $vhighlights($row) > 0} {
 +      return $vhighlights($row)
 +    }
 +    if {[info exists fhighlights($row)] && $fhighlights($row) > 0} {
 +      return $fhighlights($row)
 +    }
 +    if {[info exists rhighlights($row)] && $rhighlights($row) > 0} {
 +      return $rhighlights($row)
 +    }
 +    return 0
 +}
 +
 +proc bolden {row font} {
 +    global canv linehtag selectedline boldrows
 +
 +    lappend boldrows $row
 +    $canv itemconf $linehtag($row) -font $font
 +    if {[info exists selectedline] && $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 {[info exists selectedline] && $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 mainfont boldrows
 +
 +    set stillbold {}
 +    foreach row $boldrows {
 +      if {![ishighlighted $row]} {
 +          bolden $row $mainfont
 +      } else {
 +          lappend stillbold $row
 +      }
 +    }
 +    set boldrows $stillbold
 +}
 +
 +proc addvhighlight {n} {
 +    global hlview curview viewdata vhl_done vhighlights commitidx
  
      if {[info exists hlview]} {
 -      delhighlight
 +      delvhighlight
      }
      set hlview $n
 -    set selectedhlview $n
 -    .bar.view.hl entryconf 1 -state normal
 -    set highlighted($n) 0
 -    set highlightedrows {}
      if {$n != $curview && ![info exists viewdata($n)]} {
        set viewdata($n) [list getcommits {{}} {{}} {} {} {} 0 0 0 {}]
        set vparentlist($n) {}
        set vdisporder($n) {}
        set vcmitlisted($n) {}
        start_rev_list $n
 -    } else {
 -      highlightmore
 +    }
 +    set vhl_done $commitidx($hlview)
 +    if {$vhl_done > 0} {
 +      drawvisible
      }
  }
  
 -proc delhighlight {} {
 -    global hlview highlightedrows canv linehtag mainfont
 -    global selectedhlview selectedline
 +proc delvhighlight {} {
 +    global hlview vhighlights
  
      if {![info exists hlview]} return
      unset hlview
 -    set selectedhlview {}
 -    .bar.view.hl entryconf 1 -state disabled
 -    foreach l $highlightedrows {
 -      $canv itemconf $linehtag($l) -font $mainfont
 -      if {$l == $selectedline} {
 -          $canv delete secsel
 -          set t [eval $canv create rect [$canv bbox $linehtag($l)] \
 -                     -outline {{}} -tags secsel \
 -                     -fill [$canv cget -selectbackground]]
 -          $canv lower $t
 -      }
 -    }
 +    catch {unset vhighlights}
 +    unbolden
  }
  
 -proc highlightmore {} {
 -    global hlview highlighted commitidx highlightedrows linehtag mainfont
 -    global displayorder vdisporder curview canv commitrow selectedline
 +proc vhighlightmore {} {
 +    global hlview vhl_done commitidx vhighlights
 +    global displayorder vdisporder curview mainfont
  
      set font [concat $mainfont bold]
      set max $commitidx($hlview)
      } else {
        set disp $vdisporder($hlview)
      }
 -    for {set i $highlighted($hlview)} {$i < $max} {incr i} {
 +    set vr [visiblerows]
 +    set r0 [lindex $vr 0]
 +    set r1 [lindex $vr 1]
 +    for {set i $vhl_done} {$i < $max} {incr i} {
        set id [lindex $disp $i]
        if {[info exists commitrow($curview,$id)]} {
            set row $commitrow($curview,$id)
 -          if {[info exists linehtag($row)]} {
 -              $canv itemconf $linehtag($row) -font $font
 -              lappend highlightedrows $row
 -              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
 +          if {$r0 <= $row && $row <= $r1} {
 +              if {![highlighted $row]} {
 +                  bolden $row $font
                }
 +              set vhighlights($row) 1
            }
        }
      }
 -    set highlighted($hlview) $max
 +    set vhl_done $max
 +}
 +
 +proc askvhighlight {row id} {
 +    global hlview vhighlights commitrow iddrawn mainfont
 +
 +    if {[info exists commitrow($hlview,$id)]} {
 +      if {[info exists iddrawn($id)] && ![ishighlighted $row]} {
 +          bolden $row [concat $mainfont bold]
 +      }
 +      set vhighlights($row) 1
 +    } else {
 +      set vhighlights($row) 0
 +    }
 +}
 +
 +proc hfiles_change {name ix op} {
 +    global highlight_files filehighlight fhighlights fh_serial
 +    global mainfont highlight_paths
 +
 +    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 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 "touching paths:"} {
 +      if {[catch {set paths [shellsplit $highlight_files]}]} return
 +      set highlight_paths [makepatterns $paths]
 +      highlight_filelist
 +      set gdtargs [concat -- $paths]
 +    } else {
 +      set gdtargs [list "-S$highlight_files"]
 +    }
 +    set cmd [concat | git-diff-tree -r -s --stdin $gdtargs]
 +    set filehighlight [open $cmd r+]
 +    fconfigure $filehighlight -blocking 0
 +    fileevent $filehighlight readable 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($row) -1
 +    puts $filehighlight $id
 +}
 +
 +proc readfhighlight {} {
 +    global filehighlight fhighlights commitrow curview mainfont iddrawn
 +    global fhl_list
 +
 +    while {[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]
 +          if {[info exists commitrow($curview,$id)]} {
 +              set fhighlights($commitrow($curview,$id)) 0
 +          }
 +      }
 +      set fhl_list [lrange $fhl_list [expr {$i+1}] end]
 +      if {$line eq {}} continue
 +      if {![info exists commitrow($curview,$line)]} continue
 +      set row $commitrow($curview,$line)
 +      if {[info exists iddrawn($line)] && ![ishighlighted $row]} {
 +          bolden $row [concat $mainfont bold]
 +      }
 +      set fhighlights($row) 1
 +    }
 +    if {[eof $filehighlight]} {
 +      # strange...
 +      puts "oops, git-diff-tree died"
 +      catch {close $filehighlight}
 +      unset filehighlight
 +    }
 +    next_hlcont
 +}
 +
 +proc find_change {name ix op} {
 +    global nhighlights mainfont boldnamerows
 +    global findstring findpattern findtype
 +
 +    # delete previous highlights, if any
 +    foreach row $boldnamerows {
 +      bolden_name $row $mainfont
 +    }
 +    set boldnamerows {}
 +    catch {unset nhighlights}
 +    unbolden
 +    if {$findtype ne "Regexp"} {
 +      set e [string map {"*" "\\*" "?" "\\?" "\[" "\\\[" "\\" "\\\\"} \
 +                 $findstring]
 +      set findpattern "*$e*"
 +    }
 +    drawvisible
 +}
 +
 +proc askfindhighlight {row id} {
 +    global nhighlights commitinfo iddrawn mainfont
 +    global findstring findtype findloc findpattern
 +
 +    if {![info exists commitinfo($id)]} {
 +      getcommit $id
 +    }
 +    set info $commitinfo($id)
 +    set isbold 0
 +    set fldtypes {Headline Author Date Committer CDate Comments}
 +    foreach f $info ty $fldtypes {
 +      if {$findloc ne "All fields" && $findloc ne $ty} {
 +          continue
 +      }
 +      if {$findtype eq "Regexp"} {
 +          set doesmatch [regexp $findstring $f]
 +      } elseif {$findtype eq "IgnCase"} {
 +          set doesmatch [string match -nocase $findpattern $f]
 +      } else {
 +          set doesmatch [string match $findpattern $f]
 +      }
 +      if {$doesmatch} {
 +          if {$ty eq "Author"} {
 +              set isbold 2
 +          } else {
 +              set isbold 1
 +          }
 +      }
 +    }
 +    if {[info exists iddrawn($id)]} {
 +      if {$isbold && ![ishighlighted $row]} {
 +          bolden $row [concat $mainfont bold]
 +      }
 +      if {$isbold >= 2} {
 +          bolden_name $row [concat $mainfont bold]
 +      }
 +    }
 +    set nhighlights($row) $isbold
 +}
 +
 +proc vrel_change {name ix op} {
 +    global highlight_related
 +
 +    rhighlight_none
 +    if {$highlight_related ne "None"} {
 +      after idle 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 rhighlights
 +
 +    catch {unset descendent}
 +    set desc_todo [list $a]
 +    catch {unset ancestor}
 +    set anc_todo [list $a]
 +    if {$highlight_related ne "None"} {
 +      rhighlight_none
 +      after idle drawvisible
 +    }
 +}
 +
 +proc rhighlight_none {} {
 +    global rhighlights
 +
 +    catch {unset rhighlights}
 +    unbolden
 +}
 +
 +proc is_descendent {a} {
 +    global curview children commitrow descendent desc_todo
 +
 +    set v $curview
 +    set la $commitrow($v,$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 {$commitrow($v,$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 parentlist commitrow ancestor anc_todo
 +
 +    set v $curview
 +    set la $commitrow($v,$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 {![info exists commitrow($v,$do)] || $commitrow($v,$do) > $la} {
 +          lappend leftover $do
 +          continue
 +      }
 +      foreach np [lindex $parentlist $commitrow($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 mainfont rhighlights
 +    global selectedline ancestor
 +
 +    if {![info exists selectedline]} return
 +    set isbold 0
 +    if {$highlight_related eq "Descendent" ||
 +      $highlight_related eq "Not descendent"} {
 +      if {![info exists descendent($id)]} {
 +          is_descendent $id
 +      }
 +      if {$descendent($id) == ($highlight_related eq "Descendent")} {
 +          set isbold 1
 +      }
 +    } elseif {$highlight_related eq "Ancestor" ||
 +            $highlight_related eq "Not ancestor"} {
 +      if {![info exists ancestor($id)]} {
 +          is_ancestor $id
 +      }
 +      if {$ancestor($id) == ($highlight_related eq "Ancestor")} {
 +          set isbold 1
 +      }
 +    }
 +    if {[info exists iddrawn($id)]} {
 +      if {$isbold && ![ishighlighted $row]} {
 +          bolden $row [concat $mainfont bold]
 +      }
 +    }
 +    set rhighlights($row) $isbold
 +}
 +
 +proc next_hlcont {} {
 +    global fhl_row fhl_dirn displayorder numcommits
 +    global vhighlights fhighlights nhighlights rhighlights
 +    global hlview filehighlight findstring highlight_related
 +
 +    if {![info exists fhl_dirn] || $fhl_dirn == 0} return
 +    set row $fhl_row
 +    while {1} {
 +      if {$row < 0 || $row >= $numcommits} {
 +          bell
 +          set fhl_dirn 0
 +          return
 +      }
 +      set id [lindex $displayorder $row]
 +      if {[info exists hlview]} {
 +          if {![info exists vhighlights($row)]} {
 +              askvhighlight $row $id
 +          }
 +          if {$vhighlights($row) > 0} break
 +      }
 +      if {$findstring ne {}} {
 +          if {![info exists nhighlights($row)]} {
 +              askfindhighlight $row $id
 +          }
 +          if {$nhighlights($row) > 0} break
 +      }
 +      if {$highlight_related ne "None"} {
 +          if {![info exists rhighlights($row)]} {
 +              askrelhighlight $row $id
 +          }
 +          if {$rhighlights($row) > 0} break
 +      }
 +      if {[info exists filehighlight]} {
 +          if {![info exists fhighlights($row)]} {
 +              # ask for a few more while we're at it...
 +              set r $row
 +              for {set n 0} {$n < 100} {incr n} {
 +                  if {![info exists fhighlights($r)]} {
 +                      askfilehighlight $r [lindex $displayorder $r]
 +                  }
 +                  incr r $fhl_dirn
 +                  if {$r < 0 || $r >= $numcommits} break
 +              }
 +              flushhighlights
 +          }
 +          if {$fhighlights($row) < 0} {
 +              set fhl_row $row
 +              return
 +          }
 +          if {$fhighlights($row) > 0} break
 +      }
 +      incr row $fhl_dirn
 +    }
 +    set fhl_dirn 0
 +    selectline $row 1
 +}
 +
 +proc next_highlight {dirn} {
 +    global selectedline fhl_row fhl_dirn
 +    global hlview filehighlight findstring highlight_related
 +
 +    if {![info exists selectedline]} return
 +    if {!([info exists hlview] || $findstring ne {} ||
 +        $highlight_related ne "None" || [info exists filehighlight])} return
 +    set fhl_row [expr {$selectedline + $dirn}]
 +    set fhl_dirn $dirn
 +    next_hlcont
 +}
 +
 +proc cancel_next_highlight {} {
 +    global fhl_dirn
 +
 +    set fhl_dirn 0
  }
  
  # Graph layout functions
@@@ -2870,7 -2336,8 +2872,7 @@@ proc drawcmittext {id row col rmx} 
      global commitlisted commitinfo rowidlist
      global rowtextx idpos idtags idheads idotherrefs
      global linehtag linentag linedtag
 -    global mainfont canvxmax
 -    global hlview commitrow highlightedrows
 +    global mainfont canvxmax boldrows boldnamerows
  
      set ofill [expr {[lindex $commitlisted $row]? "blue": "white"}]
      set x [xc $row $col]
      set date [lindex $commitinfo($id) 2]
      set date [formatdate $date]
      set font $mainfont
 -    if {[info exists hlview] && [info exists commitrow($hlview,$id)]} {
 +    set nfont $mainfont
 +    set isbold [ishighlighted $row]
 +    if {$isbold > 0} {
 +      lappend boldrows $row
        lappend font bold
 -      lappend highlightedrows $row
 +      if {$isbold > 1} {
 +          lappend boldnamerows $row
 +          lappend nfont bold
 +      }
      }
      set linehtag($row) [$canv create text $xt $y -anchor w \
                            -text $headline -font $font]
      $canv bind $linehtag($row) <Button-3> "rowmenu %X %Y $id"
      set linentag($row) [$canv2 create text 3 $y -anchor w \
 -                          -text $name -font $mainfont]
 +                          -text $name -font $nfont]
      set linedtag($row) [$canv3 create text 3 $y -anchor w \
                            -text $date -font $mainfont]
      set xr [expr {$xt + [font measure $mainfont $headline]}]
@@@ -2924,9 -2385,6 +2926,9 @@@ proc drawcmitrow {row} 
      global displayorder rowidlist
      global idrangedrawn iddrawn
      global commitinfo parentlist numcommits
 +    global filehighlight fhighlights findstring nhighlights
 +    global hlview vhighlights
 +    global highlight_related rhighlights
  
      if {$row >= $numcommits} return
      foreach id [lindex $rowidlist $row] {
      }
  
      set id [lindex $displayorder $row]
 +    if {[info exists hlview] && ![info exists vhighlights($row)]} {
 +      askvhighlight $row $id
 +    }
 +    if {[info exists filehighlight] && ![info exists fhighlights($row)]} {
 +      askfilehighlight $row $id
 +    }
 +    if {$findstring ne {} && ![info exists nhighlights($row)]} {
 +      askfindhighlight $row $id
 +    }
 +    if {$highlight_related ne "None" && ![info exists rhighlights($row)]} {
 +      askrelhighlight $row $id
 +    }
      if {[info exists iddrawn($id)]} return
      set col [lsearch -exact [lindex $rowidlist $row] $id]
      if {$col < 0} {
@@@ -3007,15 -2453,10 +3009,15 @@@ proc drawvisible {} 
  
  proc clear_display {} {
      global iddrawn idrangedrawn
 +    global vhighlights fhighlights nhighlights rhighlights
  
      allcanvs delete all
      catch {unset iddrawn}
      catch {unset idrangedrawn}
 +    catch {unset vhighlights}
 +    catch {unset fhighlights}
 +    catch {unset nhighlights}
 +    catch {unset rhighlights}
  }
  
  proc findcrossings {id} {
@@@ -3319,9 -2760,12 +3321,9 @@@ proc dofind {} 
  
      stopfindproc
      unmarkmatches
 +    cancel_next_highlight
      focus .
      set matchinglines {}
 -    if {$findloc == "Pickaxe"} {
 -      findpatches
 -      return
 -    }
      if {$findtype == "IgnCase"} {
        set foundstring [string tolower $findstring]
      } else {
      if {$foundstrlen == 0} return
      regsub -all {[*?\[\\]} $foundstring {\\&} matchstring
      set matchstring "*$matchstring*"
 -    if {$findloc == "Files"} {
 -      findfiles
 -      return
 -    }
      if {![info exists selectedline]} {
        set oldsel -1
      } else {
        set oldsel $selectedline
      }
      set didsel 0
 -    set fldtypes {Headline Author Date Committer CDate Comment}
 +    set fldtypes {Headline Author Date Committer CDate Comments}
      set l -1
      foreach id $displayorder {
        set d $commitdata($id)
@@@ -3440,6 -2888,18 +3442,6 @@@ proc findprev {} 
      }
  }
  
 -proc findlocchange {name ix op} {
 -    global findloc findtype findtypemenu
 -    if {$findloc == "Pickaxe"} {
 -      set findtype Exact
 -      set state disabled
 -    } else {
 -      set state normal
 -    }
 -    $findtypemenu entryconf 1 -state $state
 -    $findtypemenu entryconf 2 -state $state
 -}
 -
  proc stopfindproc {{done 0}} {
      global findprocpid findprocfile findids
      global ctext findoldcursor phase maincursor textcursor
      notbusy find
  }
  
 -proc findpatches {} {
 -    global findstring selectedline numcommits
 -    global findprocpid findprocfile
 -    global finddidsel ctext displayorder findinprogress
 -    global findinsertpos
 -
 -    if {$numcommits == 0} return
 -
 -    # make a list of all the ids to search, starting at the one
 -    # after the selected line (if any)
 -    if {[info exists selectedline]} {
 -      set l $selectedline
 -    } else {
 -      set l -1
 -    }
 -    set inputids {}
 -    for {set i 0} {$i < $numcommits} {incr i} {
 -      if {[incr l] >= $numcommits} {
 -          set l 0
 -      }
 -      append inputids [lindex $displayorder $l] "\n"
 -    }
 -
 -    if {[catch {
 -      set f [open [list | git diff-tree --stdin -s -r -S$findstring \
 -                       << $inputids] r]
 -    } err]} {
 -      error_popup "Error starting search process: $err"
 -      return
 -    }
 -
 -    set findinsertpos end
 -    set findprocfile $f
 -    set findprocpid [pid $f]
 -    fconfigure $f -blocking 0
 -    fileevent $f readable readfindproc
 -    set finddidsel 0
 -    nowbusy find
 -    set findinprogress 1
 -}
 -
 -proc readfindproc {} {
 -    global findprocfile finddidsel
 -    global commitrow matchinglines findinsertpos curview
 -
 -    set n [gets $findprocfile line]
 -    if {$n < 0} {
 -      if {[eof $findprocfile]} {
 -          stopfindproc 1
 -          if {!$finddidsel} {
 -              bell
 -          }
 -      }
 -      return
 -    }
 -    if {![regexp {^[0-9a-f]{40}} $line id]} {
 -      error_popup "Can't parse git diff-tree output: $line"
 -      stopfindproc
 -      return
 -    }
 -    if {![info exists commitrow($curview,$id)]} {
 -      puts stderr "spurious id: $id"
 -      return
 -    }
 -    set l $commitrow($curview,$id)
 -    insertmatch $l $id
 -}
 -
 -proc insertmatch {l id} {
 -    global matchinglines findinsertpos finddidsel
 -
 -    if {$findinsertpos == "end"} {
 -      if {$matchinglines != {} && $l < [lindex $matchinglines 0]} {
 -          set matchinglines [linsert $matchinglines 0 $l]
 -          set findinsertpos 1
 -      } else {
 -          lappend matchinglines $l
 -      }
 -    } else {
 -      set matchinglines [linsert $matchinglines $findinsertpos $l]
 -      incr findinsertpos
 -    }
 -    markheadline $l $id
 -    if {!$finddidsel} {
 -      findselectline $l
 -      set finddidsel 1
 -    }
 -}
 -
 -proc findfiles {} {
 -    global selectedline numcommits displayorder ctext
 -    global ffileline finddidsel parentlist
 -    global findinprogress findstartline findinsertpos
 -    global treediffs fdiffid fdiffsneeded fdiffpos
 -    global findmergefiles
 -
 -    if {$numcommits == 0} return
 -
 -    if {[info exists selectedline]} {
 -      set l [expr {$selectedline + 1}]
 -    } else {
 -      set l 0
 -    }
 -    set ffileline $l
 -    set findstartline $l
 -    set diffsneeded {}
 -    set fdiffsneeded {}
 -    while 1 {
 -      set id [lindex $displayorder $l]
 -      if {$findmergefiles || [llength [lindex $parentlist $l]] == 1} {
 -          if {![info exists treediffs($id)]} {
 -              append diffsneeded "$id\n"
 -              lappend fdiffsneeded $id
 -          }
 -      }
 -      if {[incr l] >= $numcommits} {
 -          set l 0
 -      }
 -      if {$l == $findstartline} break
 -    }
 -
 -    # start off a git diff-tree process if needed
 -    if {$diffsneeded ne {}} {
 -      if {[catch {
 -          set df [open [list | git diff-tree -r --stdin << $diffsneeded] r]
 -      } err ]} {
 -          error_popup "Error starting search process: $err"
 -          return
 -      }
 -      catch {unset fdiffid}
 -      set fdiffpos 0
 -      fconfigure $df -blocking 0
 -      fileevent $df readable [list readfilediffs $df]
 -    }
 -
 -    set finddidsel 0
 -    set findinsertpos end
 -    set id [lindex $displayorder $l]
 -    nowbusy find
 -    set findinprogress 1
 -    findcont
 -    update
 -}
 -
 -proc readfilediffs {df} {
 -    global findid fdiffid fdiffs
 -
 -    set n [gets $df line]
 -    if {$n < 0} {
 -      if {[eof $df]} {
 -          donefilediff
 -          if {[catch {close $df} err]} {
 -              stopfindproc
 -              bell
 -              error_popup "Error in git diff-tree: $err"
 -          } elseif {[info exists findid]} {
 -              set id $findid
 -              stopfindproc
 -              bell
 -              error_popup "Couldn't find diffs for $id"
 -          }
 -      }
 -      return
 -    }
 -    if {[regexp {^([0-9a-f]{40})$} $line match id]} {
 -      # start of a new string of diffs
 -      donefilediff
 -      set fdiffid $id
 -      set fdiffs {}
 -    } elseif {[string match ":*" $line]} {
 -      lappend fdiffs [lindex $line 5]
 -    }
 -}
 -
 -proc donefilediff {} {
 -    global fdiffid fdiffs treediffs findid
 -    global fdiffsneeded fdiffpos
 -
 -    if {[info exists fdiffid]} {
 -      while {[lindex $fdiffsneeded $fdiffpos] ne $fdiffid
 -             && $fdiffpos < [llength $fdiffsneeded]} {
 -          # git diff-tree doesn't output anything for a commit
 -          # which doesn't change anything
 -          set nullid [lindex $fdiffsneeded $fdiffpos]
 -          set treediffs($nullid) {}
 -          if {[info exists findid] && $nullid eq $findid} {
 -              unset findid
 -              findcont
 -          }
 -          incr fdiffpos
 -      }
 -      incr fdiffpos
 -
 -      if {![info exists treediffs($fdiffid)]} {
 -          set treediffs($fdiffid) $fdiffs
 -      }
 -      if {[info exists findid] && $fdiffid eq $findid} {
 -          unset findid
 -          findcont
 -      }
 -    }
 -}
 -
 -proc findcont {} {
 -    global findid treediffs parentlist
 -    global ffileline findstartline finddidsel
 -    global displayorder numcommits matchinglines findinprogress
 -    global findmergefiles
 -
 -    set l $ffileline
 -    while {1} {
 -      set id [lindex $displayorder $l]
 -      if {$findmergefiles || [llength [lindex $parentlist $l]] == 1} {
 -          if {![info exists treediffs($id)]} {
 -              set findid $id
 -              set ffileline $l
 -              return
 -          }
 -          set doesmatch 0
 -          foreach f $treediffs($id) {
 -              set x [findmatches $f]
 -              if {$x != {}} {
 -                  set doesmatch 1
 -                  break
 -              }
 -          }
 -          if {$doesmatch} {
 -              insertmatch $l $id
 -          }
 -      }
 -      if {[incr l] >= $numcommits} {
 -          set l 0
 -      }
 -      if {$l == $findstartline} break
 -    }
 -    stopfindproc
 -    if {!$finddidsel} {
 -      bell
 -    }
 -}
 -
  # mark a commit as matching by putting a yellow background
  # behind the headline
  proc markheadline {l id} {
@@@ -3526,11 -3227,11 +3528,11 @@@ proc commit_descriptor {p} 
  
  # append some text to the ctext widget, and make any SHA1 ID
  # that we know about be a clickable link.
- proc appendwithlinks {text} {
+ proc appendwithlinks {text tags} {
      global ctext commitrow linknum curview
  
      set start [$ctext index "end - 1c"]
-     $ctext insert end $text
+     $ctext insert end $text $tags
      $ctext insert end "\n"
      set links [regexp -indices -all -inline {[0-9a-f]{40}} $text]
      foreach l $links {
@@@ -3578,7 -3279,6 +3580,7 @@@ proc selectline {l isnew} 
      catch {unset pending_select}
      $canv delete hover
      normalline
 +    cancel_next_highlight
      if {$l < 0 || $l >= $numcommits} return
      set y [expr {$canvy0 + $l * $linespc}]
      set ymax [lindex [$canv cget -scrollregion] 3]
      $sha1entry insert 0 $id
      $sha1entry selection from 0
      $sha1entry selection to end
 +    rhighlight_sel $id
  
      $ctext conf -state normal
 -    $ctext delete 0.0 end
 +    clear_ctext
      set linknum 0
      set info $commitinfo($id)
      set date [formatdate [lindex $info 2]]
        $ctext insert end "\n"
      }
   
-     set comment {}
+     set headers {}
      set olds [lindex $parentlist $l]
      if {[llength $olds] > 1} {
        set np 0
                set tag m$np
            }
            $ctext insert end "Parent: " $tag
-           appendwithlinks [commit_descriptor $p]
+           appendwithlinks [commit_descriptor $p] {}
            incr np
        }
      } else {
        foreach p $olds {
-           append comment "Parent: [commit_descriptor $p]\n"
+           append headers "Parent: [commit_descriptor $p]\n"
        }
      }
  
      foreach c [lindex $childlist $l] {
-       append comment "Child:  [commit_descriptor $c]\n"
+       append headers "Child:  [commit_descriptor $c]\n"
      }
-     append comment "\n"
-     append comment [lindex $info 5]
  
      # make anything that looks like a SHA1 ID be a clickable link
-     appendwithlinks $comment
+     appendwithlinks $headers {}
+     appendwithlinks [lindex $info 5] {comment}
  
      $ctext tag delete Comments
      $ctext tag remove found 1.0 end
@@@ -3750,8 -3448,6 +3751,8 @@@ proc unselectline {} 
      catch {unset selectedline}
      catch {unset currentid}
      allcanvs delete secsel
 +    rhighlight_none
 +    cancel_next_highlight
  }
  
  proc reselectline {} {
@@@ -3830,7 -3526,7 +3831,7 @@@ proc gettree {id} 
      catch {unset diffmergeid}
      if {![info exists treefilelist($id)]} {
        if {![info exists treepending]} {
-           if {[catch {set gtf [open [concat | git-ls-tree -r $id] r]}]} {
+           if {[catch {set gtf [open [concat | git ls-tree -r $id] r]}]} {
                return
            }
            set treepending $id
@@@ -3878,14 -3574,14 +3879,14 @@@ proc showfile {f} 
        return
      }
      set blob [lindex $treeidlist($diffids) $i]
-     if {[catch {set bf [open [concat | git-cat-file blob $blob] r]} err]} {
+     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
      fileevent $bf readable [list getblobline $bf $diffids]
      $ctext config -state normal
 -    $ctext delete $commentend end
 +    clear_ctext $commentend
      $ctext insert end "\n"
      $ctext insert end "$f\n" filesep
      $ctext config -state disabled
@@@ -3920,7 -3616,7 +3921,7 @@@ proc mergediff {id l} 
      set diffids $id
      # this doesn't seem to actually affect anything...
      set env(GIT_DIFF_OPTS) $diffopts
-     set cmd [concat | git-diff-tree --no-commit-id --cc $id]
+     set cmd [concat | git diff-tree --no-commit-id --cc $id]
      if {[catch {set mdf [open $cmd r]} err]} {
        error_popup "Error getting merge diffs: $err"
        return
@@@ -4032,7 -3728,7 +4033,7 @@@ proc gettreediffs {ids} 
      set treepending $ids
      set treediff {}
      if {[catch \
-        {set gdtf [open [concat | git-diff-tree --no-commit-id -r $ids] r]} \
+        {set gdtf [open [concat | git diff-tree --no-commit-id -r $ids] r]} \
        ]} return
      fconfigure $gdtf -blocking 0
      fileevent $gdtf readable [list gettreediffline $gdtf $ids]
@@@ -4068,7 -3764,7 +4069,7 @@@ proc getblobdiffs {ids} 
      global nextupdate diffinhdr treediffs
  
      set env(GIT_DIFF_OPTS) $diffopts
-     set cmd [concat | git-diff-tree --no-commit-id -r -p -C $ids]
+     set cmd [concat | git diff-tree --no-commit-id -r -p -C $ids]
      if {[catch {set bdf [open $cmd r]} err]} {
        puts "error getting diffs: $err"
        return
@@@ -4185,138 -3881,6 +4186,138 @@@ proc nextfile {} 
      }
  }
  
 +proc clear_ctext {{first 1.0}} {
 +    global ctext smarktop smarkbot
 +
 +    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
 +}
 +
 +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
 +
 +    .ctop.cdet.left.sb set $f0 $f1
 +    if {$searchstring ne {}} {
 +      searchmarkvisible 0
 +    }
 +}
 +
  proc setcoords {} {
      global linespc charspc canvx0 canvy0 mainfont
      global xspc1 xspc2 lthickness
@@@ -4551,7 -4115,7 +4552,7 @@@ proc lineclick {x y id isnew} 
      }
      # fill the details pane with info about this line
      $ctext conf -state normal
 -    $ctext delete 0.0 end
 +    clear_ctext
      $ctext tag conf link -foreground blue -underline 1
      $ctext tag bind link <Enter> { %W configure -cursor hand2 }
      $ctext tag bind link <Leave> { %W configure -cursor $curtextcursor }
@@@ -4644,7 -4208,7 +4645,7 @@@ proc doseldiff {oldid newid} 
      global commitinfo
  
      $ctext conf -state normal
 -    $ctext delete 0.0 end
 +    clear_ctext
      init_flist "Top"
      $ctext insert end "From "
      $ctext tag conf link -foreground blue -underline 1
@@@ -4737,7 -4301,7 +4738,7 @@@ proc mkpatchgo {} 
      set oldid [$patchtop.fromsha1 get]
      set newid [$patchtop.tosha1 get]
      set fname [$patchtop.fname get]
-     if {[catch {exec git-diff-tree -p $oldid $newid >$fname &} err]} {
+     if {[catch {exec git diff-tree -p $oldid $newid >$fname &} err]} {
        error_popup "Error creating patch: $err"
      }
      catch {destroy $patchtop}
@@@ -4938,14 -4502,14 +4939,14 @@@ proc showtag {tag isnew} 
        addtohistory [list showtag $tag 0]
      }
      $ctext conf -state normal
 -    $ctext delete 0.0 end
 +    clear_ctext
      set linknum 0
      if {[info exists tagcontents($tag)]} {
        set text $tagcontents($tag)
      } else {
        set text "Tag: $tag\nId:  $tagids($tag)"
      }
-     appendwithlinks $text
+     appendwithlinks $text {}
      $ctext conf -state disabled
      init_flist {}
  }
@@@ -4957,7 -4521,7 +4958,7 @@@ proc doquit {} 
  }
  
  proc doprefs {} {
 -    global maxwidth maxgraphpct diffopts findmergefiles
 +    global maxwidth maxgraphpct diffopts
      global oldprefs prefstop
  
      set top .gitkprefs
        raise $top
        return
      }
 -    foreach v {maxwidth maxgraphpct diffopts findmergefiles} {
 +    foreach v {maxwidth maxgraphpct diffopts} {
        set oldprefs($v) [set $v]
      }
      toplevel $top
        -font optionfont
      spinbox $top.maxpct -from 1 -to 100 -width 4 -textvariable maxgraphpct
      grid x $top.maxpctl $top.maxpct -sticky w
 -    checkbutton $top.findm -variable findmergefiles
 -    label $top.findml -text "Include merges for \"Find\" in \"Files\"" \
 -      -font optionfont
 -    grid $top.findm $top.findml - -sticky w
      label $top.ddisp -text "Diff display options"
      grid $top.ddisp - -sticky w -pady 10
      label $top.diffoptl -text "Options for diff program" \
  }
  
  proc prefscan {} {
 -    global maxwidth maxgraphpct diffopts findmergefiles
 +    global maxwidth maxgraphpct diffopts
      global oldprefs prefstop
  
 -    foreach v {maxwidth maxgraphpct diffopts findmergefiles} {
 +    foreach v {maxwidth maxgraphpct diffopts} {
        set $v $oldprefs($v)
      }
      catch {destroy $prefstop}
@@@ -5300,11 -4868,11 +5301,11 @@@ proc tcl_encoding {enc} 
  # defaults...
  set datemode 0
  set diffopts "-U 5 -p"
- set wrcomcmd "git-diff-tree --stdin -p --pretty"
+ set wrcomcmd "git diff-tree --stdin -p --pretty"
  
  set gitencoding {}
  catch {
-     set gitencoding [exec git-repo-config --get i18n.commitencoding]
+     set gitencoding [exec git repo-config --get i18n.commitencoding]
  }
  if {$gitencoding == ""} {
      set gitencoding "utf-8"
@@@ -5325,7 -4893,9 +5326,8 @@@ set fastdate 
  set uparrowlen 7
  set downarrowlen 7
  set mingaplen 30
 -set flistmode "flat"
  set cmitmode "patch"
+ set wrapcomment "none"
  
  set colors {green red blue magenta darkgrey brown orange}
  
@@@ -5347,7 -4917,7 +5349,7 @@@ foreach arg $argv 
  # check that we can find a .git directory somewhere...
  set gitdir [gitdir]
  if {![file isdirectory $gitdir]} {
-     show_error . "Cannot find the git directory \"$gitdir\"."
+     show_error {} . "Cannot find the git directory \"$gitdir\"."
      exit 1
  }
  
@@@ -5358,7 -4928,7 +5360,7 @@@ if {$i >= 0} 
      set revtreeargs [lrange $revtreeargs 0 [expr {$i - 1}]]
  } elseif {$revtreeargs ne {}} {
      if {[catch {
-       set f [eval exec git-rev-parse --no-revs --no-flags $revtreeargs]
+       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]
        # so look for "fatal:".
        set i [string first "fatal:" $err]
        if {$i > 0} {
-           set err [string range [expr {$i + 6}] end]
+           set err [string range $err [expr {$i + 6}] end]
        }
-       show_error . "Bad arguments to gitk:\n$err"
+       show_error {} . "Bad arguments to gitk:\n$err"
        exit 1
      }
  }
  
  set history {}
  set historyindex 0
 +set fh_serial 0
 +set nhl_names {}
 +set highlight_paths {}
 +set searchdirn -forwards
 +set boldrows {}
 +set boldnamerows {}
  
  set optim_delay 16
  
  set nextviewnum 1
  set curview 0
  set selectedview 0
 -set selectedhlview {}
 +set selectedhlview None
  set viewfiles(0) {}
  set viewperm(0) 0
  set viewargs(0) {}