1559 lines
35 KiB

# -*- mode: TCL; fill-column: 75; tab-width: 8; coding: iso-latin-1-unix -*-
#
# $Id: ComboBox.tcl,v 1.9 2008/02/28 22:39:13 hobbs Exp $
#
# tixCombobox --
#
# A combobox widget is basically a listbox widget with an entry
# widget.
#
#
# Copyright (c) 1993-1999 Ioi Kim Lam.
# Copyright (c) 2000-2001 Tix Project Group.
# Copyright (c) 2004 ActiveState
#
# See the file "license.terms" for information on usage and redistribution
# of this file, and for a DISCLAIMER OF ALL WARRANTIES.
global tkPriv
if {![llength [info globals tkPriv]]} {
tk::unsupported::ExposePrivateVariable tkPriv
}
#--------------------------------------------------------------------------
# tkPriv elements used in this file:
#
# afterId - Token returned by "after" for autoscanning.
#--------------------------------------------------------------------------
#
foreach fun {tkCancelRepeat tkListboxUpDown tkButtonUp} {
if {![llength [info commands $fun]]} {
tk::unsupported::ExposePrivateCommand $fun
}
}
unset fun
tixWidgetClass tixComboBox {
-classname TixComboBox
-superclass tixLabelWidget
-method {
addhistory align appendhistory flash invoke insert pick popdown
}
-flag {
-anchor -arrowbitmap -browsecmd -command -crossbitmap
-disablecallback -disabledforeground -dropdown -editable
-fancy -grab -histlimit -historylimit -history -listcmd
-listwidth -prunehistory -selection -selectmode -state
-tickbitmap -validatecmd -value -variable
}
-static {
-dropdown -fancy
}
-forcecall {
-variable -selectmode -state
}
-configspec {
{-arrowbitmap arrowBitmap ArrowBitmap ""}
{-anchor anchor Anchor w}
{-browsecmd browseCmd BrowseCmd ""}
{-command command Command ""}
{-crossbitmap crossBitmap CrossBitmap ""}
{-disablecallback disableCallback DisableCallback 0 tixVerifyBoolean}
{-disabledforeground disabledForeground DisabledForeground #606060}
{-dropdown dropDown DropDown true tixVerifyBoolean}
{-editable editable Editable false tixVerifyBoolean}
{-fancy fancy Fancy false tixVerifyBoolean}
{-grab grab Grab global}
{-listcmd listCmd ListCmd ""}
{-listwidth listWidth ListWidth ""}
{-historylimit historyLimit HistoryLimit ""}
{-history history History false tixVerifyBoolean}
{-prunehistory pruneHistory PruneHistory true tixVerifyBoolean}
{-selectmode selectMode SelectMode browse}
{-selection selection Selection ""}
{-state state State normal}
{-validatecmd validateCmd ValidateCmd ""}
{-value value Value ""}
{-variable variable Variable ""}
{-tickbitmap tickBitmap TickBitmap ""}
}
-alias {
{-histlimit -historylimit}
}
-default {
{*Entry.relief sunken}
{*TixScrolledListBox.scrollbar auto}
{*Listbox.exportSelection false}
{*Listbox.takeFocus false}
{*shell.borderWidth 2}
{*shell.relief raised}
{*shell.cursor arrow}
{*Button.anchor c}
{*Button.borderWidth 1}
{*Button.highlightThickness 0}
{*Button.padX 0}
{*Button.padY 0}
{*tick.width 18}
{*tick.height 18}
{*cross.width 18}
{*cross.height 18}
{*arrow.anchor c}
{*arrow.width 15}
{*arrow.height 18}
}
}
# States: normal numbers: for dropdown style
# d+digit(s) : for non-dropdown style
#
proc tixComboBox:InitWidgetRec {w} {
upvar #0 $w data
tixChainMethod $w InitWidgetRec
set data(curIndex) ""
set data(varInited) 0
set data(state) none
set data(ignore) 0
if {$data(-history)} {
set data(-editable) 1
}
if {$data(-arrowbitmap) eq ""} {
set data(-arrowbitmap) [tix getbitmap cbxarrow]
}
if {$data(-crossbitmap) eq ""} {
set data(-crossbitmap) [tix getbitmap cross]
}
if {$data(-tickbitmap) eq ""} {
set data(-tickbitmap) [tix getbitmap tick]
}
}
proc tixComboBox:ConstructFramedWidget {w frame} {
upvar #0 $w data
tixChainMethod $w ConstructFramedWidget $frame
if {$data(-dropdown)} {
tixComboBox:ConstructEntryFrame $w $frame
tixComboBox:ConstructListShell $w
} else {
set f1 [frame $frame.f1]
set f2 [frame $frame.f2]
tixComboBox:ConstructEntryFrame $w $f1
tixComboBox:ConstructListFrame $w $f2
pack $f1 -side top -pady 2 -fill x
pack $f2 -side top -pady 2 -fill both -expand yes
}
}
proc tixComboBox:ConstructEntryFrame {w frame} {
upvar #0 $w data
# (1) The entry
#
set data(w:entry) [entry $frame.entry]
if {!$data(-editable)} {
set bg [$w cget -bg]
$data(w:entry) config -bg $bg -state disabled -takefocus 1
}
# This is used during "config-state"
#
set data(entryfg) [$data(w:entry) cget -fg]
# (2) The dropdown button, not necessary when not in dropdown mode
#
set data(w:arrow) [button $frame.arrow -bitmap $data(-arrowbitmap)]
if {!$data(-dropdown)} {
set xframe [frame $frame.xframe -width 19]
}
# (3) The fancy tick and cross buttons
#
if {$data(-fancy)} {
if {$data(-editable)} {
set data(w:cross) [button $frame.cross -bitmap $data(-crossbitmap)]
set data(w:tick) [button $frame.tick -bitmap $data(-tickbitmap)]
pack $frame.cross -side left -padx 1
pack $frame.tick -side left -padx 1
} else {
set data(w:tick) [button $frame.tick -bitmap $data(-tickbitmap)]
pack $frame.tick -side left -padx 1
}
}
if {$data(-dropdown)} {
pack $data(w:arrow) -side right -padx 1
foreach wid [list $data(w:frame) $data(w:label)] {
tixAddBindTag $wid TixComboWid
tixSetMegaWidget $wid $w TixComboBox
}
} else {
pack $xframe -side right -padx 1
}
pack $frame.entry -side right -fill x -expand yes -padx 1
}
proc tixComboBox:ConstructListShell {w} {
upvar #0 $w data
# Create the shell and the list
#------------------------------
set data(w:shell) [menu $w.shell -bd 2 -relief raised -tearoff 0]
wm overrideredirect $data(w:shell) 1
wm withdraw $data(w:shell)
set data(w:slistbox) [tixScrolledListBox $data(w:shell).slistbox \
-anchor $data(-anchor) -scrollbarspace y \
-options {listbox.selectMode "browse"}]
set data(w:listbox) [$data(w:slistbox) subwidget listbox]
pack $data(w:slistbox) -expand yes -fill both -padx 2 -pady 2
}
proc tixComboBox:ConstructListFrame {w frame} {
upvar #0 $w data
set data(w:slistbox) [tixScrolledListBox $frame.slistbox \
-anchor $data(-anchor)]
set data(w:listbox) [$data(w:slistbox) subwidget listbox]
pack $data(w:slistbox) -expand yes -fill both
}
proc tixComboBox:SetBindings {w} {
upvar #0 $w data
tixChainMethod $w SetBindings
# (1) Fix the bindings for the combobox
#
bindtags $w [list $w TixComboBox [winfo toplevel $w] all]
# (2) The entry subwidget
#
tixSetMegaWidget $data(w:entry) $w TixComboBox
bindtags $data(w:entry) [list $data(w:entry) Entry TixComboEntry\
TixComboWid [winfo toplevel $data(w:entry)] all]
# (3) The listbox and slistbox
#
$data(w:slistbox) config -browsecmd \
[list tixComboBox:LbBrowse $w]
$data(w:slistbox) config -command\
[list tixComboBox:LbCommand $w]
$data(w:listbox) config -takefocus 0
tixAddBindTag $data(w:listbox) TixComboLb
tixAddBindTag $data(w:slistbox) TixComboLb
tixSetMegaWidget $data(w:listbox) $w TixComboBox
tixSetMegaWidget $data(w:slistbox) $w TixComboBox
# (4) The buttons
#
if {$data(-dropdown)} {
$data(w:arrow) config -takefocus 0
tixAddBindTag $data(w:arrow) TixComboArrow
tixSetMegaWidget $data(w:arrow) $w TixComboBox
bind $data(w:root) <1> [list tixComboBox:RootDown $w]
bind $data(w:root) <ButtonRelease-1> [list tixComboBox:RootUp $w]
}
if {$data(-fancy)} {
if {$data(-editable)} {
$data(w:cross) config -command [list tixComboBox:CrossBtn $w] \
-takefocus 0
}
$data(w:tick) config -command [list tixComboBox:Invoke $w] -takefocus 0
}
if {$data(-dropdown)} {
set data(state) 0
} else {
set data(state) n0
}
}
proc tixComboBoxBind {} {
#----------------------------------------------------------------------
# The class bindings for the TixComboBox
#
tixBind TixComboBox <Escape> {
if {[tixComboBox:EscKey %W]} {
break
}
}
tixBind TixComboBox <Configure> {
tixWidgetDoWhenIdle tixComboBox:align %W
}
# Only the two "linear" detail_fields are for tabbing (moving) among
# widgets inside the same toplevel. Other detail_fields are sort
# of irrelevant
#
tixBind TixComboBox <FocusOut> {
if {[string equal %d NotifyNonlinear] ||
[string equal %d NotifyNonlinearVirtual]} {
if {[info exists %W(cancelTab)]} {
unset %W(cancelTab)
} else {
if {[set %W(-state)] ne "disabled"} {
if {[set %W(-selection)] ne [set %W(-value)]} {
tixComboBox:Invoke %W
}
}
}
}
}
tixBind TixComboBox <FocusIn> {
if {"%d" eq "NotifyNonlinear" || "%d" eq "NotifyNonlinearVirtual"} {
focus [%W subwidget entry]
# CYGNUS: Setting the selection if there is no data
# causes backspace to misbehave.
if {[[set %W(w:entry)] get] ne ""} {
[set %W(w:entry)] selection from 0
[set %W(w:entry)] selection to end
}
}
}
#----------------------------------------------------------------------
# The class tixBindings for the arrow button widget inside the TixComboBox
#
tixBind TixComboArrow <1> {
tixComboBox:ArrowDown [tixGetMegaWidget %W TixComboBox]
}
tixBind TixComboArrow <ButtonRelease-1> {
tixComboBox:ArrowUp [tixGetMegaWidget %W TixComboBox]
}
tixBind TixComboArrow <Escape> {
if {[tixComboBox:EscKey [tixGetMegaWidget %W TixComboBox]]} {
break
}
}
#----------------------------------------------------------------------
# The class tixBindings for the entry widget inside the TixComboBox
#
tixBind TixComboEntry <Up> {
tixComboBox:EntDirKey [tixGetMegaWidget %W TixComboBox] up
}
tixBind TixComboEntry <Down> {
tixComboBox:EntDirKey [tixGetMegaWidget %W TixComboBox] down
}
tixBind TixComboEntry <Prior> {
tixComboBox:EntDirKey [tixGetMegaWidget %W TixComboBox] pageup
}
tixBind TixComboEntry <Next> {
tixComboBox:EntDirKey [tixGetMegaWidget %W TixComboBox] pagedown
}
tixBind TixComboEntry <Return> {
tixComboBox:EntReturnKey [tixGetMegaWidget %W TixComboBox]
}
tixBind TixComboEntry <KeyPress> {
tixComboBox:EntKeyPress [tixGetMegaWidget %W TixComboBox]
}
tixBind TixComboEntry <Escape> {
if {[tixComboBox:EscKey [tixGetMegaWidget %W TixComboBox]]} {
break
}
}
tixBind TixComboEntry <Tab> {
if {[set [tixGetMegaWidget %W TixComboBox](-state)] ne "disabled"} {
if {[tixComboBox:EntTab [tixGetMegaWidget %W TixComboBox]]} {
break
}
}
}
tixBind TixComboEntry <1> {
if {[set [tixGetMegaWidget %W TixComboBox](-state)] ne "disabled"} {
focus %W
}
}
tixBind TixComboEntry <ButtonRelease-2> {
tixComboBox:EntKeyPress [tixGetMegaWidget %W TixComboBox]
}
#----------------------------------------------------------------------
# The class bindings for the listbox subwidget
#
tixBind TixComboWid <Escape> {
if {[tixComboBox:EscKey [tixGetMegaWidget %W TixComboBox]]} {
break
}
}
#----------------------------------------------------------------------
# The class bindings for some widgets inside ComboBox
#
tixBind TixComboWid <ButtonRelease-1> {
tixComboBox:WidUp [tixGetMegaWidget %W TixComboBox]
}
tixBind TixComboWid <Escape> {
if {[tixComboBox:EscKey [tixGetMegaWidget %W TixComboBox]]} {
break
}
}
}
#----------------------------------------------------------------------
# Cooked events
#----------------------------------------------------------------------
proc tixComboBox:ArrowDown {w} {
upvar #0 $w data
if {$data(-state) eq "disabled"} {
return
}
switch -exact -- $data(state) {
0 { tixComboBox:GoState 1 $w }
2 { tixComboBox:GoState 19 $w }
default { tixComboBox:StateError $w }
}
}
proc tixComboBox:ArrowUp {w} {
upvar #0 $w data
switch -exact -- $data(state) {
1 { tixComboBox:GoState 2 $w }
19 {
# data(ignore) was already set in state 19
tixComboBox:GoState 4 $w
}
5 { tixComboBox:GoState 13 $w }
default { tixComboBox:StateError $w }
}
}
proc tixComboBox:RootDown {w} {
upvar #0 $w data
switch -exact -- $data(state) {
0 {
# Ignore
}
2 { tixComboBox:GoState 3 $w }
default { tixComboBox:StateError $w }
}
}
proc tixComboBox:RootUp {w} {
upvar #0 $w data
switch -exact -- $data(state) {
{1} {
tixComboBox:GoState 12 $w
}
{3} {
# data(ignore) was already set in state 3
tixComboBox:GoState 4 $w
}
{5} {
tixComboBox:GoState 7 $w
}
default {
tixComboBox:StateError $w
}
}
}
proc tixComboBox:WidUp {w} {
upvar #0 $w data
switch -exact -- $data(state) {
{1} {
tixComboBox:GoState 12 $w
}
{5} {
tixComboBox:GoState 13 $w
}
}
}
proc tixComboBox:LbBrowse {w args} {
upvar #0 $w data
set event [tixEvent type]
set x [tixEvent flag x]
set y [tixEvent flag y]
set X [tixEvent flag X]
set Y [tixEvent flag Y]
if {$data(-state) eq "disabled"} { return }
switch -exact -- $event {
<1> {
case $data(state) {
{2} {
tixComboBox:GoState 5 $w $x $y $X $Y
}
{5} {
tixComboBox:GoState 5 $w $x $y $X $Y
}
{n0} {
tixComboBox:GoState n6 $w $x $y $X $Y
}
default {
tixComboBox:StateError $w
}
}
}
<ButtonRelease-1> {
case $data(state) {
{5} {
tixComboBox:GoState 6 $w $x $y $X $Y
}
{n6} {
tixComboBox:GoState n0 $w
}
default {
tixComboBox:StateError $w
}
}
}
default {
# Must be a motion event
case $data(state) {
{1} {
tixComboBox:GoState 9 $w $x $y $X $Y
}
{5} {
tixComboBox:GoState 5 $w $x $y $X $Y
}
{n6} {
tixComboBox:GoState n6 $w $x $y $X $Y
}
default {
tixComboBox:StateError $w
}
}
}
}
}
proc tixComboBox:LbCommand {w} {
upvar #0 $w data
if {$data(state) eq "n0"} {
tixComboBox:GoState n1 $w
}
}
#----------------------------------------------------------------------
# General keyboard event
# returns 1 if the combobox is in some special state and the Escape key
# shouldn't be handled by the toplevel bind tag. As a result, when a combobox
# is popped up in a dialog box, Escape will popdown the combo. If the combo
# is not popped up, Escape will invoke the toplevel bindtag (which can
# pop down the dialog box)
#
proc tixComboBox:EscKey {w} {
upvar #0 $w data
if {$data(-state) eq "disabled"} { return 0 }
switch -exact -- $data(state) {
{0} {
tixComboBox:GoState 17 $w
}
{2} {
tixComboBox:GoState 16 $w
return 1
}
{n0} {
tixComboBox:GoState n4 $w
}
default {
# ignore
return 1
}
}
return 0
}
#----------------------------------------
# Keyboard events
#----------------------------------------
proc tixComboBox:EntDirKey {w dir} {
upvar #0 $w data
if {$data(-state) eq "disabled"} { return }
switch -exact -- $data(state) {
{0} {
tixComboBox:GoState 10 $w $dir
}
{2} {
tixComboBox:GoState 11 $w $dir
}
{5} {
# ignore
}
{n0} {
tixComboBox:GoState n3 $w $dir
}
}
}
proc tixComboBox:EntReturnKey {w} {
upvar #0 $w data
if {$data(-state) eq "disabled"} { return }
switch -exact -- $data(state) {
{0} {
tixComboBox:GoState 14 $w
}
{2} {
tixComboBox:GoState 15 $w
}
{5} {
# ignore
}
{n0} {
tixComboBox:GoState n1 $w
}
}
}
# Return 1 == break from the binding == no keyboard focus traversal
proc tixComboBox:EntTab {w} {
upvar #0 $w data
switch -exact -- $data(state) {
{0} {
tixComboBox:GoState 14 $w
set data(cancelTab) ""
return 0
}
{2} {
tixComboBox:GoState 15 $w
set data(cancelTab) ""
return 0
}
{n0} {
tixComboBox:GoState n1 $w
set data(cancelTab) ""
return 0
}
default {
return 1
}
}
}
proc tixComboBox:EntKeyPress {w} {
upvar #0 $w data
if {$data(-state) eq "disabled" || !$data(-editable)} { return }
switch -exact -- $data(state) {
0 - 2 - n0 {
tixComboBox:ClearListboxSelection $w
tixComboBox:SetSelection $w [$data(w:entry) get] 0 0
}
}
}
#----------------------------------------------------------------------
proc tixComboBox:HandleDirKey {w dir} {
upvar #0 $w data
if {[tixComboBox:CheckListboxSelection $w]} {
switch -exact -- $dir {
"up" {
tkListboxUpDown $data(w:listbox) -1
set data(curIndex) [lindex [$data(w:listbox) curselection] 0]
tixComboBox:SetSelectionFromListbox $w
}
"down" {
tkListboxUpDown $data(w:listbox) 1
set data(curIndex) [lindex [$data(w:listbox) curselection] 0]
tixComboBox:SetSelectionFromListbox $w
}
"pageup" {
$data(w:listbox) yview scroll -1 pages
}
"pagedown" {
$data(w:listbox) yview scroll 1 pages
}
}
} else {
# There wasn't good selection in the listbox.
#
tixComboBox:SetSelectionFromListbox $w
}
}
proc tixComboBox:Invoke {w} {
upvar #0 $w data
tixComboBox:SetValue $w $data(-selection)
if {![winfo exists $w]} {
return
}
if {$data(-history)} {
tixComboBox:addhistory $w $data(-value)
set data(curIndex) 0
}
$data(w:entry) selection from 0
$data(w:entry) selection to end
$data(w:entry) icursor end
}
#----------------------------------------------------------------------
# MAINTAINING THE -VALUE
#----------------------------------------------------------------------
proc tixComboBox:SetValue {w newValue {noUpdate 0} {updateEnt 1}} {
upvar #0 $w data
if {[llength $data(-validatecmd)]} {
set data(-value) [tixEvalCmdBinding $w $data(-validatecmd) "" $newValue]
} else {
set data(-value) $newValue
}
if {! $noUpdate} {
tixVariable:UpdateVariable $w
}
if {$updateEnt} {
if {!$data(-editable)} {
$data(w:entry) delete 0 end
$data(w:entry) insert 0 $data(-value)
}
}
if {!$data(-disablecallback) && [llength $data(-command)]} {
if {![info exists data(varInited)]} {
set bind(specs) {%V}
set bind(%V) $data(-value)
tixEvalCmdBinding $w $data(-command) bind $data(-value)
if {![winfo exists $w]} {
# The user destroyed the window!
return
}
}
}
set data(-selection) $data(-value)
if {$updateEnt} {
tixSetEntry $data(w:entry) $data(-value)
if {$data(-anchor) eq "e"} {
tixComboBox:EntryAlignEnd $w
}
}
}
# markSel: should the all the text in the entry be highlighted?
#
proc tixComboBox:SetSelection {w value {markSel 1} {setent 1}} {
upvar #0 $w data
if {$setent} {
tixSetEntry $data(w:entry) $value
}
set data(-selection) $value
if {$data(-selectmode) eq "browse"} {
if {$markSel} {
$data(w:entry) selection range 0 end
}
if {[llength $data(-browsecmd)]} {
set bind(specs) {%V}
set bind(%V) [$data(w:entry) get]
tixEvalCmdBinding $w $data(-browsecmd) bind [$data(w:entry) get]
}
} else {
tixComboBox:SetValue $w $value 0 0
}
}
proc tixComboBox:ClearListboxSelection {w} {
upvar #0 $w data
if {![winfo exists $data(w:listbox)]} {
tixDebug "tixComboBox:ClearListboxSelection error non-existent $data(w:listbox)"
return
}
$data(w:listbox) selection clear 0 end
}
proc tixComboBox:UpdateListboxSelection {w index} {
upvar #0 $w data
if {![winfo exists $data(w:listbox)]} {
tixDebug "tixComboBox:UpdateListboxSelection error non-existent $data(w:listbox)"
return
}
if {$index != ""} {
$data(w:listbox) selection set $index
$data(w:listbox) selection anchor $index
}
}
proc tixComboBox:Cancel {w} {
upvar #0 $w data
tixSetEntry $data(w:entry) $data(-value)
tixComboBox:SetSelection $w $data(-value)
if {[tixComboBox:LbGetSelection $w] ne $data(-selection)} {
tixComboBox:ClearListboxSelection $w
}
}
proc tixComboBox:flash {w} {
tixComboBox:BlinkEntry $w
}
# Make the entry blink when the user selects a choice
#
proc tixComboBox:BlinkEntry {w} {
upvar #0 $w data
if {![info exists data(entryBlacken)]} {
set old_bg [$data(w:entry) cget -bg]
set old_fg [$data(w:entry) cget -fg]
$data(w:entry) config -fg $old_bg
$data(w:entry) config -bg $old_fg
set data(entryBlacken) 1
after 50 tixComboBox:RestoreBlink $w [list $old_bg] [list $old_fg]
}
}
proc tixComboBox:RestoreBlink {w old_bg old_fg} {
upvar #0 $w data
if {[info exists data(w:entry)] && [winfo exists $data(w:entry)]} {
$data(w:entry) config -fg $old_fg
$data(w:entry) config -bg $old_bg
}
if {[info exists data(entryBlacken)]} {
unset data(entryBlacken)
}
}
#----------------------------------------
# Handle events inside the list box
#----------------------------------------
proc tixComboBox:LbIndex {w {flag ""}} {
upvar #0 $w data
if {![winfo exists $data(w:listbox)]} {
tixDebug "tixComboBox:LbIndex error non-existent $data(w:listbox)"
if {$flag eq "emptyOK"} {
return ""
} else {
return 0
}
}
set sel [lindex [$data(w:listbox) curselection] 0]
if {$sel != ""} {
return $sel
} else {
if {$flag eq "emptyOK"} {
return ""
} else {
return 0
}
}
}
#----------------------------------------------------------------------
#
# STATE MANIPULATION
#
#----------------------------------------------------------------------
proc tixComboBox:GoState-0 {w} {
upvar #0 $w data
if {[info exists data(w:root)] && [grab current] eq "$data(w:root)"} {
grab release $w
}
}
proc tixComboBox:GoState-1 {w} {
upvar #0 $w data
tixComboBox:Popup $w
}
proc tixComboBox:GoState-2 {w} {
upvar #0 $w data
}
proc tixComboBox:GoState-3 {w} {
upvar #0 $w data
set data(ignore) 1
tixComboBox:Popdown $w
}
proc tixComboBox:GoState-4 {w} {
upvar #0 $w data
tixComboBox:Ungrab $w
if {$data(ignore)} {
tixComboBox:Cancel $w
} else {
tixComboBox:Invoke $w
}
tixComboBox:GoState 0 $w
}
proc tixComboBox:GoState-5 {w x y X Y} {
upvar #0 $w data
tixComboBox:LbSelect $w $x $y $X $Y
}
proc tixComboBox:GoState-6 {w x y X Y} {
upvar #0 $w data
tixComboBox:Popdown $w
if {[tixWithinWindow $data(w:shell) $X $Y]} {
set data(ignore) 0
} else {
set data(ignore) 1
}
tixComboBox:GoState 4 $w
}
proc tixComboBox:GoState-7 {w} {
upvar #0 $w data
tixComboBox:Popdown $w
set data(ignore) 1
catch {
global tkPriv
if {$tkPriv(afterId) != ""} {
tkCancelRepeat
}
}
set data(ignore) 1
tixComboBox:GoState 4 $w
}
proc tixComboBox:GoState-9 {w x y X Y} {
upvar #0 $w data
catch {
tkButtonUp $data(w:arrow)
}
tixComboBox:GoState 5 $w $x $y $X $Y
}
proc tixComboBox:GoState-10 {w dir} {
upvar #0 $w data
tixComboBox:Popup $w
if {![tixComboBox:CheckListboxSelection $w]} {
# There wasn't good selection in the listbox.
#
tixComboBox:SetSelectionFromListbox $w
}
tixComboBox:GoState 2 $w
}
proc tixComboBox:GoState-11 {w dir} {
upvar #0 $w data
tixComboBox:HandleDirKey $w $dir
tixComboBox:GoState 2 $w
}
proc tixComboBox:GoState-12 {w} {
upvar #0 $w data
catch {
tkButtonUp $data(w:arrow)
}
tixComboBox:GoState 2 $w
}
proc tixComboBox:GoState-13 {w} {
upvar #0 $w data
catch {
global tkPriv
if {$tkPriv(afterId) != ""} {
tkCancelRepeat
}
}
tixComboBox:GoState 2 $w
}
proc tixComboBox:GoState-14 {w} {
upvar #0 $w data
tixComboBox:Invoke $w
tixComboBox:GoState 0 $w
}
proc tixComboBox:GoState-15 {w} {
upvar #0 $w data
tixComboBox:Popdown $w
set data(ignore) 0
tixComboBox:GoState 4 $w
}
proc tixComboBox:GoState-16 {w} {
upvar #0 $w data
tixComboBox:Popdown $w
tixComboBox:Cancel $w
set data(ignore) 1
tixComboBox:GoState 4 $w
}
proc tixComboBox:GoState-17 {w} {
upvar #0 $w data
tixComboBox:Cancel $w
tixComboBox:GoState 0 $w
}
proc tixComboBox:GoState-19 {w} {
upvar #0 $w data
set data(ignore) [string equal $data(-selection) $data(-value)]
tixComboBox:Popdown $w
}
#----------------------------------------------------------------------
# Non-dropdown states
#----------------------------------------------------------------------
proc tixComboBox:GoState-n0 {w} {
upvar #0 $w data
}
proc tixComboBox:GoState-n1 {w} {
upvar #0 $w data
tixComboBox:Invoke $w
tixComboBox:GoState n0 $w
}
proc tixComboBox:GoState-n3 {w dir} {
upvar #0 $w data
tixComboBox:HandleDirKey $w $dir
tixComboBox:GoState n0 $w
}
proc tixComboBox:GoState-n4 {w} {
upvar #0 $w data
tixComboBox:Cancel $w
tixComboBox:GoState n0 $w
}
proc tixComboBox:GoState-n6 {w x y X Y} {
upvar #0 $w data
tixComboBox:LbSelect $w $x $y $X $Y
}
#----------------------------------------------------------------------
# General State Manipulation
#----------------------------------------------------------------------
proc tixComboBox:GoState {s w args} {
upvar #0 $w data
tixComboBox:SetState $w $s
eval tixComboBox:GoState-$s $w $args
}
proc tixComboBox:SetState {w s} {
upvar #0 $w data
# catch {puts [info level -2]}
# puts "setting state $data(state) --> $s"
set data(state) $s
}
proc tixComboBox:StateError {w} {
upvar #0 $w data
# error "wrong state $data(state)"
}
#----------------------------------------------------------------------
# Listbox handling
#----------------------------------------------------------------------
# Set a selection if there isn't one. Returns true if there was already
# a good selection inside the listbox
#
proc tixComboBox:CheckListboxSelection {w} {
upvar #0 $w data
if {![winfo exists $data(w:listbox)]} {
tixDebug "tixComboBox:CheckListboxSelection error non-existent $data(w:listbox)"
return 0
}
if {[$data(w:listbox) curselection] == ""} {
if {$data(curIndex) == ""} {
set data(curIndex) 0
}
$data(w:listbox) activate $data(curIndex)
$data(w:listbox) selection clear 0 end
$data(w:listbox) selection set $data(curIndex)
$data(w:listbox) see $data(curIndex)
return 0
} else {
return 1
}
}
proc tixComboBox:SetSelectionFromListbox {w} {
upvar #0 $w data
set string [$data(w:listbox) get $data(curIndex)]
tixComboBox:SetSelection $w $string
tixComboBox:UpdateListboxSelection $w $data(curIndex)
}
proc tixComboBox:LbGetSelection {w} {
upvar #0 $w data
set index [tixComboBox:LbIndex $w emptyOK]
if {$index >=0} {
return [$data(w:listbox) get $index]
} else {
return ""
}
}
proc tixComboBox:LbSelect {w x y X Y} {
upvar #0 $w data
set index [tixComboBox:LbIndex $w emptyOK]
if {$index == ""} {
set index [$data(w:listbox) nearest $y]
}
if {$index >= 0} {
if {[focus -lastfor $data(w:entry)] ne $data(w:entry) &&
[focus -lastfor $data(w:entry)] ne $data(w:listbox)} {
focus $data(w:entry)
}
set string [$data(w:listbox) get $index]
tixComboBox:SetSelection $w $string
tixComboBox:UpdateListboxSelection $w $index
}
}
#----------------------------------------------------------------------
# Internal commands
#----------------------------------------------------------------------
proc tixComboBox:CrossBtn {w} {
upvar #0 $w data
$data(w:entry) delete 0 end
tixComboBox:ClearListboxSelection $w
tixComboBox:SetSelection $w ""
}
#--------------------------------------------------
# Popping up list shell
#--------------------------------------------------
# Popup the listbox and grab
#
#
proc tixComboBox:Popup {w} {
global tcl_platform
upvar #0 $w data
if {![winfo ismapped $data(w:root)]} {
return
}
#---------------------------------------------------------------------
# Pop up
#
if {$data(-listcmd) != ""} {
# This option allows the user to fill in the listbox on demand
#
tixEvalCmdBinding $w $data(-listcmd)
}
# calculate the size
set y [winfo rooty $data(w:entry)]
incr y [winfo height $data(w:entry)]
incr y 3
set bd [$data(w:shell) cget -bd]
# incr bd [$data(w:shell) cget -highlightthickness]
set height [expr {[winfo reqheight $data(w:slistbox)] + 2*$bd}]
set x1 [winfo rootx $data(w:entry)]
if {$data(-listwidth) == ""} {
if {[winfo ismapped $data(w:arrow)]} {
set x2 [winfo rootx $data(w:arrow)]
if {$x2 >= $x1} {
incr x2 [winfo width $data(w:arrow)]
set width [expr {$x2 - $x1}]
} else {
set width [winfo width $data(w:entry)]
set x2 [expr {$x1 + $width}]
}
} else {
set width [winfo width $data(w:entry)]
set x2 [expr {$x1 + $width}]
}
} else {
set width $data(-listwidth)
set x2 [expr {$x1 + $width}]
}
set reqwidth [winfo reqwidth $data(w:shell)]
if {$reqwidth < $width} {
set reqwidth $width
} else {
if {$reqwidth > [expr {$width *3}]} {
set reqwidth [expr {$width *3}]
}
if {$reqwidth > [winfo vrootwidth .]} {
set reqwidth [winfo vrootwidth .]
}
}
set width $reqwidth
# If the listbox is too far right, pull it back to the left
#
set scrwidth [winfo vrootwidth .]
if {$x2 > $scrwidth} {
set x1 [expr {$scrwidth - $width}]
}
# If the listbox is too far left, pull it back to the right
#
if {$x1 < 0} {
set x1 0
}
# If the listbox is below bottom of screen, put it upwards
#
set scrheight [winfo vrootheight .]
set bottom [expr {$y+$height}]
if {$bottom > $scrheight} {
set y [expr {$y-$height-[winfo height $data(w:entry)]-5}]
}
# OK , popup the shell
#
global tcl_platform
wm geometry $data(w:shell) $reqwidth\x$height+$x1+$y
if {$tcl_platform(platform) eq "windows"} {
update
}
wm deiconify $data(w:shell)
if {$tcl_platform(platform) eq "windows"} {
update
}
raise $data(w:shell)
focus $data(w:entry)
set data(popped) 1
# add for safety
update
tixComboBox:Grab $w
}
proc tixComboBox:SetCursor {w cursor} {
upvar #0 $w data
$w config -cursor $cursor
}
proc tixComboBox:Popdown {w} {
upvar #0 $w data
wm withdraw $data(w:shell)
tixComboBox:SetCursor $w ""
}
# Grab the server so that user cannot move the windows around
proc tixComboBox:Grab {w} {
upvar #0 $w data
tixComboBox:SetCursor $w arrow
if {[catch {
# We catch here because grab may fail under a lot of circumstances
# Just don't want to break the code ...
switch -exact -- $data(-grab) {
global { tixPushGrab -global $data(w:root) }
local { tixPushGrab $data(w:root) }
}
} err]} {
tixDebug "tixComboBox:Grab+: Error grabbing $data(w:root)\n$err"
}
}
proc tixComboBox:Ungrab {w} {
upvar #0 $w data
if {[catch {
catch {
switch -exact -- $data(-grab) {
global { tixPopGrab }
local { tixPopGrab }
}
}
} err]} {
tixDebug "tixComboBox:Grab+: Error grabbing $data(w:root)\n$err"
}
}
#----------------------------------------------------------------------
# Alignment
#----------------------------------------------------------------------
# The following two routines can emulate a "right align mode" for the
# entry in the combo box.
proc tixComboBox:EntryAlignEnd {w} {
upvar #0 $w data
$data(w:entry) xview end
}
proc tixComboBox:Destructor {w} {
upvar #0 $w data
tixUnsetMegaWidget $data(w:entry)
tixVariable:DeleteVariable $w
# Chain this to the superclass
#
tixChainMethod $w Destructor
}
#----------------------------------------------------------------------
# CONFIG OPTIONS
#----------------------------------------------------------------------
proc tixComboBox:config-state {w value} {
upvar #0 $w data
catch {if {[$data(w:arrow) cget -state] eq $value} {set a 1}}
if {[info exists a]} {
return
}
catch {$data(w:arrow) config -state $value}
catch {$data(w:tick) config -state $value}
catch {$data(w:cross) config -state $value}
catch {$data(w:slistbox) config -state $value}
if {[string equal $value normal]} {
set fg [$data(w:arrow) cget -fg]
set entryFg $data(entryfg)
set lbSelFg [lindex [$data(w:listbox) config -selectforeground] 3]
set lbSelBg [lindex [$data(w:listbox) config -selectbackground] 3]
set entrySelFg [lindex [$data(w:entry) config -selectforeground] 3]
set entrySelBg [lindex [$data(w:entry) config -selectbackground] 3]
} else {
set fg [$data(w:arrow) cget -disabledforeground]
set entryFg $data(-disabledforeground)
set lbSelFg $entryFg
set lbSelBg [$data(w:listbox) cget -bg]
set entrySelFg $entryFg
set entrySelBg [$data(w:entry) cget -bg]
}
if {$fg ne ""} {
$data(w:label) config -fg $fg
$data(w:listbox) config -fg $fg -selectforeground $lbSelFg \
-selectbackground $lbSelBg
}
$data(w:entry) config -fg $entryFg -selectforeground $entrySelFg \
-selectbackground $entrySelBg
if {$value eq "normal"} {
if {$data(-editable)} {
$data(w:entry) config -state normal
}
$data(w:entry) config -takefocus 1
} else {
if {$data(-editable)} {
$data(w:entry) config -state disabled
}
$data(w:entry) config -takefocus 0
}
}
proc tixComboBox:config-value {w value} {
upvar #0 $w data
tixComboBox:SetValue $w $value
set data(-selection) $value
if {[tixComboBox:LbGetSelection $w] ne $value} {
tixComboBox:ClearListboxSelection $w
}
}
proc tixComboBox:config-selection {w value} {
upvar #0 $w data
tixComboBox:SetSelection $w $value
if {[tixComboBox:LbGetSelection $w] ne $value} {
tixComboBox:ClearListboxSelection $w
}
}
proc tixComboBox:config-variable {w arg} {
upvar #0 $w data
if {[tixVariable:ConfigVariable $w $arg]} {
# The value of data(-value) is changed if tixVariable:ConfigVariable
# returns true
set data(-selection) $data(-value)
tixComboBox:SetValue $w $data(-value) 1
}
catch {
unset data(varInited)
}
set data(-variable) $arg
}
#----------------------------------------------------------------------
# WIDGET COMMANDS
#----------------------------------------------------------------------
proc tixComboBox:align {w args} {
upvar #0 $w data
if {$data(-anchor) eq "e"} {
tixComboBox:EntryAlignEnd $w
}
}
proc tixComboBox:addhistory {w value} {
upvar #0 $w data
tixComboBox:insert $w 0 $value
$data(w:listbox) selection clear 0 end
if {$data(-prunehistory)} {
# Prune from the end
#
set max [$data(w:listbox) size]
if {$max <= 1} {
return
}
for {set i [expr {$max -1}]} {$i >= 1} {incr i -1} {
if {[$data(w:listbox) get $i] eq $value} {
$data(w:listbox) delete $i
break
}
}
}
}
proc tixComboBox:appendhistory {w value} {
upvar #0 $w data
tixComboBox:insert $w end $value
$data(w:listbox) selection clear 0 end
if {$data(-prunehistory)} {
# Prune from the end
#
set max [$data(w:listbox) size]
if {$max <= 1} {
return
}
for {set i [expr {$max -2}]} {$i >= 0} {incr i -1} {
if {[$data(w:listbox) get $i] eq $value} {
$data(w:listbox) delete $i
break
}
}
}
}
proc tixComboBox:insert {w index newitem} {
upvar #0 $w data
$data(w:listbox) insert $index $newitem
if {$data(-history) && $data(-historylimit) != ""
&& [$data(w:listbox) size] eq $data(-historylimit)} {
$data(w:listbox) delete 0
}
}
proc tixComboBox:pick {w index} {
upvar #0 $w data
$data(w:listbox) activate $index
$data(w:listbox) selection clear 0 end
$data(w:listbox) selection set active
$data(w:listbox) see active
set text [$data(w:listbox) get $index]
tixComboBox:SetValue $w $text
set data(curIndex) $index
}
proc tixComboBox:invoke {w} {
tixComboBox:Invoke $w
}
proc tixComboBox:popdown {w} {
upvar #0 $w data
if {$data(-dropdown)} {
tixComboBox:Popdown $w
}
}