c80d7be5e1
This patch enables the use of themed Tk widgets with Tk 8.5 and above. These make a significant difference on Windows in making the application appear native. On Windows and MacOSX ttk defaults to the native look as much as possible. On X11 the user may select a theme using the TkTheme XRDB resource class by adding an line to the .Xresources file. The set of installed theme names is available using the Tk command 'ttk::themes'. The default on X11 is similar to the current un-themed style - a kind of thin bordered motif look. A new git config variable 'gui.usettk' may be set to disable this if the user prefers the classic Tk look. Using Tk 8.4 will also avoid the use of themed widgets as these are only available since 8.5. Some support is included for Tk 8.6 features (themed spinbox and native font chooser for MacOSX and Windows). Signed-off-by: Pat Thoyts <patthoyts@users.sourceforge.net> Signed-off-by: Shawn O. Pearce <spearce@spearce.org>
199 lines
4.2 KiB
Tcl
199 lines
4.2 KiB
Tcl
# incremental search panel
|
|
# based on code from gitk, Copyright (C) Paul Mackerras
|
|
|
|
class searchbar {
|
|
|
|
field w
|
|
field ctext
|
|
|
|
field searchstring {}
|
|
field casesensitive 1
|
|
field searchdirn -forwards
|
|
|
|
field smarktop
|
|
field smarkbot
|
|
|
|
constructor new {i_w i_text args} {
|
|
global use_ttk NS
|
|
set w $i_w
|
|
set ctext $i_text
|
|
|
|
${NS}::frame $w
|
|
${NS}::label $w.l -text [mc Find:]
|
|
entry $w.ent -textvariable ${__this}::searchstring -background lightgreen
|
|
${NS}::button $w.bn -text [mc Next] -command [cb find_next]
|
|
${NS}::button $w.bp -text [mc Prev] -command [cb find_prev]
|
|
${NS}::checkbutton $w.cs -text [mc Case-Sensitive] \
|
|
-variable ${__this}::casesensitive -command [cb _incrsearch]
|
|
pack $w.l -side left
|
|
pack $w.cs -side right
|
|
pack $w.bp -side right
|
|
pack $w.bn -side right
|
|
pack $w.ent -side left -expand 1 -fill x
|
|
|
|
eval grid conf $w -sticky we $args
|
|
grid remove $w
|
|
|
|
trace add variable searchstring write [cb _incrsearch_cb]
|
|
|
|
bind $w <Destroy> [list delete_this $this]
|
|
return $this
|
|
}
|
|
|
|
method show {} {
|
|
if {![visible $this]} {
|
|
grid $w
|
|
}
|
|
focus -force $w.ent
|
|
}
|
|
|
|
method hide {} {
|
|
if {[visible $this]} {
|
|
focus $ctext
|
|
grid remove $w
|
|
}
|
|
}
|
|
|
|
method visible {} {
|
|
return [winfo ismapped $w]
|
|
}
|
|
|
|
method editor {} {
|
|
return $w.ent
|
|
}
|
|
|
|
method _get_new_anchor {} {
|
|
# use start of selection if it is visible,
|
|
# or the bounds of the visible area
|
|
set top [$ctext index @0,0]
|
|
set bottom [$ctext index @0,[winfo height $ctext]]
|
|
set sel [$ctext tag ranges sel]
|
|
if {$sel ne {}} {
|
|
set spos [lindex $sel 0]
|
|
if {[lindex $spos 0] >= [lindex $top 0] &&
|
|
[lindex $spos 0] <= [lindex $bottom 0]} {
|
|
return $spos
|
|
}
|
|
}
|
|
if {$searchdirn eq "-forwards"} {
|
|
return $top
|
|
} else {
|
|
return $bottom
|
|
}
|
|
}
|
|
|
|
method _get_wrap_anchor {dir} {
|
|
if {$dir eq "-forwards"} {
|
|
return 1.0
|
|
} else {
|
|
return end
|
|
}
|
|
}
|
|
|
|
method _do_search {start {mlenvar {}} {dir {}} {endbound {}}} {
|
|
set cmd [list $ctext search]
|
|
if {$mlenvar ne {}} {
|
|
upvar $mlenvar mlen
|
|
lappend cmd -count mlen
|
|
}
|
|
if {!$casesensitive} {
|
|
lappend cmd -nocase
|
|
}
|
|
if {$dir eq {}} {
|
|
set dir $searchdirn
|
|
}
|
|
lappend cmd $dir -- $searchstring
|
|
if {$endbound ne {}} {
|
|
set here [eval $cmd [list $start] [list $endbound]]
|
|
} else {
|
|
set here [eval $cmd [list $start]]
|
|
if {$here eq {}} {
|
|
set here [eval $cmd [_get_wrap_anchor $this $dir]]
|
|
}
|
|
}
|
|
return $here
|
|
}
|
|
|
|
method _incrsearch_cb {name ix op} {
|
|
after idle [cb _incrsearch]
|
|
}
|
|
|
|
method _incrsearch {} {
|
|
$ctext tag remove found 1.0 end
|
|
if {[catch {$ctext index anchor}]} {
|
|
$ctext mark set anchor [_get_new_anchor $this]
|
|
}
|
|
if {$searchstring ne {}} {
|
|
set here [_do_search $this anchor mlen]
|
|
if {$here ne {}} {
|
|
$ctext see $here
|
|
$ctext tag remove sel 1.0 end
|
|
$ctext tag add sel $here "$here + $mlen c"
|
|
$w.ent configure -background lightgreen
|
|
_set_marks $this 1
|
|
} else {
|
|
$w.ent configure -background lightpink
|
|
}
|
|
}
|
|
}
|
|
|
|
method find_prev {} {
|
|
find_next $this -backwards
|
|
}
|
|
|
|
method find_next {{dir -forwards}} {
|
|
focus $w.ent
|
|
$w.ent icursor end
|
|
set searchdirn $dir
|
|
$ctext mark unset anchor
|
|
if {$searchstring ne {}} {
|
|
set start [_get_new_anchor $this]
|
|
if {$dir eq "-forwards"} {
|
|
set start "$start + 1c"
|
|
}
|
|
set match [_do_search $this $start mlen]
|
|
$ctext tag remove sel 1.0 end
|
|
if {$match ne {}} {
|
|
$ctext see $match
|
|
$ctext tag add sel $match "$match + $mlen c"
|
|
}
|
|
}
|
|
}
|
|
|
|
method _mark_range {first last} {
|
|
set mend $first.0
|
|
while {1} {
|
|
set match [_do_search $this $mend mlen -forwards $last.end]
|
|
if {$match eq {}} break
|
|
set mend "$match + $mlen c"
|
|
$ctext tag add found $match $mend
|
|
}
|
|
}
|
|
|
|
method _set_marks {doall} {
|
|
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
|
|
_mark_range $this $topline $botline
|
|
set smarktop $topline
|
|
set smarkbot $botline
|
|
} else {
|
|
if {$topline < $smarktop} {
|
|
_mark_range $this $topline [expr {$smarktop-1}]
|
|
set smarktop $topline
|
|
}
|
|
if {$botline > $smarkbot} {
|
|
_mark_range $this [expr {$smarkbot+1}] $botline
|
|
set smarkbot $botline
|
|
}
|
|
}
|
|
}
|
|
|
|
method scrolled {} {
|
|
if {$searchstring ne {}} {
|
|
after idle [cb _set_marks 0]
|
|
}
|
|
}
|
|
|
|
} |