#!/bin/wish84.exe # RCS: @(#) $Id: demo.tcl,v 1.53 2006/11/18 04:39:45 treectrl Exp $ set VERSION 2.2 package require Tk 8.4 set thisPlatform $::tcl_platform(platform) if {$thisPlatform eq "unix" && [tk windowingsystem] eq "aqua"} { set thisPlatform "macosx" } switch -- [tk windowingsystem] { aqua { set thisPlatform "macosx" } classic { set thisPlatform "macintosh" } win32 { set thisPlatform "windows" } x11 { set thisPlatform "unix" } } # Get full pathname to this file set ScriptDir [file normalize [file dirname [info script]]] # Command to create a full pathname in this file's directory proc Path {args} { return [file normalize [eval [list file join $::ScriptDir] $args]] } # Create some photo images on demand proc InitPics {args} { foreach pattern $args { if {[lsearch [image names] $pattern] == -1} { foreach file [glob -directory [Path pics] $pattern.gif] { set imageName [file root [file tail $file]] # I created an image called "file", which clobbered the # original Tcl command "file". Then I got confused. if {[llength [info commands $imageName]]} { error "don't want to create image called \"$imageName\"" } image create photo $imageName -file $file # Hack -- Create a "selected" version too image create photo ${imageName}Sel ${imageName}Sel copy $imageName imagetint ${imageName}Sel $::SystemHighlight 128 } } } return } if {[catch { package require dbwin 1.0 }]} { proc dbwin s {puts -nonewline $s} } set tile 0 set tileFull 0 ; # 1 if using tile-aware treectrl catch { package require tile 0.7.8 namespace export style namespace eval ::tile { namespace export setTheme } namespace eval ::ttk { namespace import ::style namespace import ::tile::setTheme } set tile 1 } if {$tile} { # Don't import ttk::entry, it messes up the edit bindings, and I'm not # sure how to get/set the equivalent -borderwidth, -selectborderwidth # etc options of a TEntry. set entryCmd ::ttk::entry set buttonCmd ::ttk::button set checkbuttonCmd ::ttk::checkbutton set radiobuttonCmd ttk::radiobutton set scrollbarCmd ::ttk::scrollbar } else { set entryCmd ::entry set buttonCmd ::button set checkbuttonCmd ::checkbutton set radiobuttonCmd ::radiobutton set scrollbarCmd ::scrollbar } # This gets called if 'package require' won't work during development. proc LoadSharedLibrary {} { switch -- $::thisPlatform { macintosh { set pattern treectrl*.shlb } macosx { set pattern treectrl*.dylib } unix { set pattern libtreectrl*[info sharedlibextension]* } windows { set pattern treectrl*[info sharedlibextension] } } set SHLIB [glob -nocomplain -directory [Path ..] $pattern] if {[llength $SHLIB] != 1} { return 0 } # When using configure/make, the "make demo" Makefile target sets the value of # the TREECTRL_LIBRARY environment variable which is used by tcl_findLibrary to # find our treectrl.tcl file. When *not* using configure/make, we set the value # of TREECTRL_LIBRARY and load the shared library manually. Note that # tcl_findLibrary is called by the Treectrl_Init() routine in C. set ::env(TREECTRL_LIBRARY) [Path .. library] load $SHLIB return 1 } # See if treectrl is already loaded for some reason if {[llength [info commands treectrl]]} { puts "demo.tcl: using previously-loaded treectrl package v[package provide treectrl]" if {$VERSION ne [package provide treectrl]} { puts "demo.tcl: WARNING: expected v$VERSION" } # For 'package require' to work with the development version, make sure the # TCLLIBPATH and TREECTRL_LIBRARY environment variables are set by your # Makefile/Jamfile/IDE etc. } elseif {![catch {package require treectrl $VERSION} err]} { puts "demo.tcl: 'package require' succeeded" } else { puts "demo.tcl: 'package require' failed: >>> $err <<<" if {[LoadSharedLibrary]} { puts "demo.tcl: loaded treectrl library by hand" } else { error "demo.tcl: can't load treectrl package" } } # Display path of shared library that was loaded foreach list [info loaded] { set file [lindex $list 0] set pkg [lindex $list 1] if {$pkg ne "Treectrl"} continue puts "demo.tcl: using '$file'" break } if {[info exists env(TREECTRL_LIBRARY)]} { puts "demo.tcl: TREECTRL_LIBRARY=$env(TREECTRL_LIBRARY)" } else { puts "demo.tcl: TREECTRL_LIBRARY undefined" } puts "demo.tcl: treectrl_library=$treectrl_library" option add *TreeCtrl.useTheme 1 #option add *TreeCtrl.itemPrefix item #option add *TreeCtrl.ColumnPrefix col # Resizing columns can be done in realtime or by displaying a proxy line switch -- $::thisPlatform { macosx - unix { option add *TreeCtrl.columnResizeMode realtime } windows { if {$::tcl_platform(os) eq "Windows NT"} { option add *TreeCtrl.columnResizeMode realtime } } } # Demo sources foreach file { biglist bitmaps column-lock explorer firefox help imovie layout mailwasher mycomputer outlook-folders outlook-newgroup random span textvariable www-options } { source [Path $file.tcl] } # Get default colors set w [listbox .listbox] set SystemButtonFace [$w cget -highlightbackground] set SystemHighlight [$w cget -selectbackground] set SystemHighlightText [$w cget -selectforeground] destroy $w if {$thisPlatform == "unix"} { # I hate that gray selection color set SystemHighlight #316ac5 set SystemHighlightText White } proc MakeMenuBar {} { set m [menu .menubar] . configure -menu $m set m2 [menu $m.mFile -tearoff no] if {$::thisPlatform ne "unix" && [info commands console] ne ""} { console eval { wm title . "TkTreeCtrl Console" .console configure -font {Courier 9} -height 8 # ::tk::ConsolePrompt wm geometry . +0-100 } $m2 add command -label "Console" -command { if {[console eval {winfo ismapped .}]} { console hide } else { console show } } } $m2 add command -label "Event Browser" -command ToggleEventsWindow $m2 add command -label "Identify" -command ToggleIdentifyWindow $m2 add command -label "Style Editor" -command ToggleStyleEditorWindow $m2 add command -label "View Source" -command ToggleSourceWindow $m2 add command -label "Magnifier" -command ToggleLoupeWindow $m2 add command -label "Quit" -command exit $m add cascade -label "File" -menu $m2 if {$::tile} { set m2 [menu $m.mTheme -tearoff no] $m add cascade -label "Theme" -menu $m2 foreach theme [lsort -dictionary [ttk::style theme names]] { $m2 add radiobutton -label $theme -command [list ttk::setTheme $theme] \ -variable ::DemoTheme -value $theme } $m2 add separator $m2 add command -label "Inspector" -command ToggleThemeWindow } return } proc MakeEventsWindow {} { set w [toplevel .events] wm withdraw $w # wm transient $w . wm title $w "TkTreeCtrl Events" set m [menu $w.menubar] $w configure -menu $m set m1 [menu $m.m1 -tearoff 0] $m1 add cascade -label "Static" -menu [menu $m1.m1 -tearoff 0] $m1 add cascade -label "Dynamic" -menu [menu $m1.m2 -tearoff 0] $m1 add command -label "Clear Window" -command "$w.f.t item delete all" \ -accelerator Ctrl+X $m1 add command -label "Rebuild Menus" -command "RebuildEventsMenus $w.f.t $m" $m add cascade -label "Events" -menu $m1 bind $w "$w.f.t item delete all" TreePlusScrollbarsInAFrame $w.f 1 1 pack $w.f -expand yes -fill both set T $w.f.t $T configure -showheader no -showroot no -showrootlines no -height 300 $T column create -tags C0 $T configure -treecolumn C0 $T element create e1 text -fill [list $::SystemHighlightText {selected focus}] $T element create e2 text -fill [list $::SystemHighlightText {selected focus}] $T element create e3 rect -fill [list $::SystemHighlight {selected focus} gray {selected !focus}] \ -showfocus yes $T element create e4 rect -fill blue -width 100 -height 2 set S [$T style create s1] $T style elements $S {e3 e1} $T style layout $S e3 -union [list e1] -ipadx 1 -ipady {0 1} set S [$T style create s2] $T style elements $S {e3 e1 e2} $T style layout $S e1 -width 20 -sticky w $T style layout $S e3 -union [list e1 e2] -ipadx 1 -ipady {0 1} set S [$T style create s3] $T style elements $S {e4} $T column configure C0 -itemstyle s1 RebuildEventsMenus $T $m wm protocol $w WM_DELETE_WINDOW "ToggleEventsWindow" switch -- $::thisPlatform { macintosh - macosx { wm geometry $w -40+40 } default { wm geometry $w -0+0 } } return } proc RebuildEventsMenus {T m} { foreach event [lsort -dictionary [.f2.f1.t notify eventnames]] { set details [lsort -dictionary [.f2.f1.t notify detailnames $event]] foreach detail $details { set pattern <$event-$detail> set linkage [.f2.f1.t notify linkage $pattern] lappend patterns $pattern $linkage lappend patterns2($linkage) $pattern } if {![llength $details]} { set pattern <$event> set linkage [.f2.f1.t notify linkage $pattern] lappend patterns $pattern $linkage lappend patterns2($linkage) $pattern } } $m.m1.m1 delete 0 end $m.m1.m2 delete 0 end set menu(static) $m.m1.m1 set menu(dynamic) $m.m1.m2 foreach {pattern linkage} $patterns { if {![info exists ::EventTrack($pattern)]} { set ::EventTrack($pattern) 1 } $menu($linkage) add checkbutton -label $pattern \ -variable EventTrack($pattern) \ -command [list ToggleEvent $T $pattern] } foreach linkage {static dynamic} { $menu($linkage) add separator $menu($linkage) add command -label "Toggle All" \ -command [list ToggleEvents $T $patterns2($linkage)] } set ::Events {} set ::EventsId "" foreach {pattern linkage} $patterns { .f2.f1.t notify bind $T $pattern { lappend Events %? if {$EventsId eq ""} { set EventsId [after idle [list RecordEvents %W]] } } } return } proc RecordEvents {T} { set ::EventsId "" set events $::Events set ::Events {} if {![winfo ismapped .events]} return if {[$T item numchildren root] > 2000} { set N [expr {[$T item numchildren root] - 2000}] $T item delete "root firstchild" "root child $N" } if {0 && [$T item count] > 1} { set I [$T item create] $T item style set $I 0 s3 $T item lastchild root $I } set open 1 if {[llength $events] > 50} { set open 0 } foreach list $events { RecordEvent $T $list $open } $T see "last visible" return } proc RecordEvent {T list open} { set I [$T item create -open $open] array set map $list $T item text $I C0 $map(P) $T item lastchild root $I foreach {char value} $list { if {[string first $char "TWPed"] != -1} continue set I2 [$T item create] $T item style set $I2 C0 s2 $T item element configure $I2 C0 e1 -text $char + e2 -text $value $T item lastchild $I $I2 $T item configure $I -button yes } return } proc ToggleEventsWindow {} { set w .events if {![winfo exists $w]} { MakeEventsWindow } if {[winfo ismapped $w]} { wm withdraw $w } else { wm deiconify $w } return } proc ToggleEvent {T pattern} { .f2.f1.t notify configure $T $pattern -active $::EventTrack($pattern) return } proc ToggleEvents {T patterns} { foreach pattern $patterns { set ::EventTrack($pattern) [expr {!$::EventTrack($pattern)}] ToggleEvent $T $pattern } return } proc MakeIdentifyWindow {} { set w .identify toplevel $w wm withdraw $w wm title $w "TkTreeCtrl Identify" set wText $w.text text $wText -state disabled -width 50 -height 2 -font [.f2.f1.t cget -font] $wText tag configure tagBold -font "[.f2.f1.t cget -font] bold" pack $wText -expand yes -fill both wm protocol $w WM_DELETE_WINDOW "ToggleIdentifyWindow" return } proc UpdateIdentifyWindow {T x y} { set w .identify if {![winfo exists $w]} return set wText $w.text $wText configure -state normal $wText delete 1.0 end $wText insert end x= tagBold "$x " {} y= tagBold $y\n foreach {key val} [$T identify $x $y] { $wText insert end $key tagBold " $val " } $wText configure -state disabled return } proc ToggleIdentifyWindow {} { set w .identify if {![winfo exists $w]} { MakeIdentifyWindow } if {[winfo ismapped $w]} { wm withdraw $w } else { wm deiconify $w } return } proc MakeSourceWindow {} { set w [toplevel .source] wm withdraw $w # wm transient $w . set f [frame $w.f -borderwidth 0] switch -- $::thisPlatform { macintosh - macosx { set font {Geneva 9} } unix { set font {Courier -12} } default { set font {Courier 9} } } text $f.t -font $font -tabs [font measure $font 1234] -wrap none \ -yscrollcommand "$f.sv set" -xscrollcommand "$f.sh set" $::scrollbarCmd $f.sv -orient vertical -command "$f.t yview" $::scrollbarCmd $f.sh -orient horizontal -command "$f.t xview" pack $f -expand yes -fill both grid columnconfigure $f 0 -weight 1 grid rowconfigure $f 0 -weight 1 grid configure $f.t -row 0 -column 0 -sticky news grid configure $f.sh -row 1 -column 0 -sticky we grid configure $f.sv -row 0 -column 1 -sticky ns wm protocol $w WM_DELETE_WINDOW "ToggleSourceWindow" switch -- $::thisPlatform { macintosh - macosx { wm geometry $w -40+40 } default { wm geometry $w -0+0 } } return } proc ShowSource {file} { wm title .source "TkTreeCtrl Source: $file" set path [Path $file] set t .source.f.t set chan [open $path] $t delete 1.0 end $t insert end [read $chan] $t mark set insert 1.0 close $chan return } proc ToggleSourceWindow {} { set w .source if {[winfo ismapped $w]} { wm withdraw $w } else { wm deiconify $w } return } proc ToggleStyleEditorWindow {} { set w .styleEditor if {![winfo exists $w]} { source [Path style-editor.tcl] StyleEditor::Init .f2.f1.t StyleEditor::SetListOfStyles } elseif {[winfo ismapped $w]} { wm withdraw $w } else { wm deiconify $w StyleEditor::SetListOfStyles } return } proc MakeThemeWindow {} { set w [toplevel .theme] wm withdraw $w # wm transient $w . wm title $w "TkTreeCtrl Themes" set m [menu $w.menubar] $w configure -menu $m set m1 [menu $m.m1 -tearoff 0] $m1 add command -label "Set List" -command SetThemeWindow $m add cascade -label "Theme" -menu $m1 TreePlusScrollbarsInAFrame $w.f 1 1 pack $w.f -expand yes -fill both set T $w.f.t $T configure -showheader no -showroot no -showrootlines no -height 300 $T column create -tags C0 $T configure -treecolumn C0 $T element create e1 text -fill [list $::SystemHighlightText {selected focus}] $T element create e3 rect -fill [list $::SystemHighlight {selected focus} gray {selected !focus}] \ -showfocus yes set S [$T style create s1] $T style elements $S {e3 e1} $T style layout $S e3 -union [list e1] -ipadx 1 -ipady {0 1} $T column configure C0 -itemstyle s1 SetThemeWindow wm protocol $w WM_DELETE_WINDOW "ToggleThemeWindow" return } proc ToggleThemeWindow {} { set w .theme if {![winfo exists $w]} { MakeThemeWindow } if {[winfo ismapped $w]} { wm withdraw $w } else { wm deiconify $w } return } proc SetThemeWindow {} { set w .theme set T $w.f.t $T item delete all # # Themes # foreach theme [lsort -dictionary [ttk::style theme names]] { set I [$T item create -button yes -open no -tags theme -parent root] $T item text $I C0 $theme ttk::style theme settings $theme { set I2 [$T item create -button yes -open no -parent $I] $T item text $I2 C0 ELEMENTS # # Elements # foreach element [lsort -dictionary [ttk::style element names]] { # # Element options # set options [ttk::style element options $element] set I3 [$T item create -button [llength $options] -open no -tags element -parent $I2] $T item text $I3 C0 $element foreach option [lsort -dictionary $options] { set I4 [$T item create -open no -tags {element option} -parent $I3] $T item text $I4 C0 $option } } # # Styles # set I2 [$T item create -button yes -open no -parent $I] $T item text $I2 C0 STYLES set styles [list "."] ; # [ttk::style names] please! foreach style [lsort -dictionary $styles] { # # Style options # set cfg [ttk::style configure $style] set I3 [$T item create -button [llength $cfg] -open no -tags style -parent $I2] $T item text $I3 C0 $style foreach {option value} $cfg { set I4 [$T item create -open no -tags {style option} -parent $I3] $T item text $I4 C0 "$option $value" } } } } return } MakeSourceWindow MakeMenuBar # http://wiki.tcl.tk/950 proc sbset {sb first last} { if {$first <= 0 && $last >= 1} { grid remove $sb } else { grid $sb } $sb set $first $last return } proc TreePlusScrollbarsInAFrame {f h v} { if {$::tileFull} { frame $f -borderwidth 0 } else { frame $f -borderwidth 1 -relief sunken } switch -- $::thisPlatform { macintosh { set font {Geneva 9} } macosx { set font {{Lucida Grande} 11} } unix { set font {Helvetica -12} } default { # There is a bug on my Win98 box with Tk_MeasureChars() and # MS Sans Serif 8. set font {{MS Sans} 8} } } treectrl $f.t -highlightthickness 0 -borderwidth 0 -font $font $f.t configure -xscrollincrement 20 # $f.t configure -itemprefix item# -columnprefix column# $f.t debug configure -enable no -display yes -erasecolor pink \ -drawcolor orange -displaydelay 30 -textlayout 0 -data 0 if {$h} { $::scrollbarCmd $f.sh -orient horizontal -command "$f.t xview" # $f.t configure -xscrollcommand "$f.sh set" $f.t notify bind $f.sh { sbset %W %l %u } bind $f.sh "focus $f.t" } if {$v} { $::scrollbarCmd $f.sv -orient vertical -command "$f.t yview" # $f.t configure -yscrollcommand "$f.sv set" $f.t notify bind $f.sv { sbset %W %l %u } bind $f.sv "focus $f.t" } grid columnconfigure $f 0 -weight 1 grid rowconfigure $f 0 -weight 1 grid configure $f.t -row 0 -column 0 -sticky news if {$h} { grid configure $f.sh -row 1 -column 0 -sticky we } if {$v} { grid configure $f.sv -row 0 -column 1 -sticky ns } bind $f.t { TreeCtrl::MarqueeBegin %W %x %y set DebugExpose(x1) %x set DebugExpose(y1) %y break } bind $f.t { TreeCtrl::MarqueeUpdate %W %x %y set DebugExpose(x2) %x set DebugExpose(y2) %y break } bind $f.t { TreeCtrl::MarqueeEnd %W %x %y %W debug expose $DebugExpose(x1) $DebugExpose(y1) $DebugExpose(x2) $DebugExpose(y2) break } MakeListPopup $f.t MakeHeaderPopup $f.t bind $f.t { ShowPopup %W %x %y %X %Y } if {[tk windowingsystem] eq "aqua"} { bind $f.t { ShowPopup %W %x %y %X %Y } } return } proc MakeMainWindow {} { wm title . "TkTreeCtrl Demo" switch -- $::thisPlatform { macintosh - macosx { wm geometry . +40+40 set ::ShowLines 0 } default { wm geometry . +0+0 set ::ShowLines 1 } } panedwindow .pw2 -orient horizontal -borderwidth 0 panedwindow .pw1 -orient vertical -borderwidth 0 # Tree + scrollbar: demos TreePlusScrollbarsInAFrame .f1 1 1 .f1.t configure -showbuttons no -showlines no -showroot no -height 100 .f1.t column create -text "List of Demos" -expand yes -button no -tags C0 .f1.t configure -treecolumn C0 # Tree + scrollbar: styles + elements in list TreePlusScrollbarsInAFrame .f4 1 1 .f4.t configure -showlines $::ShowLines -showroot no -height 140 .f4.t column create -text "Elements and Styles" -expand yes -button no -tags C0 .f4.t configure -treecolumn C0 # Tree + scrollbar: styles + elements in selected item TreePlusScrollbarsInAFrame .f3 1 1 .f3.t configure -showlines $::ShowLines -showroot no .f3.t column create -text "Styles in Item" -expand yes -button no -tags C0 .f3.t configure -treecolumn C0 .pw1 add .f1 .f4 .f3 -height 150 # Frame on right frame .f2 # Tree + scrollbars TreePlusScrollbarsInAFrame .f2.f1 1 1 .f2.f1.t configure -indent 19 # Give it a big border to debug drawing if {!$::tileFull} { .f2.f1.t configure -borderwidth 6 -relief ridge -highlightthickness 3 } grid columnconfigure .f2 0 -weight 1 grid rowconfigure .f2 0 -weight 1 grid configure .f2.f1 -row 0 -column 0 -sticky news -pady 0 # Window to display result of "T identify" bind TagIdentify { if {"%W" ne ".f2.f1.t"} { set x [expr {%X - [winfo rootx .f2.f1.t]}] set y [expr {%Y - [winfo rooty .f2.f1.t]}] } else { set x %x set y %y } UpdateIdentifyWindow .f2.f1.t $x $y } AddBindTag .f2.f1.t TagIdentify .pw2 add .pw1 -width 200 .pw2 add .f2 -width 450 pack .pw2 -expand yes -fill both ### # A treectrl widget can generate the following built-in events: # called when the active item changes # called before an item is closed # called after an item is closed # called before an item is opened # called after an item is opened # called before items are deleted # called when horizontal scroll position changes # called when vertical scroll position changes # called when items are added to or removed from the selection # # The application programmer can define custom events to be # generated by the "notify generate" command. The following events # are generated by the library scripts. .f2.f1.t notify install .f2.f1.t notify install .f2.f1.t notify install .f2.f1.t notify install .f2.f1.t notify install .f2.f1.t notify install .f2.f1.t notify install .f2.f1.t notify install .f2.f1.t notify install .f2.f1.t notify install ### return } proc MakeListPopup {T} { set m [menu $T.mTree -tearoff no] set m2 [menu $m.mCollapse -tearoff no] $m add cascade -label Collapse -menu $m2 set m2 [menu $m.mExpand -tearoff no] $m add cascade -label Expand -menu $m2 set m2 [menu $m.mBgImg -tearoff no] $m2 add radiobutton -label none -variable Popup(bgimg) -value none \ -command {$Popup(T) configure -backgroundimage ""} $m2 add radiobutton -label sky -variable Popup(bgimg) -value sky \ -command {$Popup(T) configure -backgroundimage $Popup(bgimg)} $m add cascade -label "Background Image" -menu $m2 set m2 [menu $m.mBgMode -tearoff no] foreach value {column order ordervisible row} { $m2 add radiobutton -label $value -variable Popup(bgmode) -value $value \ -command {$Popup(T) configure -backgroundmode $Popup(bgmode)} } $m add cascade -label "Background Mode" -menu $m2 set m2 [menu $m.mVisible -tearoff no] $m add cascade -label Columns -menu $m2 set m2 [menu $m.mColumnResizeMode -tearoff no] $m2 add radiobutton -label proxy -variable Popup(columnresizemode) -value proxy \ -command {$Popup(T) configure -columnresizemode $Popup(columnresizemode)} $m2 add radiobutton -label realtime -variable Popup(columnresizemode) -value realtime \ -command {$Popup(T) configure -columnresizemode $Popup(columnresizemode)} $m add cascade -label "Column Resize Mode" -menu $m2 set m2 [menu $m.mDebug -tearoff no] $m2 add checkbutton -label Data -variable Popup(debug,data) \ -command {$Popup(T) debug configure -data $Popup(debug,data)} $m2 add checkbutton -label Display -variable Popup(debug,display) \ -command {$Popup(T) debug configure -display $Popup(debug,display)} $m2 add checkbutton -label "Text Layout" -variable Popup(debug,textlayout) \ -command {$Popup(T) debug configure -textlayout $Popup(debug,textlayout)} $m2 add separator set m3 [menu $m2.mDelay -tearoff no] foreach n {10 20 30 40 50 60 70 80 90 100} { $m3 add radiobutton -label $n -variable Popup(debug,displaydelay) -value $n \ -command {$Popup(T) debug configure -displaydelay $Popup(debug,displaydelay)} } $m2 add cascade -label "Display Delay" -menu $m3 $m2 add separator $m2 add checkbutton -label Enable -variable Popup(debug,enable) \ -command {$Popup(T) debug configure -enable $Popup(debug,enable)} $m add cascade -label Debug -menu $m2 set m2 [menu $m.mBuffer -tearoff no] $m2 add radiobutton -label "none" -variable Popup(doublebuffer) -value none \ -command {$Popup(T) configure -doublebuffer $Popup(doublebuffer)} $m2 add radiobutton -label "item" -variable Popup(doublebuffer) -value item \ -command {$Popup(T) configure -doublebuffer $Popup(doublebuffer)} $m2 add radiobutton -label "window" -variable Popup(doublebuffer) -value window \ -command {$Popup(T) configure -doublebuffer $Popup(doublebuffer)} $m add cascade -label Buffering -menu $m2 set m2 [menu $m.mLineStyle -tearoff no] $m2 add radiobutton -label "dot" -variable Popup(linestyle) -value dot \ -command {$Popup(T) configure -linestyle $Popup(linestyle)} $m2 add radiobutton -label "solid" -variable Popup(linestyle) -value solid \ -command {$Popup(T) configure -linestyle $Popup(linestyle)} $m add cascade -label "Line style" -menu $m2 set m2 [menu $m.mOrient -tearoff no] $m2 add radiobutton -label "Horizontal" -variable Popup(orient) -value horizontal \ -command {$Popup(T) configure -orient $Popup(orient)} $m2 add radiobutton -label "Vertical" -variable Popup(orient) -value vertical \ -command {$Popup(T) configure -orient $Popup(orient)} $m add cascade -label Orient -menu $m2 set m2 [menu $m.mSelectMode -tearoff no] foreach mode [list browse extended multiple single] { $m2 add radiobutton -label $mode -variable Popup(selectmode) -value $mode \ -command {$Popup(T) configure -selectmode $Popup(selectmode)} } $m add cascade -label Selectmode -menu $m2 set m2 [menu $m.mShow -tearoff no] $m2 add checkbutton -label "Buttons" -variable Popup(showbuttons) \ -command {$Popup(T) configure -showbuttons $Popup(showbuttons)} $m2 add checkbutton -label "Header" -variable Popup(showheader) \ -command {$Popup(T) configure -showheader $Popup(showheader)} $m2 add checkbutton -label "Lines" -variable Popup(showlines) \ -command {$Popup(T) configure -showlines $Popup(showlines)} $m2 add checkbutton -label "Root Lines" -variable Popup(showrootlines) \ -command {$Popup(T) configure -showrootlines $Popup(showrootlines)} $m2 add checkbutton -label "Root" -variable Popup(showroot) \ -command {$Popup(T) configure -showroot $Popup(showroot)} $m2 add checkbutton -label "Root Button" -variable Popup(showrootbutton) \ -command {$Popup(T) configure -showrootbutton $Popup(showrootbutton)} $m add cascade -label Show -menu $m2 set m2 [menu $m.mSpan -tearoff no] $m add cascade -label Span -menu $m2 $m add checkbutton -label "Use Theme" -variable Popup(usetheme) \ -command {$Popup(T) configure -usetheme $Popup(usetheme)} return } proc MakeHeaderPopup {T} { set m [menu $T.mHeader -tearoff no] set m2 [menu $m.mArrow -tearoff no] $m add cascade -label Arrow -menu $m2 $m2 add radiobutton -label "None" -variable Popup(arrow) -value none \ -command {$Popup(T) column configure $Popup(column) -arrow none} $m2 add radiobutton -label "Up" -variable Popup(arrow) -value up \ -command {$Popup(T) column configure $Popup(column) -arrow up} $m2 add radiobutton -label "Down" -variable Popup(arrow) -value down \ -command {$Popup(T) column configure $Popup(column) -arrow down} $m2 add separator $m2 add radiobutton -label "Side Left" -variable Popup(arrow,side) -value left \ -command {$Popup(T) column configure $Popup(column) -arrowside left} $m2 add radiobutton -label "Side Right" -variable Popup(arrow,side) -value right \ -command {$Popup(T) column configure $Popup(column) -arrowside right} $m2 add separator $m2 add radiobutton -label "Gravity Left" -variable Popup(arrow,gravity) -value left \ -command {$Popup(T) column configure $Popup(column) -arrowgravity left} $m2 add radiobutton -label "Gravity Right" -variable Popup(arrow,gravity) -value right \ -command {$Popup(T) column configure $Popup(column) -arrowgravity right} $m add checkbutton -label "Button" -variable Popup(button) \ -command {$Popup(T) column configure $Popup(column) -button $Popup(button)} $m add checkbutton -label "Expand" -variable Popup(expand) \ -command {$Popup(T) column configure $Popup(column) -expand $Popup(expand)} set m2 [menu $m.mJustify -tearoff no] $m add cascade -label Justify -menu $m2 $m2 add radiobutton -label "Left" -variable Popup(justify) -value left \ -command {$Popup(T) column configure $Popup(column) -justify left} $m2 add radiobutton -label "Center" -variable Popup(justify) -value center \ -command {$Popup(T) column configure $Popup(column) -justify center} $m2 add radiobutton -label "Right" -variable Popup(justify) -value right \ -command {$Popup(T) column configure $Popup(column) -justify right} set m2 [menu $m.mLock -tearoff no] $m add cascade -label Lock -menu $m2 $m2 add radiobutton -label "Left" -variable Popup(lock) -value left \ -command {$Popup(T) column configure $Popup(column) -lock left} $m2 add radiobutton -label "None" -variable Popup(lock) -value none \ -command {$Popup(T) column configure $Popup(column) -lock none} $m2 add radiobutton -label "Right" -variable Popup(lock) -value right \ -command {$Popup(T) column configure $Popup(column) -lock right} $m add checkbutton -label "Resize" -variable Popup(resize) \ -command {$Popup(T) column configure $Popup(column) -resize $Popup(resize)} $m add checkbutton -label "Squeeze" -variable Popup(squeeze) \ -command {$Popup(T) column configure $Popup(column) -squeeze $Popup(squeeze)} $m add checkbutton -label "Tree Column" -variable Popup(treecolumn) \ -command {$Popup(T) configure -treecolumn [expr {$Popup(treecolumn) ? $Popup(column) : ""}]} return } proc AddBindTag {w tag} { if {[lsearch -exact [bindtags $w] $tag] == -1} { bindtags $w [concat [bindtags $w] $tag] } foreach child [winfo children $w] { AddBindTag $child $tag } return } MakeMainWindow InitPics sky proc ShowPopup {T x y X Y} { global Popup set Popup(T) $T set id [$T identify $x $y] if {$id ne ""} { if {[lindex $id 0] eq "header"} { set Popup(column) [lindex $id 1] set Popup(arrow) [$T column cget $Popup(column) -arrow] set Popup(arrow,side) [$T column cget $Popup(column) -arrowside] set Popup(arrow,gravity) [$T column cget $Popup(column) -arrowgravity] set Popup(button) [$T column cget $Popup(column) -button] set Popup(expand) [$T column cget $Popup(column) -expand] set Popup(resize) [$T column cget $Popup(column) -resize] set Popup(squeeze) [$T column cget $Popup(column) -squeeze] set Popup(justify) [$T column cget $Popup(column) -justify] set Popup(lock) [$T column cget $Popup(column) -lock] set Popup(treecolumn) [expr {[$T column id tree] eq $Popup(column)}] tk_popup $T.mHeader $X $Y return } } set menu $T.mTree set m $menu.mCollapse $m delete 0 end $m add command -label "All" -command {$Popup(T) item collapse all} if {$id ne ""} { if {[lindex $id 0] eq "item"} { set item [lindex $id 1] $m add command -label "Item $item" -command "$T item collapse $item" $m add command -label "Item $item (recurse)" -command "$T item collapse $item -recurse" } } set m $menu.mExpand $m delete 0 end $m add command -label "All" -command {$Popup(T) item expand all} if {$id ne ""} { if {[lindex $id 0] eq "item"} { set item [lindex $id 1] $m add command -label "Item $item" -command "$T item expand $item" $m add command -label "Item $item (recurse)" -command "$T item expand $item -recurse" } } foreach option {data display displaydelay enable textlayout} { set Popup(debug,$option) [$T debug cget -$option] } set Popup(bgimg) [$T cget -backgroundimage] if {$Popup(bgimg) eq ""} { set Popup(bgimg) none } set Popup(bgmode) [$T cget -backgroundmode] set Popup(columnresizemode) [$T cget -columnresizemode] set Popup(doublebuffer) [$T cget -doublebuffer] set Popup(linestyle) [$T cget -linestyle] set Popup(orient) [$T cget -orient] set Popup(selectmode) [$T cget -selectmode] set Popup(showbuttons) [$T cget -showbuttons] set Popup(showheader) [$T cget -showheader] set Popup(showlines) [$T cget -showlines] set Popup(showrootlines) [$T cget -showrootlines] set Popup(showroot) [$T cget -showroot] set Popup(showrootbutton) [$T cget -showrootbutton] set m $menu.mVisible $m delete 0 end foreach C [$T column list] { set break [expr {!([$T column order $C] % 20)}] set Popup(visible,$C) [$T column cget $C -visible] $m add checkbutton -label "Column $C \"[$T column cget $C -text]\" \[[$T column cget $C -image]\]" -variable Popup(visible,$C) \ -command "$T column configure $C -visible \$Popup(visible,$C)" \ -columnbreak $break } set m $menu.mSpan $m delete 0 end if {[llength $id] >= 4 && [lindex $id 2] eq "column"} { set item [lindex $id 1] set column [lindex $id 3] for {set i 1} {$i <= [$T column count] - [$T column order $column]} {incr i} { set break [expr {!($i % 20)}] $m add radiobutton -label $i -command "$T item span $item $column $i" \ -variable Popup(span) -value $i -columnbreak $break } set Popup(span) [$T item span $item $column] } else { $m add command -label "no item column" -state disabled } set Popup(usetheme) [$T cget -usetheme] tk_popup $menu $X $Y return } # Allow "scan" bindings if {$::thisPlatform eq "windows"} { bind .f2.f1.t { } } # # List of demos # proc InitDemoList {} { global DemoCmd global DemoFile set t .f1.t $t element create e1 text -fill [list $::SystemHighlightText {selected focus}] $t element create e2 rect -fill [list $::SystemHighlight {selected focus} gray {selected !focus}] \ -showfocus yes $t style create s1 $t style elements s1 {e2 e1} # Tk listbox has linespace + 1 height $t style layout s1 e2 -union [list e1] -ipadx 2 -ipady {0 1} -iexpand e $t column configure C0 -itemstyle s1 # "Picture Catalog" DemoPictureCatalog # "Picture Catalog 2" DemoPictureCatalog2 # "Folder Contents (Vertical)" DemoExplorerFilesV foreach {label command file} [list \ "Random $::RandomN Items" DemoRandom random.tcl \ "Random $::RandomN Items, Button Images" DemoRandom2 random.tcl \ "Outlook Express (Folders)" DemoOutlookFolders outlook-folders.tcl \ "Outlook Express (Newsgroup)" DemoOutlookNewsgroup outlook-newgroup.tcl \ "Explorer (Details)" DemoExplorerDetails explorer.tcl \ "Explorer (List)" DemoExplorerList explorer.tcl \ "Explorer (Large icons)" DemoExplorerLargeIcons explorer.tcl \ "Explorer (Small icons)" DemoExplorerSmallIcons explorer.tcl \ "Internet Options" DemoInternetOptions www-options.tcl \ "Help Contents" DemoHelpContents help.tcl \ "Layout" DemoLayout layout.tcl \ "MailWasher" DemoMailWasher mailwasher.tcl \ "Bitmaps" DemoBitmaps bitmaps.tcl \ "iMovie" DemoIMovie imovie.tcl \ "Firefox Privacy" DemoFirefoxPrivacy firefox.tcl \ "Textvariable" DemoTextvariable textvariable.tcl \ "Big List" DemoBigList biglist.tcl \ "Column Spanning" DemoSpan span.tcl \ "My Computer" DemoMyComputer mycomputer.tcl \ "Column Locking" DemoColumnLock column-lock.tcl ] { set item [$t item create] $t item lastchild root $item # $t item style set $item C0 s1 $t item text $item C0 $label set DemoCmd($item) $command set DemoFile($item) $file } $t yview moveto 0.0 return } InitDemoList proc ClicksToSeconds {clicks} { return [format "%.2g" [expr {$clicks / 1000000.0}]] } proc DemoSet {cmd file} { DemoClear set clicks [clock clicks] uplevel #0 $cmd set clicks [expr {[clock clicks] - $clicks}] dbwin "set list in [ClicksToSeconds $clicks] seconds ($clicks clicks)\n" .f2.f1.t xview moveto 0 .f2.f1.t yview moveto 0 update DisplayStylesInList ShowSource $file catch { if {[winfo ismapped .styleEditor]} { StyleEditor::SetListOfStyles } } AddBindTag .f2.f1.t TagIdentify return } .f1.t notify bind .f1.t { if {%c == 1} { set selection [%T selection get] set item [lindex $selection 0] DemoSet $DemoCmd($item) $DemoFile($item) } } proc DisplayStylesInList {} { set T .f2.f1.t set t .f4.t # Create elements and styles the first time this is called if {[llength [$t style names]] == 0} { $t element create e1 text -fill [list $::SystemHighlightText {selected focus}] $t element create e2 text -fill [list $::SystemHighlightText {selected focus} "" {selected !focus} blue {}] $t element create e3 rect -fill [list $::SystemHighlight {selected focus} gray {selected !focus}] \ -showfocus yes $t style create s1 $t style elements s1 {e3 e1} $t style layout s1 e3 -union [list e1] -ipadx 1 -ipady {0 1} $t style create s2 $t style elements s2 {e3 e1 e2} $t style layout s2 e1 -padx {0 4} $t style layout s2 e3 -union [list e1 e2] -ipadx 1 -ipady {0 1} } # Clear the list $t item delete all # One item for each element in the demo list foreach elem [lsort -dictionary [$T element names]] { set item [$t item create -button yes -open no] $t item style set $item C0 s1 $t item text $item C0 "Element $elem ([$T element type $elem])" # One item for each configuration option for this element foreach list [$T element configure $elem] { foreach {name x y default current} $list {} set item2 [$t item create] if {[string equal $default $current]} { $t item style set $item2 C0 s1 $t item element configure $item2 C0 e1 -text [list $name $current] } else { $t item style set $item2 C0 s2 $t item element configure $item2 C0 e1 -text $name + e2 -text [list $current] } $t item lastchild $item $item2 } $t item lastchild root $item } # One item for each style in the demo list foreach style [lsort -dictionary [$T style names]] { set item [$t item create -button yes -open no] $t item style set $item C0 s1 $t item text $item C0 "Style $style" # One item for each element in the style foreach elem [$T style elements $style] { set item2 [$t item create -button yes -open no] $t item style set $item2 C0 s1 $t item text $item2 C0 "Element $elem ([$T element type $elem])" # One item for each layout option for this element in this style foreach {option value} [$T style layout $style $elem] { set item3 [$t item create] # $t item hasbutton $item3 no $t item style set $item3 C0 s1 $t item text $item3 C0 [list $option $value] $t item lastchild $item2 $item3 } $t item lastchild $item $item2 } $t item lastchild root $item } $t xview moveto 0 $t yview moveto 0 return } proc DisplayStylesInItem {item} { set T .f2.f1.t set t .f3.t $t column configure C0 -text "Styles in item [$T item id $item]" # Create elements and styles the first time this is called if {[llength [$t style names]] == 0} { $t element create e1 text -fill [list $::SystemHighlightText {selected focus}] $t element create e2 text -fill [list $::SystemHighlightText {selected focus} "" {selected !focus} blue {}] $t element create e3 rect -fill [list $::SystemHighlight {selected focus} gray {selected !focus}] \ -showfocus yes $t style create s1 $t style elements s1 {e3 e1} $t style layout s1 e3 -union [list e1] -ipadx {1 2} -ipady {0 1} $t style create s2 $t style elements s2 {e3 e1 e2} $t style layout s2 e1 -padx {0 4} $t style layout s2 e3 -union [list e1 e2] -ipadx 1 -ipady {0 1} } # Clear the list $t item delete all # One item for each item-column foreach style [$T item style set $item] column [$T column list] { set item2 [$t item create -open no] $t item style set $item2 C0 s1 if {$style ne ""} { $t item element configure $item2 C0 e1 \ -text "Column $column: Style $style" } else { $t item element configure $item2 C0 e1 \ -text "Column $column: no style" } # One item for each element in this style if {[string length $style]} { set button 0 foreach elem [$T item style elements $item $column] { set button 1 set item3 [$t item create -button yes -open no] $t item style set $item3 C0 s1 $t item element configure $item3 C0 e1 \ -text "Element $elem ([$T element type $elem])" # One item for each configuration option in this element foreach list [$T item element configure $item $column $elem] { foreach {name x y default current} $list {} set item4 [$t item create] set masterDefault [$T element cget $elem $name] set sameAsMaster [string equal $masterDefault $current] if {!$sameAsMaster && ![string length $current]} { set sameAsMaster 1 set current $masterDefault } if {$sameAsMaster} { $t item style set $item4 C0 s1 $t item element configure $item4 C0 e1 -text "$name [list $current]" } else { $t item style set $item4 C0 s2 $t item element configure $item4 C0 e1 -text $name + e2 -text [list $current] } $t item lastchild $item3 $item4 } $t item lastchild $item2 $item3 } if {$button} { $t item configure $item2 -button yes } } $t item lastchild root $item2 } $t xview moveto 0 $t yview moveto 0 return } # When one item is selected in the demo list, display the styles in that item. # See DemoClear for why the tag "DontDelete" is used .f2.f1.t notify bind DontDelete { if {%c == 1} { set selection [%T selection get] DisplayStylesInItem [lindex $selection 0] } } # Move columns when ColumnDrag-receive is generated # See DemoClear for why the tag "DontDelete" is used .f2.f1.t notify bind DontDelete { %T column move %C %b } proc DemoClear {} { set T .f2.f1.t # Clear the demo list $T item delete all # Clear all bindings on the demo list added by the previous demo. # This is why DontDelete is used for some bindings (see above). $T notify unbind $T # Clear all run-time states eval $T state undefine [$T state names] # Clear the styles-in-item list .f3.t item delete all # Delete columns in demo list $T column delete all # Delete all styles in demo list eval $T style delete [$T style names] # Delete all elements in demo list eval $T element delete [$T element names] # Delete -window windows foreach child [winfo children $T] { if {[string equal $child $T.mTree] || [string equal $child $T.mHeader]} continue destroy $child } $T item configure root -button no $T item expand root # Restore some happy defaults to the demo list $T configure -orient vertical -wrap "" -xscrollincrement 0 \ -yscrollincrement 0 -itemheight 0 -showheader yes \ -background white -scrollmargin 0 -xscrolldelay 50 -yscrolldelay 50 \ -buttonbitmap "" -buttonimage "" -backgroundmode row \ -indent 19 -defaultstyle {} -backgroundimage "" \ -showrootlines yes -minitemheight 0 -borderwidth [expr {$::tileFull ? 0 : 6}] \ -highlightthickness [expr {$::tileFull ? 0 : 3}] -usetheme yes -cursor {} \ -itemwidth 0 -itemwidthequal no -itemwidthmultiple 0 \ -font [.f4.t cget -font] # Undo "column configure all" in a demo $T column configure tail -background \ [lindex [$T column configure tail -background] 3] # Enable drag-and-drop column reordering. This also requires the # event be installed. $T column dragconfigure -enable yes # Restore default bindings to the demo list bindtags $T [list $T TreeCtrl [winfo toplevel $T] all] catch {destroy $T.entry} catch {destroy $T.text} return } # # Demo: Picture catalog # proc DemoPictureCatalog {} { set T .f2.f1.t $T configure -showroot no -showbuttons no -showlines no \ -selectmode multiple -orient horizontal -wrap window \ -yscrollincrement 50 -showheader no $T element create elemTxt text -fill {SystemHighlightText {selected focus}} $T element create elemSelTxt rect -fill {SystemHighlight {selected focus}} $T element create elemSelImg rect -outline {SystemHighlight {selected focus}} \ -outlinewidth 4 $T element create elemImg rect -fill gray -width 80 -height 120 set S [$T style create STYLE -orient vertical] $T style elements $S {elemSelImg elemImg elemSelTxt elemTxt} $T style layout $S elemSelImg -union elemImg -ipadx 6 -ipady 6 $T style layout $S elemSelTxt -union elemTxt $T style layout $S elemImg -pady {0 6} for {set i 1} {$i <= 10} {incr i} { set I [$T item create] $T item style set $I 0 $S $T item text $I 0 "Picture #$i" $T item lastchild root $I } return } # # Demo: Picture catalog # proc DemoPictureCatalog2 {} { set T .f2.f1.t $T configure -showroot no -showbuttons no -showlines no \ -selectmode multiple -orient horizontal -wrap window \ -yscrollincrement 50 -showheader no $T element create elemTxt text -fill {SystemHighlightText {selected focus}} \ -justify left -wrap word -lines 2 $T element create elemSelTxt rect -fill {SystemHighlight {selected focus}} $T element create elemSelImg rect -outline {SystemHighlight {selected focus}} \ -outlinewidth 4 $T element create elemImg rect -fill gray set S [$T style create STYLE -orient vertical] $T style elements $S {elemSelImg elemImg elemSelTxt elemTxt} $T style layout $S elemSelImg -union elemImg \ -ipadx 6 -ipady 6 $T style layout $S elemSelTxt -union elemTxt $T style layout $S elemImg -pady {0 6} $T style layout $S elemImg -expand n $T style layout $S elemTxt -expand s for {set i 1} {$i <= 10} {incr i} { set I [$T item create] $T item style set $I 0 $S $T item text $I 0 "This is\nPicture\n#$i" $T item element configure $I 0 elemImg -width [expr int(20 + rand() * 80)] \ -height [expr int(20 + rand() * 120)] $T item lastchild root $I } return } proc CursorWindow {} { set w .cursors if {[winfo exists $w]} { destroy $w } toplevel $w set c [canvas $w.canvas -background white -width [expr {50 * 10}] \ -highlightthickness 0 -borderwidth 0] pack $c -expand yes -fill both set cursors { X_cursor arrow based_arrow_down based_arrow_up boat bogosity bottom_left_corner bottom_right_corner bottom_side bottom_tee box_spiral center_ptr circle clock coffee_mug cross cross_reverse crosshair diamond_cross dot dotbox double_arrow draft_large draft_small draped_box exchange fleur gobbler gumby hand1 hand2 heart icon iron_cross left_ptr left_side left_tee leftbutton ll_angle lr_angle man middlebutton mouse pencil pirate plus question_arrow right_ptr right_side right_tee rightbutton rtl_logo sailboat sb_down_arrow sb_h_double_arrow sb_left_arrow sb_right_arrow sb_up_arrow sb_v_double_arrow shuttle sizing spider spraycan star target tcross top_left_arrow top_left_corner top_right_corner top_side top_tee trek ul_angle umbrella ur_angle watch xterm } set col 0 set row 0 foreach cursor $cursors { set x [expr {$col * 50}] set y [expr {$row * 40}] $c create rectangle $x $y [expr {$x + 50}] [expr {$y + 40}] \ -fill gray90 -outline black -width 2 -tags $cursor.rect $c create text [expr {$x + 50 / 2}] [expr {$y + 4}] -text $cursor \ -anchor n -width 42 -tags $cursor.text if {[incr col] == 10} { set col 0 incr row } $c bind $cursor.rect " $c configure -cursor $cursor $c itemconfigure $cursor.rect -fill linen " $c bind $cursor.rect " $c configure -cursor {} $c itemconfigure $cursor.rect -fill gray90 " $c bind $cursor.text " $c configure -cursor $cursor " $c bind $cursor.text " $c configure -cursor {} " } $c configure -height [expr {($row + 1) * 40}] return } # A little screen magnifier if {[llength [info commands loupe]]} { set Loupe(zoom) 2 set Loupe(x) 0 set Loupe(y) 0 set Loupe(auto) 1 set Loupe(afterId) "" proc LoupeAfter {} { global Loupe set x [winfo pointerx .] set y [winfo pointery .] if {$Loupe(auto) || ($Loupe(x) != $x) || ($Loupe(y) != $y)} { set w [image width $Loupe(image)] set h [image height $Loupe(image)] loupe $Loupe(image) $x $y $w $h $::Loupe(zoom) set Loupe(x) $x set Loupe(y) $y } set Loupe(afterId) [after $Loupe(delay) LoupeAfter] return } proc MakeLoupeWindow {} { global Loupe set w [toplevel .loupe] wm title $w "TreeCtrl Magnifier" wm withdraw $w wm geometry $w -0+0 image create photo ImageLoupe -width 280 -height 150 pack [label $w.label -image ImageLoupe -borderwidth 1 -relief sunken] \ -expand yes -fill both set f [frame $w.zoom -borderwidth 0] radiobutton $f.r1 -text "1x" -variable ::Loupe(zoom) -value 1 radiobutton $f.r2 -text "2x" -variable ::Loupe(zoom) -value 2 radiobutton $f.r4 -text "4x" -variable ::Loupe(zoom) -value 4 radiobutton $f.r8 -text "8x" -variable ::Loupe(zoom) -value 8 pack $f.r1 $f.r2 $f.r4 $f.r8 -side left pack $f -side bottom -anchor e # Resize the image with the window bind LoupeWindow { set w [expr {%w - 2}] set h [expr {%h - 2}] if {$w != [$Loupe(image) cget -width] || $h != [$Loupe(image) cget -height]} { $Loupe(image) configure -width $w -height $h loupe $Loupe(image) $Loupe(x) $Loupe(y) $w $h $Loupe(zoom) } } bindtags $w.label [concat [bindtags .loupe] LoupeWindow] wm protocol $w WM_DELETE_WINDOW "ToggleLoupeWindow" set Loupe(image) ImageLoupe set Loupe(delay) 500 return } proc ToggleLoupeWindow {} { global Loupe set w .loupe if {![winfo exists $w]} { MakeLoupeWindow } if {[winfo ismapped $w]} { after cancel $Loupe(afterId) wm withdraw $w } else { LoupeAfter wm deiconify $w } return } }