home *** CD-ROM | disk | FTP | other *** search
/ PC World 2000 December / PCWorld_2000-12_cd.bin / Komunikace / Comanche / comanche.exe / lib / iwidgets2.2.0 / scripts / toolbar.itk < prev    next >
Text File  |  1999-02-24  |  32KB  |  991 lines

  1. #
  2. # Toolbar
  3. # ----------------------------------------------------------------------
  4. #
  5. # The Toolbar command creates a new window (given by the pathName 
  6. # argument) and makes it into a Tool Bar widget. Additional options, 
  7. # described above may be specified on the command line or in the 
  8. # option database to configure aspects of the Toolbar such as its 
  9. # colors, font, and orientation. The Toolbar command returns its 
  10. # pathName argument. At the time this command is invoked, there 
  11. # must not exist a window named pathName, but pathName's parent 
  12. # must exist.
  13. # A Toolbar is a widget that displays a collection of widgets arranged 
  14. # either in a row or a column (depending on the value of the -orient 
  15. # option). This collection of widgets is usually for user convenience 
  16. # to give access to a set of commands or settings. Any widget may be 
  17. # placed on a Toolbar. However, command or value-oriented widgets (such 
  18. # as button, radiobutton, etc.) are usually the most useful kind of 
  19. # widgets to appear on a Toolbar.
  20. #
  21. # WISH LIST: 
  22. #   This section lists possible future enhancements.  
  23. #
  24. #    Toggle between text and image/bitmap so that the toolbar could
  25. #     display either all text or all image/bitmaps.
  26. #   Implementation of the -toolbarfile option that allows toolbar
  27. #     add commands to be read in from a file.
  28. # ----------------------------------------------------------------------
  29. #  AUTHOR: Bill W. Scott                 EMAIL: bscott@spd.dsccc.com
  30. #
  31. #  @(#) $Id: toolbar.itk,v 1.1 1998/07/27 18:49:55 stanton Exp $
  32. # ----------------------------------------------------------------------
  33. #            Copyright (c) 1995 DSC Technologies Corporation
  34. # ======================================================================
  35. # Permission to use, copy, modify, distribute and license this software 
  36. # and its documentation for any purpose, and without fee or written 
  37. # agreement with DSC, is hereby granted, provided that the above copyright 
  38. # notice appears in all copies and that both the copyright notice and 
  39. # warranty disclaimer below appear in supporting documentation, and that 
  40. # the names of DSC Technologies Corporation or DSC Communications 
  41. # Corporation not be used in advertising or publicity pertaining to the 
  42. # software without specific, written prior permission.
  43. # DSC DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING 
  44. # ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, AND NON-
  45. # INFRINGEMENT. THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, AND THE
  46. # AUTHORS AND DISTRIBUTORS HAVE NO OBLIGATION TO PROVIDE MAINTENANCE, 
  47. # SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS. IN NO EVENT SHALL 
  48. # DSC BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR 
  49. # ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, 
  50. # WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTUOUS ACTION,
  51. # ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS 
  52. # SOFTWARE.
  53. # ======================================================================
  54.  
  55. #
  56. # Default resources.
  57. #
  58. option add *Toolbar*padX 5 widgetDefault
  59. option add *Toolbar*padY 5 widgetDefault
  60. option add *Toolbar*orient horizontal widgetDefault
  61. option add *Toolbar*highlightThickness 0 widgetDefault
  62. option add *Toolbar*indicatorOn false widgetDefault
  63. option add *Toolbar*selectColor [. cget -bg] widgetDefault
  64.  
  65. #
  66. # Usual options.
  67. #
  68. itk::usual Toolbar {
  69.     keep -activebackground -activeforeground -background -balloonbackground \
  70.      -balloondelay1 -balloondelay2 -balloonfont -balloonforeground \
  71.      -borderwidth -cursor -disabledforeground -font -foreground \
  72.      -highlightbackground -highlightcolor -highlightthickness \
  73.      -insertbackground -insertforeground -selectbackground \
  74.      -selectborderwidth -selectcolor -selectforeground -troughcolor
  75. }
  76.  
  77. # ------------------------------------------------------------------
  78. #                            TOOLBAR
  79. # ------------------------------------------------------------------
  80. class iwidgets::Toolbar {
  81.     inherit itk::Widget
  82.     
  83.     constructor {args} {}
  84.     
  85.     itk_option define -balloonbackground \
  86.         balloonBackground BalloonBackground yellow 
  87.     itk_option define -balloonforeground \
  88.         balloonForeground BalloonForeground black 
  89.     itk_option define -balloonfont balloonFont BalloonFont 6x10 
  90.     itk_option define -balloondelay1 \
  91.         balloonDelay1 BalloonDelay1 1000
  92.     itk_option define -balloondelay2 \
  93.         balloonDelay2 BalloonDelay2 200
  94.     itk_option define -helpvariable helpVariable HelpVariable {} 
  95.     itk_option define -orient orient Orient "horizontal" 
  96.     
  97.     #
  98.     # The following options implement propogated configurations to
  99.     # any widget that might be added to us. The problem is this is
  100.     # not deterministic as someone might add a new kind of widget with
  101.     # and option like -armbackground, so we would not be aware of
  102.     # this kind of option. Anyway we support as many of the obvious
  103.     # ones that we can. They can always configure them with itemconfigures.
  104.     #
  105.     itk_option define -activebackground activeBackground Foreground #c3c3c3
  106.     itk_option define -activeforeground activeForeground Background Black 
  107.     itk_option define -background background Background #d9d9d9 
  108.     itk_option define -borderwidth borderWidth BorderWidth 2 
  109.     itk_option define -cursor cursor Cursor {}
  110.     itk_option define -disabledforeground \
  111.         disabledForeground DisabledForeground #a3a3a3 
  112.     itk_option define -font \
  113.         font Font "-Adobe-Helvetica-Bold-R-Normal--*-120-*-*-*-*-*-*" 
  114.     itk_option define -foreground foreground Foreground #000000000000 
  115.     itk_option define -highlightbackground \
  116.         highlightBackground HighlightBackground #d9d9d9 
  117.     itk_option define -highlightcolor highlightColor HighlightColor Black 
  118.     itk_option define -highlightthickness \
  119.         highlightThickness HighlightThickness 0 
  120.     itk_option define -insertforeground insertForeground Background #c3c3c3 
  121.     itk_option define -insertbackground insertBackground Foreground Black 
  122.     itk_option define -selectbackground selectBackground Foreground #c3c3c3 
  123.     itk_option define -selectborderwidth selectBorderWidth BorderWidth {} 
  124.     itk_option define -selectcolor selectColor Background #b03060 
  125.     itk_option define -selectforeground selectForeground Background Black 
  126.     itk_option define -state state State normal 
  127.     itk_option define -troughcolor troughColor Background #c3c3c3 
  128.     
  129.     public method add {widgetCommand name args} 
  130.     public method delete {args} 
  131.     public method index {index} 
  132.     public method insert {beforeIndex widgetCommand name args} 
  133.     public method itemcget {index args} 
  134.     public method itemconfigure {index args} 
  135.     
  136.     public method _resetBalloonTimer {}
  137.     public method _startBalloonDelay {window}
  138.     public method _stopBalloonDelay {window balloonClick}
  139.  
  140.     private method _deleteWidgets {index1 index2} 
  141.     private method _addWidget {widgetCommand name args}
  142.     private method _index {toolList index} 
  143.     private method _getAttachedOption {optionListName widget args retValue} 
  144.     private method _setAttachedOption {optionListName widget option args} 
  145.     private method _packToolbar {} 
  146.     
  147.     public method hideHelp {} 
  148.     public method showHelp {window} 
  149.     public method showBalloon {window} 
  150.     public method hideBalloon {} 
  151.     
  152.     private variable _balloonTimer 0
  153.     private variable _balloonAfterID 0
  154.     private variable _balloonClick false
  155.     
  156.     private variable _initialMapping 1   ;# Is this the first mapping?
  157.     private variable _toolList {}        ;# List of all widgets on toolbar
  158.     private variable _opts               ;# New options for child widgets
  159.     private variable _currHelpWidget {}  ;# Widget currently displaying help for
  160.     private common   _hintWindow {}      ;# Balloon help bubble.
  161.     
  162.     # list of options we want to propogate to widgets added to toolbar.
  163.     private common _optionList {
  164.     -activebackground \
  165.         -activeforeground \
  166.         -background \
  167.         -borderwidth \
  168.         -cursor \
  169.         -disabledforeground \
  170.         -font \
  171.         -foreground \
  172.         -highlightbackground \
  173.         -highlightcolor \
  174.         -highlightthickness \
  175.         -insertbackground \
  176.         -insertforeground \
  177.         -selectbackground \
  178.         -selectborderwidth \
  179.         -selectcolor \
  180.         -selectforeground \
  181.         -state \
  182.         -troughcolor \
  183.         }
  184. }
  185.  
  186. # ------------------------------------------------------------------
  187. #                            CONSTRUCTOR 
  188. # ------------------------------------------------------------------
  189. body iwidgets::Toolbar::constructor {args} {
  190.     #
  191.     # Create the Toolbar frame
  192.     #
  193.     itk_component add tbFrame {
  194.     frame $itk_interior.tbFrame 
  195.     } {
  196.     keep -background -cursor -relief 
  197.     }
  198.     pack $itk_component(tbFrame) -fill x
  199.     
  200.     #
  201.     # Handle configs
  202.     #
  203.     eval itk_initialize $args
  204.     
  205.     # If variable has NOT been set
  206.     if { $_hintWindow == {} } {
  207.     # if widget does NOT exist
  208.     if { ! [winfo exists $_hintWindow] } {
  209.         # create menu and one liner
  210.         set _hintWindow \
  211.             [menu .balloonHintWindow \
  212.             -tearoff no \
  213.             -foreground $itk_option(-balloonforeground) \
  214.             -background $itk_option(-balloonbackground) \
  215.             -borderwidth 1 \
  216.             -activeforeground $itk_option(-balloonforeground) \
  217.             -activebackground $itk_option(-balloonbackground) \
  218.             -activeborderwidth 0 \
  219.             -font $itk_option(-balloonfont) ]
  220.         $_hintWindow add command
  221.     }
  222.     }
  223.     
  224.     # ... Attach help handler to this widget
  225.     bind toolbar-help-$itk_component(hull) \
  226.         <Enter> "+[code $this showHelp %W]"
  227.     bind toolbar-help-$itk_component(hull) \
  228.         <Leave> "+[code $this hideHelp]"
  229.     
  230.     # ... Set up Microsoft style balloon help display.
  231.     set _balloonTimer $itk_option(-balloondelay1)
  232.     bind $itk_component(tbFrame) \
  233.         <Leave> "+[code $this _resetBalloonTimer]"
  234.     bind toolbar-balloon-$itk_component(hull) \
  235.         <Enter> "+[code $this _startBalloonDelay %W]"
  236.     bind toolbar-balloon-$itk_component(hull) \
  237.         <Leave> "+[code $this _stopBalloonDelay %W false]"
  238.     bind toolbar-balloon-$itk_component(hull) \
  239.         <Button-1> "+[code $this _stopBalloonDelay %W true]"
  240. }
  241.  
  242. #
  243. # Provide a lowercase access method for the Toolbar class
  244. #
  245. proc ::iwidgets::toolbar {pathName args} {
  246.     uplevel ::iwidgets::Toolbar $pathName $args
  247. }
  248.  
  249. # ------------------------------------------------------------------
  250. #                          OPTIONS
  251. # ------------------------------------------------------------------
  252.  
  253. # ------------------------------------------------------------------
  254. # OPTION -balloonbackground
  255. # ------------------------------------------------------------------
  256. configbody iwidgets::Toolbar::balloonbackground {
  257.     if { [winfo exists .balloonHintWindow] } {
  258.     if { $itk_option(-balloonbackground) != {} } {
  259.         .balloonHintWindow configure \
  260.             -bg $itk_option(-balloonbackground) \
  261.             -activebackground $itk_option(-balloonbackground)
  262.     }
  263.     }
  264. }
  265.  
  266. # ------------------------------------------------------------------
  267. # OPTION -balloonforeground
  268. # ------------------------------------------------------------------
  269. configbody iwidgets::Toolbar::balloonforeground {
  270.     if { [winfo exists .balloonHintWindow] } {
  271.     if { $itk_option(-balloonforeground) != {} } {
  272.         .balloonHintWindow configure \
  273.             -foreground $itk_option(-balloonforeground) \
  274.             -activeforeground $itk_option(-balloonforeground)
  275.     }
  276.     }
  277. }
  278.  
  279. # ------------------------------------------------------------------
  280. # OPTION -balloonfont
  281. # ------------------------------------------------------------------
  282. configbody iwidgets::Toolbar::balloonfont {
  283.     if { [winfo exists .balloonHintWindow] } {
  284.     if { $itk_option(-balloonfont) != {} } {
  285.         .balloonHintWindow configure \
  286.             -font $itk_option(-balloonfont) 
  287.     }
  288.     }
  289. }
  290.  
  291. # ------------------------------------------------------------------
  292. # OPTION: -orient
  293. #
  294. # Position buttons either horizontally or vertically.
  295. # ------------------------------------------------------------------
  296. configbody iwidgets::Toolbar::orient {
  297.     switch $itk_option(-orient) {
  298.     "horizontal" - "vertical" {
  299.         _packToolbar
  300.     }
  301.     default {error "Invalid orientation. Must be either \
  302.         horizontal or vertical"
  303.         }
  304.     }
  305. }
  306.  
  307. # ------------------------------------------------------------------
  308. #                            METHODS
  309. # ------------------------------------------------------------------
  310.     
  311. # -------------------------------------------------------------
  312. # METHOD: add widgetCommand name ?option value?
  313. #
  314. # Adds a widget with the command widgetCommand whose name is 
  315. # name to the Toolbar.   If widgetCommand is radiobutton 
  316. # or checkbutton, its packing is slightly padded to match the 
  317. # geometry of button widgets.
  318. # -------------------------------------------------------------
  319. body iwidgets::Toolbar::add { widgetCommand name args } {
  320.     
  321.     eval "_addWidget $widgetCommand $name $args"
  322.     
  323.     lappend _toolList $itk_component($name)
  324.     
  325.     if { $widgetCommand == "radiobutton" || \
  326.         $widgetCommand == "checkbutton" } {
  327.     set iPad 1
  328.     } else {
  329.     set iPad 0
  330.     }
  331.     
  332.     if {$itk_option(-orient) == "horizontal" } {
  333.     pack $itk_component($name) \
  334.         -anchor nw -side left -fill x -ipadx $iPad -ipady $iPad
  335.     } else {
  336.     pack $itk_component($name) \
  337.         -side top -anchor nw -fill y -ipadx $iPad -ipady $iPad
  338.     }
  339.     
  340.     return $itk_component($name)
  341.     
  342. }
  343.  
  344. # -------------------------------------------------------------
  345. #
  346. # METHOD: delete index ?index2?
  347. #
  348. # This command deletes all components between index and 
  349. # index2 inclusive. If index2 is omitted then it defaults 
  350. # to index. Returns an empty string
  351. #
  352. # -------------------------------------------------------------
  353. body iwidgets::Toolbar::delete { args } {
  354.     # empty toolbar
  355.     if { $_toolList == {} } {
  356.     error "can't delete widget, no widgets in the Toolbar \
  357.         \"$itk_component(hull)\""
  358.     }
  359.     
  360.     set len [llength $args]
  361.     switch -- $len {
  362.     1 {
  363.         set fromWidget [_index $_toolList [lindex $args 0]]
  364.         
  365.         if { $fromWidget < 0 || $fromWidget >= [llength $_toolList] } {
  366.         error "bad Toolbar widget index in delete method: \
  367.             should be between 0 and [expr [llength $_toolList] - 1]"
  368.         }
  369.         
  370.         set toWidget $fromWidget
  371.         _deleteWidgets $fromWidget $toWidget
  372.     }
  373.     
  374.     2 {
  375.         set fromWidget [_index $_toolList [lindex $args 0]]
  376.         
  377.         if { $fromWidget < 0 || $fromWidget >= [llength $_toolList] } {
  378.         error "bad Toolbar widget index1 in delete method: \
  379.             should be between 0 and [expr [llength $_toolList] - 1]"
  380.         }
  381.         
  382.         set toWidget [_index $_toolList [lindex $args 1]]
  383.         
  384.         if { $toWidget < 0 || $toWidget >= [llength $_toolList] } {
  385.         error "bad Toolbar widget index2 in delete method: \
  386.             should be between 0 and [expr [llength $_toolList] - 1]"
  387.         }
  388.         
  389.         if { $fromWidget > $toWidget } {
  390.         error "bad Toolbar widget index1 in delete method: \
  391.             index1 is greater than index2"
  392.         }
  393.         
  394.         _deleteWidgets $fromWidget $toWidget
  395.     }
  396.     
  397.     default {
  398.         # ... too few/many parameters passed
  399.         error "wrong # args: should be \
  400.             \"$itk_component(hull) delete index1 ?index2?\""
  401.     }
  402.     }
  403.     
  404.     return {}
  405. }
  406.  
  407.  
  408. # -------------------------------------------------------------
  409. #
  410. # METHOD: index index 
  411. #
  412. # Returns the widget's numerical index for the entry corresponding 
  413. # to index. If index is not found, -1 is returned
  414. #
  415. # -------------------------------------------------------------
  416. body iwidgets::Toolbar::index { index } {
  417.     
  418.     return [_index $_toolList $index]
  419.     
  420. }
  421.  
  422. # -------------------------------------------------------------
  423. #
  424. # METHOD: insert beforeIndex widgetCommand name ?option value?
  425. #
  426. # Insert a new component named name with the command 
  427. # widgetCommand before the com ponent specified by beforeIndex. 
  428. # If widgetCommand is radiobutton or checkbutton, its packing 
  429. # is slightly padded to match the geometry of button widgets.
  430. #
  431. # -------------------------------------------------------------
  432. body iwidgets::Toolbar::insert {  beforeIndex widgetCommand name args } {
  433.     
  434.     set beforeIndex [_index $_toolList $beforeIndex]
  435.     
  436.     if {$beforeIndex < 0 || $beforeIndex > [llength $_toolList] } {
  437.     error "bad toolbar entry index $beforeIndex"
  438.     }
  439.     
  440.     eval "_addWidget $widgetCommand $name $args"
  441.     
  442.     # linsert into list
  443.     set _toolList [linsert $_toolList $beforeIndex $itk_component($name)]
  444.     
  445.     # repack the tool bar
  446.     _packToolbar
  447.     
  448.     return $itk_component($name)
  449.     
  450. }
  451.  
  452. # ----------------------------------------------------------------------
  453. # METHOD: itemcget index ?option? 
  454. #
  455. # Returns the value for the option setting of the widget at index $index.
  456. # index can be numeric or widget name
  457. #
  458. # ----------------------------------------------------------------------
  459. body iwidgets::Toolbar::itemcget { index args} {
  460.     
  461.     return [lindex [eval itemconfigure $index $args] 4]
  462. }
  463.  
  464. # -------------------------------------------------------------
  465. #
  466. # METHOD: itemconfigure index ?option? ?value? ?option value...?
  467. #
  468. # Query or modify the configuration options of the widget of 
  469. # the Toolbar specified by index. If no option is specified, 
  470. # returns a list describing all of the available options for 
  471. # index (see Tk_ConfigureInfo for information on the format 
  472. # of this list). If option is specified with no value, then 
  473. # the command returns a list describing the one named option 
  474. # (this list will be identical to the corresponding sublist 
  475. # of the value returned if no option is specified). If one 
  476. # or more option-value pairs are specified, then the command 
  477. # modifies the given widget option(s) to have the given 
  478. # value(s); in this case the command returns an empty string. 
  479. # The component type of index determines the valid available options.
  480. #
  481. # -------------------------------------------------------------
  482. body iwidgets::Toolbar::itemconfigure { index args } {
  483.     
  484.     # Get a numeric index.
  485.     set index [_index $_toolList $index]
  486.     
  487.     # Get the tool path
  488.     set toolPath [lindex $_toolList $index]
  489.     
  490.     set len [llength $args]
  491.     
  492.     switch $len {
  493.     0 {
  494.         # show all options
  495.         # ''''''''''''''''
  496.         
  497.         # support display of -helpstr and -balloonstr configs
  498.         set optList [$toolPath configure]
  499.         
  500.         ## @@@ might want to use _getAttachedOption instead...
  501.         if { [info exists _opts($toolPath,-helpstr)] } {
  502.         set value $_opts($toolPath,-helpstr)
  503.         } else {
  504.         set value {}
  505.         }
  506.         lappend optList [list -helpstr helpStr HelpStr {} $value]
  507.         if { [info exists _opts($toolPath,-balloonstr)] } {
  508.         set value $_opts($toolPath,-balloonstr)
  509.         } else {
  510.         set value {}
  511.         }
  512.         lappend optList [list -balloonstr balloonStr BalloonStr {} $value]
  513.         return $optList
  514.     }
  515.     1 {
  516.         # show only option specified
  517.         # ''''''''''''''''''''''''''
  518.         # did we satisfy the option get request?
  519.         
  520.         if { [regexp -- {-helpstr} $args] } {
  521.         if { [info exists _opts($toolPath,-helpstr)] } {
  522.             set value $_opts($toolPath,-helpstr)
  523.         } else {
  524.             set value {}
  525.         }
  526.         return [list -helpstr helpStr HelpStr {} $value]
  527.         } elseif { [regexp -- {-balloonstr} $args] } {
  528.         if { [info exists _opts($toolPath,-balloonstr)] } {
  529.             set value $_opts($toolPath,-balloonstr)
  530.         } else {
  531.             set value {}
  532.         }
  533.         return [list -balloonstr balloonStr BalloonStr {} $value]
  534.         } else {
  535.         return [eval $toolPath configure $args]
  536.         }
  537.         
  538.     }
  539.     default {
  540.         # ... do a normal configure
  541.         
  542.         # first screen for all our child options we are adding
  543.         _setAttachedOption \
  544.             _opts \
  545.             $toolPath \
  546.             "-helpstr" \
  547.             $args
  548.         
  549.         _setAttachedOption \
  550.             _opts \
  551.             $toolPath \
  552.             "-balloonstr" \
  553.             $args 
  554.         
  555.         # with a clean args list do a configure
  556.         
  557.         # if the stripping process brought us down to no options
  558.         # to set, then forget the configure of widget.
  559.         if { [llength $args] != 0 } {
  560.         return [eval $toolPath configure $args]
  561.         } else {
  562.         return ""
  563.         }
  564.     }
  565.     }
  566.     
  567. }
  568.  
  569. # -------------------------------------------------------------
  570. #
  571. # METHOD: _resetBalloonDelay1 
  572. #
  573. # Sets the delay that will occur before a balloon could be popped
  574. # up to balloonDelay1
  575. #
  576. # -------------------------------------------------------------
  577. body iwidgets::Toolbar::_resetBalloonTimer {} {
  578.     set _balloonTimer $itk_option(-balloondelay1)
  579.     
  580.     # reset the <1> longer delay
  581.     set _balloonClick false
  582. }
  583.  
  584. # -------------------------------------------------------------
  585. #
  586. # METHOD: _startBalloonDelay 
  587. #
  588. # Starts waiting to pop up a balloon id
  589. #
  590. # -------------------------------------------------------------
  591. body iwidgets::Toolbar::_startBalloonDelay {window} {
  592.     set _balloonAfterID [after $_balloonTimer [code $this showBalloon $window]]
  593. }
  594.  
  595. # -------------------------------------------------------------
  596. #
  597. # METHOD: _stopBalloonDelay  
  598. #
  599. # This method will stop the timer for a balloon popup if one is
  600. # in progress. If however there is already a balloon window up
  601. # it will hide the balloon window and set timing to delay 2 stage.
  602. #
  603. # -------------------------------------------------------------
  604. body iwidgets::Toolbar::_stopBalloonDelay { window balloonClick } {
  605.     
  606.     # If <1> then got a click cancel
  607.     if { $balloonClick } {
  608.     set _balloonClick true
  609.     }
  610.     if { $_balloonAfterID != 0 } {
  611.     after cancel $_balloonAfterID
  612.     set _balloonAfterID 0
  613.     } else {
  614.     hideBalloon
  615.     
  616.     # If this was cancelled with a <1> use longer delay.
  617.     if { $_balloonClick } {
  618.         set _balloonTimer $itk_option(-balloondelay1)
  619.     } else {
  620.         set _balloonTimer $itk_option(-balloondelay2)
  621.     }
  622.     }
  623. }
  624.  
  625. # -------------------------------------------------------------
  626. # PRIVATE METHOD: _addWidget
  627. #
  628. # widgetCommand : command to invoke to create the added widget
  629. # name          : name of the new widget to add
  630. # args          : options for the widget create command
  631. #
  632. # Looks for -helpstr, -balloonstr and grabs them, strips from
  633. # args list. Then tries to add a component and keeps based
  634. # on known type. If it fails, it tries to clean up. Then it
  635. # binds handlers for helpstatus and balloon help.
  636. #
  637. # Returns the path of the widget added.
  638. #
  639. # -------------------------------------------------------------
  640. body iwidgets::Toolbar::_addWidget { widgetCommand name args } {
  641.     
  642.     # ,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,,
  643.     # Add the widget to the tool bar
  644.     # '''''''''''''''''''''''''''''''''''''''''''''''''''''
  645.     
  646.     # ... Strip out and save the -helpstr, -balloonstr options from args
  647.     #     and save it in _opts
  648.     _setAttachedOption \
  649.         _opts \
  650.         $itk_component(tbFrame).$name \
  651.         -helpstr \
  652.         $args 
  653.     
  654.     _setAttachedOption \
  655.         _opts \
  656.         $itk_component(tbFrame).$name \
  657.         -balloonstr \
  658.         $args
  659.     
  660.     
  661.     # ... Add the new widget as a component (catch an error if occurs)
  662.     set createFailed [catch { 
  663.     itk_component add $name {
  664.         eval $widgetCommand $itk_component(tbFrame).$name $args
  665.     } {
  666.     }
  667.     } errMsg]
  668.     
  669.     # ... Clean up if the create failed, and exit.
  670.     #     The _opts list if it has -helpstr, -balloonstr just entered for
  671.     #     this, it must be cleaned up.
  672.     if { $createFailed } {
  673.     # clean up
  674.     if {![catch {set _opts($itk_component(tbFrame).$name,-helpstr)}]} {
  675.         set lastIndex [\
  676.             expr [llength \
  677.             $_opts($itk_component(tbFrame).$name,-helpstr) ]-1]
  678.         lreplace $_opts($itk_component(tbFrame).$name,-helpstr) \
  679.             $lastIndex $lastIndex ""
  680.     }
  681.     if {![catch {set _opts($itk_component(tbFrame).$name,-balloonstr)}]} {
  682.         set lastIndex [\
  683.             expr [llength \
  684.             $_opts($itk_component(tbFrame).$name,-balloonstr) ]-1]
  685.         lreplace $_opts($itk_component(tbFrame).$name,-balloonstr) \
  686.             $lastIndex $lastIndex ""
  687.     }
  688.     error $errMsg
  689.     }
  690.     
  691.     # ... Add in dynamic options that apply from the _optionList
  692.     foreach optionSet [$itk_component($name) configure] {
  693.     set option [lindex $optionSet 0]
  694.     if { [lsearch $_optionList $option] != -1 } {
  695.         itk_option add $name.$option
  696.     }
  697.     }
  698.     
  699.     bindtags $itk_component($name) \
  700.         [linsert [bindtags $itk_component($name)] end \
  701.         toolbar-help-$itk_component(hull)]
  702.     bindtags $itk_component($name) \
  703.         [linsert [bindtags $itk_component($name)] end \
  704.         toolbar-balloon-$itk_component(hull)]
  705.     
  706.     return $itk_component($name)
  707. }
  708.  
  709. # -------------------------------------------------------------
  710. #
  711. # PRIVATE METHOD: _deleteWidgets
  712. #
  713. # deletes widget range by numerical index numbers.
  714. #
  715. # -------------------------------------------------------------
  716. body iwidgets::Toolbar::_deleteWidgets { index1 index2 } {
  717.     
  718.     for { set index $index1 } { $index <= $index2 } { incr index } {
  719.     
  720.     # kill the widget
  721.     set component [lindex $_toolList $index]
  722.     destroy $component
  723.     
  724.     }
  725.     
  726.     # physically remove the page
  727.     set _toolList [lreplace $_toolList $index1 $index2]
  728.     
  729. }
  730.  
  731. # -------------------------------------------------------------
  732. # PRIVATE METHOD: _index
  733. #
  734. # toolList : list of widget names to search thru if index 
  735. #            is non-numeric
  736. # index    : either number, 'end', 'last', or pattern
  737. #
  738. # _index takes takes the value $index converts it to
  739. # a numeric identifier. If the value is not already
  740. # an integer it looks it up in the $toolList array.
  741. # If it fails it returns -1
  742. #
  743. # -------------------------------------------------------------
  744. body iwidgets::Toolbar::_index { toolList index } {
  745.     
  746.     switch -- $index {
  747.     end - last {
  748.         set number [expr [llength $toolList] -1]
  749.     }
  750.     default {
  751.         # is it a number already? Then just use the number
  752.         if { [regexp {^[0-9]+$} $index] } {
  753.         set number $index
  754.         # check bounds
  755.         if { $number < 0 || $number >= [llength $toolList] } {
  756.             set number -1
  757.         }
  758.         # otherwise it is a widget name
  759.         } else {
  760.         if { [catch { set itk_component($index) } ] } {
  761.             set number -1
  762.         } else {
  763.             set number [lsearch -exact $toolList \
  764.                 $itk_component($index)]
  765.         }
  766.         }
  767.     }
  768.     }
  769.     
  770.     return $number
  771. }
  772.     
  773. # ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  774. # STATUS HELP for linking to helpVariable
  775. # ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  776. # -------------------------------------------------------------
  777. # PUBLIC METHOD: hideHelp
  778. #
  779. # Bound to the <Leave> event on a toolbar widget. This clears the
  780. # status widget help area and resets the help entry.
  781. #
  782. # -------------------------------------------------------------
  783. body iwidgets::Toolbar::hideHelp {} {
  784.     if { $itk_option(-helpvariable) != {} } {
  785.         upvar #0 $itk_option(-helpvariable) helpvar
  786.     set helpvar {}
  787.     }
  788.     set _currHelpWidget {}
  789. }
  790.  
  791. # -------------------------------------------------------------
  792. # PUBLIC METHOD: showHelp
  793. #
  794. # Bound to the <Motion> event on a tool bar widget. This puts the
  795. # help string associated with the tool bar widget into the 
  796. # status widget help area. If no help exists for the current
  797. # entry, the status widget is cleared.
  798. #
  799. # -------------------------------------------------------------
  800. body iwidgets::Toolbar::showHelp { window } {
  801.     
  802.     set widgetPath $window
  803.     # already on this item?
  804.     if { $window == $_currHelpWidget } {
  805.     return
  806.     }
  807.     
  808.     set _currHelpWidget $window
  809.     
  810.     # Do we have a helpvariable set on the toolbar?
  811.     if { $itk_option(-helpvariable) != {} } {
  812.         upvar #0 $itk_option(-helpvariable) helpvar
  813.     
  814.     # is the -helpstr set for this widget?
  815.     set args "-helpstr"
  816.     if {[_getAttachedOption _opts \
  817.         $window args value]} {
  818.         set helpvar $value.
  819.     } else {
  820.         set helpvar {}
  821.     }
  822.     }
  823. }
  824.  
  825. # ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  826. # BALLOON HELP for show/hide of hint window
  827. # ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  828. # -------------------------------------------------------------
  829. # PUBLIC METHOD: showBalloon
  830. #
  831. # -------------------------------------------------------------
  832. body iwidgets::Toolbar::showBalloon {window} {
  833.     set _balloonClick false
  834.     set _balloonAfterID 0
  835.     # Are we still inside the window?
  836.     set mouseWindow \
  837.         [winfo containing [winfo pointerx .] [winfo pointery .]]
  838.  
  839.     if { [string match $window* $mouseWindow] } {
  840.     # set up the balloonString
  841.     set args "-balloonstr"
  842.     if {[_getAttachedOption _opts \
  843.         $window args hintStr]} {
  844.         # configure the popup menu (balloon help)
  845.         $_hintWindow entryconfigure 0 -label $hintStr
  846.         
  847.         # Coordinates of the balloon
  848.         set balloonLeft \
  849.             [expr [winfo rootx $window] + round(([winfo width $window]/2.0))]
  850.         set balloonTop \
  851.             [expr [winfo rooty $window] + [winfo height $window]]
  852.         
  853.         # put up balloon window
  854.         $_hintWindow post $balloonLeft $balloonTop
  855.     } else {
  856.         #NO BALLOON HELP AVAILABLE
  857.     }
  858.     } else {
  859.     #NOT IN BUTTON
  860.     }
  861.     
  862. }
  863.  
  864. # -------------------------------------------------------------
  865. # PUBLIC METHOD: hideBalloon
  866. #
  867. # -------------------------------------------------------------
  868. body iwidgets::Toolbar::hideBalloon {} {
  869.     $_hintWindow unpost 
  870. }
  871.  
  872. # ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  873. # OPTION MANAGEMENT for -helpstr, -balloonstr
  874. # ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  875. # -------------------------------------------------------------
  876. # PRIVATE METHOD: _getAttachedOption
  877. #
  878. # optionListName : the name of the array that holds all attached
  879. #              options. It is indexed via widget,option to get
  880. #              the value.
  881. # widget     : the widget that the option is associated with
  882. # option     : the option whose value we are looking for on 
  883. #              this widget.
  884. #
  885. # expects to be called only if the $option is length 1
  886. # -------------------------------------------------------------
  887. body iwidgets::Toolbar::_getAttachedOption { optionListName widget args retValue} {
  888.     
  889.     # get a reference to the option, so we can change it.
  890.     upvar $args argsRef
  891.     upvar $retValue retValueRef
  892.     
  893.     set success false
  894.     
  895.     if { ![catch { set retValueRef \
  896.         [eval set [subst [set optionListName]]($widget,$argsRef)]}]} {
  897.     
  898.     # remove the option argument 
  899.     set success true
  900.     set argsRef ""
  901.     }
  902.     
  903.     return $success
  904. }
  905.  
  906. # -------------------------------------------------------------
  907. # PRIVATE METHOD: _setAttachedOption
  908. #
  909. # This method allows us to attach new options to a widget. It
  910. # catches the 'option' to be attached, strips it out of 'args'
  911. # attaches it to the 'widget' by stuffing the value into
  912. # 'optionList(widget,option)'
  913. #
  914. # optionListName:  where to store the option and widget association
  915. # widget: is the widget we want to associate the attached option
  916. # option: is the attached option (unknown to this widget)
  917. # args:   the arg list to search and remove the option from (if found)
  918. #
  919. # Modifies the args parameter.
  920. # Returns boolean indicating the success of the method
  921. #
  922. # -------------------------------------------------------------
  923. body iwidgets::Toolbar::_setAttachedOption {optionListName widget option args} {
  924.     
  925.     upvar args argsRef
  926.     
  927.     set success false
  928.     
  929.     # check for 'option' in the 'args' list for the 'widget'
  930.     set optPos [eval lsearch $args $option]
  931.     
  932.     # ... found it
  933.     if { $optPos != -1 } {
  934.     # grab a copy of the option from arg list
  935.     set [subst [set optionListName]]($widget,$option) \
  936.         [eval lindex $args [expr $optPos + 1]]
  937.     
  938.     # remove the option argument and value from the arg list
  939.     set argsRef [eval lreplace $args $optPos [expr $optPos + 1]]
  940.     set success true
  941.     }
  942.     # ... if not found, will leave args alone
  943.     
  944.     return $success
  945. }
  946.  
  947. # ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  948. # GEOMETRY MANAGEMENT for tool widgets
  949. # ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  950. # -------------------------------------------------------------
  951. # PRIVATE METHOD: _packToolbar
  952. #
  953. #
  954. #
  955. # -------------------------------------------------------------
  956. body iwidgets::Toolbar::_packToolbar {} {
  957.     
  958.     # forget the previous locations
  959.     foreach tool $_toolList {
  960.     pack forget $tool
  961.     }
  962.     
  963.     # pack in order of _toolList.
  964.     foreach tool $_toolList {
  965.     # adjust for radios and checks to match buttons
  966.     if { [winfo class $tool] == "Radiobutton" || 
  967.     [winfo class $tool] == "Checkbutton" } {
  968.         set iPad 1
  969.     } else {
  970.         set iPad 0
  971.     }
  972.     
  973.     # pack by horizontal or vertical orientation
  974.     if {$itk_option(-orient) == "horizontal" } {
  975.         pack $tool -anchor nw -side left -fill x \
  976.             -ipadx $iPad -ipady $iPad
  977.     } else {
  978.         pack $tool -side top -anchor nw -fill y \
  979.             -ipadx $iPad -ipady $iPad
  980.     }
  981.     }
  982. }
  983.  
  984.