443 lines
10 KiB
443 lines
10 KiB
# -*- mode: TCL; fill-column: 75; tab-width: 8; coding: iso-latin-1-unix -*-
|
|
#
|
|
# $Id: Utils.tcl,v 1.4 2004/03/28 02:44:57 hobbs Exp $
|
|
#
|
|
# Util.tcl --
|
|
#
|
|
# The Tix utility commands. Some of these commands are
|
|
# replacement of or extensions to the existing TK
|
|
# commands. Occasionaly, you have to use the commands inside
|
|
# this file instead of thestandard TK commands to make your
|
|
# applicatiion work better with Tix. Please read the
|
|
# documentations (programmer's guide, man pages) for information
|
|
# about these utility commands.
|
|
#
|
|
# 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.
|
|
#
|
|
|
|
|
|
#
|
|
# kludge: should be able to handle all kinds of flags
|
|
# now only handles "-flag value" pairs.
|
|
#
|
|
proc tixHandleArgv {p_argv p_options validFlags} {
|
|
upvar $p_options opt
|
|
upvar $p_argv argv
|
|
|
|
set old_argv $argv
|
|
set argv ""
|
|
|
|
foreac {flag value} $old_argv {
|
|
if {[lsearch $validFlags $flag] != -1} {
|
|
# The caller will handle this option exclusively
|
|
# It won't be added back to the original arglist
|
|
#
|
|
eval $opt($flag,action) $value
|
|
} else {
|
|
# The caller does not handle this option
|
|
#
|
|
lappend argv $flag
|
|
lappend argv $value
|
|
}
|
|
}
|
|
}
|
|
|
|
#-----------------------------------------------------------------------
|
|
# tixDisableAll -
|
|
#
|
|
# Disable all members in a sub widget tree
|
|
#
|
|
proc tixDisableAll {w} {
|
|
foreach x [tixDescendants $w] {
|
|
catch {$x config -state disabled}
|
|
}
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# tixEnableAll -
|
|
#
|
|
# enable all members in a sub widget tree
|
|
#
|
|
proc tixEnableAll {w} {
|
|
foreach x [tixDescendants $w] {
|
|
catch {$x config -state normal}
|
|
}
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# tixDescendants -
|
|
#
|
|
# Return a list of all the member of a widget subtree, including
|
|
# the tree's root widget.
|
|
#
|
|
proc tixDescendants {parent} {
|
|
set des ""
|
|
lappend des $parent
|
|
|
|
foreach w [winfo children $parent] {
|
|
foreach x [tixDescendants $w] {
|
|
lappend des $x
|
|
}
|
|
}
|
|
return $des
|
|
}
|
|
|
|
#----------------------------------------------------------------------
|
|
# tixTopLevel -
|
|
#
|
|
# Create a toplevel widget and unmap it immediately. This will ensure
|
|
# that this toplevel widgets will not be popped up prematurely when you
|
|
# create Tix widgets inside it.
|
|
#
|
|
# "tixTopLevel" also provide options for you to specify the appearance
|
|
# and behavior of this toplevel.
|
|
#
|
|
#
|
|
#
|
|
proc tixTopLevel {w args} {
|
|
set opt (-geometry) ""
|
|
set opt (-minsize) ""
|
|
set opt (-maxsize) ""
|
|
set opt (-width) ""
|
|
set opt (-height) ""
|
|
|
|
eval [linsert $args 0 toplevel $w]
|
|
wm withdraw $w
|
|
}
|
|
|
|
# This is a big kludge
|
|
#
|
|
# Substitutes all [...] and $.. in the string in $args
|
|
#
|
|
proc tixInt_Expand {args} {
|
|
return $args
|
|
}
|
|
|
|
# Print out all the config options of a widget
|
|
#
|
|
proc tixPConfig {w} {
|
|
puts [join [lsort [$w config]] \n]
|
|
}
|
|
|
|
proc tixAppendBindTag {w tag} {
|
|
bindtags $w [concat [bindtags $w] $tag]
|
|
}
|
|
|
|
proc tixAddBindTag {w tag} {
|
|
bindtags $w [concat $tag [bindtags $w] ]
|
|
}
|
|
|
|
proc tixSubwidgetRef {sub} {
|
|
return $::tixSRef($sub)
|
|
}
|
|
|
|
proc tixSubwidgetRetCreate {sub ref} {
|
|
set ::tixSRef($sub) $ref
|
|
}
|
|
|
|
proc tixSubwidgetRetDelete {sub} {
|
|
catch {unset ::tixSRef($sub)}
|
|
}
|
|
|
|
proc tixListboxGetCurrent {listbox} {
|
|
return [tixEvent flag V]
|
|
}
|
|
|
|
|
|
# tixSetMegaWidget --
|
|
#
|
|
# Associate a subwidget with its mega widget "owner". This is mainly
|
|
# used when we add a new bindtag to a subwidget and we need to find out
|
|
# the name of the mega widget inside the binding.
|
|
#
|
|
proc tixSetMegaWidget {w mega {type any}} {
|
|
set ::tixMega($type,$w) $mega
|
|
}
|
|
|
|
proc tixGetMegaWidget {w {type any}} {
|
|
return $::tixMega($type,$w)
|
|
}
|
|
|
|
proc tixUnsetMegaWidget {w} {
|
|
if {[info exists ::tixMega($w)]} { unset ::tixMega($w) }
|
|
}
|
|
|
|
# tixBusy : display busy cursors on a window
|
|
#
|
|
#
|
|
# Should flush the event queue (but not do any idle tasks) before blocking
|
|
# the target window (I am not sure if it is aready doing so )
|
|
#
|
|
# ToDo: should take some additional windows to raise
|
|
#
|
|
proc tixBusy {w flag {focuswin ""}} {
|
|
|
|
if {[info command tixInputOnly] == ""} {
|
|
return
|
|
}
|
|
|
|
global tixBusy
|
|
set toplevel [winfo toplevel $w]
|
|
|
|
if {![info exists tixBusy(cursor)]} {
|
|
set tixBusy(cursor) watch
|
|
# set tixBusy(cursor) "[tix getbitmap hourglass] \
|
|
# [string range [tix getbitmap hourglass.mask] 1 end]\
|
|
# black white"
|
|
}
|
|
|
|
if {$toplevel eq "."} {
|
|
set inputonly0 .__tix__busy0
|
|
set inputonly1 .__tix__busy1
|
|
set inputonly2 .__tix__busy2
|
|
set inputonly3 .__tix__busy3
|
|
} else {
|
|
set inputonly0 $toplevel.__tix__busy0
|
|
set inputonly1 $toplevel.__tix__busy1
|
|
set inputonly2 $toplevel.__tix__busy2
|
|
set inputonly3 $toplevel.__tix__busy3
|
|
}
|
|
|
|
if {![winfo exists $inputonly0]} {
|
|
for {set i 0} {$i < 4} {incr i} {
|
|
tixInputOnly [set inputonly$i] -cursor $tixBusy(cursor)
|
|
}
|
|
}
|
|
|
|
if {$flag eq "on"} {
|
|
if {$focuswin != "" && [winfo id $focuswin] != 0} {
|
|
if {[info exists tixBusy($focuswin,oldcursor)]} {
|
|
return
|
|
}
|
|
set tixBusy($focuswin,oldcursor) [$focuswin cget -cursor]
|
|
$focuswin config -cursor $tixBusy(cursor)
|
|
|
|
set x1 [expr {[winfo rootx $focuswin]-[winfo rootx $toplevel]}]
|
|
set y1 [expr {[winfo rooty $focuswin]-[winfo rooty $toplevel]}]
|
|
|
|
set W [winfo width $focuswin]
|
|
set H [winfo height $focuswin]
|
|
set x2 [expr {$x1 + $W}]
|
|
set y2 [expr {$y1 + $H}]
|
|
|
|
|
|
if {$y1 > 0} {
|
|
tixMoveResizeWindow $inputonly0 0 0 10000 $y1
|
|
}
|
|
if {$x1 > 0} {
|
|
tixMoveResizeWindow $inputonly1 0 0 $x1 10000
|
|
}
|
|
tixMoveResizeWindow $inputonly2 0 $y2 10000 10000
|
|
tixMoveResizeWindow $inputonly3 $x2 0 10000 10000
|
|
|
|
for {set i 0} {$i < 4} {incr i} {
|
|
tixMapWindow [set inputonly$i]
|
|
tixRaiseWindow [set inputonly$i]
|
|
}
|
|
tixFlushX $w
|
|
} else {
|
|
tixMoveResizeWindow $inputonly0 0 0 10000 10000
|
|
tixMapWindow $inputonly0
|
|
tixRaiseWindow $inputonly0
|
|
}
|
|
} else {
|
|
tixUnmapWindow $inputonly0
|
|
tixUnmapWindow $inputonly1
|
|
tixUnmapWindow $inputonly2
|
|
tixUnmapWindow $inputonly3
|
|
|
|
if {$focuswin != "" && [winfo id $focuswin] != 0} {
|
|
if {[info exists tixBusy($focuswin,oldcursor)]} {
|
|
$focuswin config -cursor $tixBusy($focuswin,oldcursor)
|
|
if {[info exists tixBusy($focuswin,oldcursor)]} {
|
|
unset tixBusy($focuswin,oldcursor)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
proc tixOptionName {w} {
|
|
return [string range $w 1 end]
|
|
}
|
|
|
|
proc tixSetSilent {chooser value} {
|
|
$chooser config -disablecallback true
|
|
$chooser config -value $value
|
|
$chooser config -disablecallback false
|
|
}
|
|
|
|
# This command is useful if you want to ingore the arguments
|
|
# passed by the -command or -browsecmd options of the Tix widgets. E.g
|
|
#
|
|
# tixFileSelectDialog .c -command "puts foo; tixBreak"
|
|
#
|
|
#
|
|
proc tixBreak {args} {}
|
|
|
|
#----------------------------------------------------------------------
|
|
# tixDestroy -- deletes a Tix class object (not widget classes)
|
|
#----------------------------------------------------------------------
|
|
proc tixDestroy {w} {
|
|
upvar #0 $w data
|
|
|
|
set destructor ""
|
|
if {[info exists data(className)]} {
|
|
catch {
|
|
set destructor [tixGetMethod $w $data(className) Destructor]
|
|
}
|
|
}
|
|
if {$destructor != ""} {
|
|
$destructor $w
|
|
}
|
|
catch {rename $w ""}
|
|
catch {unset data}
|
|
return ""
|
|
}
|
|
|
|
proc tixPushGrab {args} {
|
|
global tix_priv
|
|
|
|
if {![info exists tix_priv(grab-list)]} {
|
|
set tix_priv(grab-list) ""
|
|
set tix_priv(grab-mode) ""
|
|
set tix_priv(grab-nopush) ""
|
|
}
|
|
|
|
set len [llength $args]
|
|
if {$len == 1} {
|
|
set opt ""
|
|
set w [lindex $args 0]
|
|
} elseif {$len == 2} {
|
|
set opt [lindex $args 0]
|
|
set w [lindex $args 1]
|
|
} else {
|
|
error "wrong # of arguments: tixPushGrab ?-global? window"
|
|
}
|
|
|
|
# Not everyone will call tixPushGrab. If someone else has a grab already
|
|
# save that one as well, so that we can restore that later
|
|
#
|
|
set last [lindex $tix_priv(grab-list) end]
|
|
set current [grab current $w]
|
|
|
|
if {$current ne "" && $current ne $last} {
|
|
# Someone called "grab" directly
|
|
#
|
|
lappend tix_priv(grab-list) $current
|
|
lappend tix_priv(grab-mode) [grab status $current]
|
|
lappend tix_priv(grab-nopush) 1
|
|
}
|
|
|
|
# Now push myself into the stack
|
|
#
|
|
lappend tix_priv(grab-list) $w
|
|
lappend tix_priv(grab-mode) $opt
|
|
lappend tix_priv(grab-nopush) 0
|
|
|
|
if {$opt eq "-global"} {
|
|
grab -global $w
|
|
} else {
|
|
grab $w
|
|
}
|
|
}
|
|
|
|
proc tixPopGrab {} {
|
|
global tix_priv
|
|
|
|
if {![info exists tix_priv(grab-list)]} {
|
|
set tix_priv(grab-list) ""
|
|
set tix_priv(grab-mode) ""
|
|
set tix_priv(grab-nopush) ""
|
|
}
|
|
|
|
set len [llength $tix_priv(grab-list)]
|
|
if {$len <= 0} {
|
|
error "no window is grabbed by tixGrab"
|
|
}
|
|
|
|
set w [lindex $tix_priv(grab-list) end]
|
|
grab release $w
|
|
|
|
if {$len > 1} {
|
|
set tix_priv(grab-list) [lrange $tix_priv(grab-list) 0 end-1]
|
|
set tix_priv(grab-mode) [lrange $tix_priv(grab-mode) 0 end-1]
|
|
set tix_priv(grab-nopush) [lrange $tix_priv(grab-nopush) 0 end-1]
|
|
|
|
set w [lindex $tix_priv(grab-list) end]
|
|
set m [lindex $tix_priv(grab-list) end]
|
|
set np [lindex $tix_priv(grab-nopush) end]
|
|
|
|
if {$np == 1} {
|
|
# We have a grab set by "grab"
|
|
#
|
|
set len [llength $tix_priv(grab-list)]
|
|
|
|
if {$len > 1} {
|
|
set tix_priv(grab-list) [lrange $tix_priv(grab-list) 0 end-1]
|
|
set tix_priv(grab-mode) [lrange $tix_priv(grab-mode) 0 end-1]
|
|
set tix_priv(grab-nopush) \
|
|
[lrange $tix_priv(grab-nopush) 0 end-1]
|
|
} else {
|
|
set tix_priv(grab-list) ""
|
|
set tix_priv(grab-mode) ""
|
|
set tix_priv(grab-nopush) ""
|
|
}
|
|
}
|
|
|
|
if {$m == "-global"} {
|
|
grab -global $w
|
|
} else {
|
|
grab $w
|
|
}
|
|
} else {
|
|
set tix_priv(grab-list) ""
|
|
set tix_priv(grab-mode) ""
|
|
set tix_priv(grab-nopush) ""
|
|
}
|
|
}
|
|
|
|
proc tixWithinWindow {wid rootX rootY} {
|
|
set wc [winfo containing $rootX $rootY]
|
|
if {$wid eq $wc} { return 1 }
|
|
|
|
# no see if it is an enclosing parent
|
|
set rx1 [winfo rootx $wid]
|
|
set ry1 [winfo rooty $wid]
|
|
set rw [winfo width $wid]
|
|
set rh [winfo height $wid]
|
|
set rx2 [expr {$rx1+$rw}]
|
|
set ry2 [expr {$ry1+$rh}]
|
|
|
|
if {$rootX >= $rx1 && $rootX < $rx2 && $rootY >= $ry1 && $rootY < $ry2} {
|
|
return 1
|
|
} else {
|
|
return 0
|
|
}
|
|
}
|
|
|
|
proc tixWinWidth {w} {
|
|
set W [winfo width $w]
|
|
set bd [expr {[$w cget -bd] + [$w cget -highlightthickness]}]
|
|
|
|
return [expr {$W - 2*$bd}]
|
|
}
|
|
|
|
proc tixWinHeight {w} {
|
|
set H [winfo height $w]
|
|
set bd [expr {[$w cget -bd] + [$w cget -highlightthickness]}]
|
|
|
|
return [expr {$H - 2*$bd}]
|
|
}
|
|
|
|
# junk?
|
|
#
|
|
proc tixWinCmd {w} {
|
|
return [winfo command $w]
|
|
}
|