# tkfbox.tcl --
#
# Implements the "TK" standard file selection dialog box. This dialog
# box is used on the Unix platforms whenever the tk_strictMotif flag is
# not set.
#
# The "TK" standard file selection dialog box is similar to the file
# selection dialog box on Win95(TM). The user can navigate the
# directories by clicking on the folder icons or by selecting the
# "Directory" option menu. The user can select files by clicking on the
# file icons or by entering a filename in the "Filename:" entry.
#
# Copyright © 1994-1998 Sun Microsystems, Inc.
#
# See the file "license.terms" for information on usage and redistribution
# of this file, and for a DISCLAIMER OF ALL WARRANTIES.
#
namespace eval ::tk::dialog {}
namespace eval ::tk::dialog::file {
namespace import -force ::tk::msgcat::*
variable showHiddenBtn 0
variable showHiddenVar 1
# Based on Vimix/16/actions/go-up.svg
# See https://github.com/vinceliuice/vimix-icon-theme
variable updirImageData {
}
proc UpdateUpdirImageData {} {
variable updirImageData
set idx1 [string first "#000000" $updirImageData]
set idx2 [expr {$idx1 + 6}]
set fgColor [ttk::style lookup . -foreground {} black]
lassign [winfo rgb . $fgColor] r g b
set fgColor [format "#%02x%02x%02x" \
[expr {$r >> 8}] [expr {$g >> 8}] [expr {$b >> 8}]]
return [string replace $updirImageData $idx1 $idx2 $fgColor]
}
# Based on https://icons8.com/icon/JXYalxb9XWWd/folder
variable folderImageData {
}
# Based on https://icons8.com/icon/mEF_vyjYlnE3/file
variable fileImageData {
}
# Create the images if they did not already exist.
if {![info exists ::tk::Priv(updirImage)] ||
$::tk::Priv(updirImage) ni [image names]} {
set ::tk::Priv(updirImage) [image create photo \
-format $::tk::svgFmt -data [UpdateUpdirImageData]]
if {"TkFileDialog" ni [bindtags .]} {
bindtags . [linsert [bindtags .] 1 TkFileDialog]
bind TkFileDialog <> {
if {$::tk::Priv(updirImage) in [image names]} {
$::tk::Priv(updirImage) configure \
-data [::tk::dialog::file::UpdateUpdirImageData]
}
}
}
}
if {![info exists ::tk::Priv(folderImage)] ||
$::tk::Priv(folderImage) ni [image names]} {
set ::tk::Priv(folderImage) [image create photo \
-format $::tk::svgFmt -data $folderImageData]
}
if {![info exists ::tk::Priv(fileImage)] ||
$::tk::Priv(fileImage) ni [image names]} {
set ::tk::Priv(fileImage) [image create photo \
-format $::tk::svgFmt -data $fileImageData]
}
}
# ::tk::dialog::file:: --
#
# Implements the TK file selection dialog. This dialog is used when the
# tk_strictMotif flag is set to false. This procedure shouldn't be
# called directly. Call tk_getOpenFile or tk_getSaveFile instead.
#
# Arguments:
# type "open" or "save"
# args Options parsed by the procedure.
#
proc ::tk::dialog::file:: {type args} {
variable ::tk::Priv
variable showHiddenBtn
set dataName __tk_filedialog
upvar ::tk::dialog::file::$dataName data
Config $dataName $type $args
if {$data(-parent) eq "."} {
set w .$dataName
} else {
set w $data(-parent).$dataName
}
# (re)create the dialog box if necessary
#
if {![winfo exists $w]} {
Create $w TkFDialog
} elseif {[winfo class $w] ne "TkFDialog"} {
destroy $w
Create $w TkFDialog
} else {
set data(dirMenuBtn) $w.contents.f1.menu
set data(dirMenu) $w.contents.f1.menu.menu
set data(upBtn) $w.contents.f1.up
set data(icons) $w.contents.icons
set data(ent) $w.contents.f2.ent
set data(typeMenuLab) $w.contents.f2.lab2
set data(typeMenuBtn) $w.contents.f2.menu
set data(typeMenu) $data(typeMenuBtn).m
set data(okBtn) $w.contents.f2.ok
set data(cancelBtn) $w.contents.f2.cancel
set data(hiddenBtn) $w.contents.f2.hidden
SetSelectMode $w $data(-multiple)
}
if {$showHiddenBtn} {
$data(hiddenBtn) configure -state normal
grid $data(hiddenBtn)
} else {
$data(hiddenBtn) configure -state disabled
grid remove $data(hiddenBtn)
}
# Make sure subseqent uses of this dialog are independent [Bug 845189]
unset -nocomplain data(extUsed)
# Dialog boxes should be transient with respect to their parent, so that
# they will always stay on top of their parent window. However, some
# window managers will create the window as withdrawn if the parent window
# is withdrawn or iconified. Combined with the grab we put on the window,
# this can hang the entire application. Therefore we only make the dialog
# transient if the parent is viewable.
if {[winfo viewable [winfo toplevel $data(-parent)]]} {
wm transient $w $data(-parent)
}
# Add traces on the selectPath variable
#
trace add variable data(selectPath) write \
[list ::tk::dialog::file::SetPath $w]
$data(dirMenuBtn) configure \
-textvariable ::tk::dialog::file::${dataName}(selectPath)
# Cleanup previous menu
#
$data(typeMenu) delete 0 end
$data(typeMenuBtn) configure -state normal -text ""
# Initialize the file types menu
#
if {[llength $data(-filetypes)]} {
# Default type and name to first entry
set initialtype [lindex $data(-filetypes) 0]
set initialTypeName [lindex $initialtype 0]
if {$data(-typevariable) ne ""} {
upvar #0 $data(-typevariable) typeVariable
if {[info exists typeVariable]} {
set initialTypeName $typeVariable
}
}
foreach type $data(-filetypes) {
set title [lindex $type 0]
set filter [lindex $type 1]
$data(typeMenu) add command -label $title \
-command [list ::tk::dialog::file::SetFilter $w $type]
# [string first] avoids glob-pattern char issues
if {[string first ${initialTypeName} $title] == 0} {
set initialtype $type
}
}
SetFilter $w $initialtype
$data(typeMenuBtn) configure -state normal
$data(typeMenuLab) configure -state normal
} else {
set data(filter) "*"
$data(typeMenuBtn) configure -state disabled -takefocus 0
$data(typeMenuLab) configure -state disabled
}
UpdateWhenIdle $w
# Withdraw the window, then update all the geometry information
# so we know how big it wants to be, then center the window in the
# display (Motif style) and de-iconify it.
::tk::PlaceWindow $w widget $data(-parent)
wm title $w $data(-title)
# Set a grab and claim the focus too.
::tk::SetFocusGrab $w $data(ent)
$data(ent) delete 0 end
$data(ent) insert 0 $data(selectFile)
$data(ent) selection range 0 end
$data(ent) icursor end
# Wait for the user to respond, then restore the focus and return the
# index of the selected button. Restore the focus before deleting the
# window, since otherwise the window manager may take the focus away so we
# can't redirect it. Finally, restore any grab that was in effect.
vwait ::tk::Priv(selectFilePath)
::tk::RestoreFocusGrab $w $data(ent) withdraw
# Cleanup traces on selectPath variable
#
foreach trace [trace info variable data(selectPath)] {
trace remove variable data(selectPath) {*}$trace
}
if {[winfo exists $data(dirMenuBtn)]} {
$data(dirMenuBtn) configure -textvariable {}
}
return $Priv(selectFilePath)
}
# ::tk::dialog::file::Config --
#
# Configures the TK filedialog according to the argument list
#
proc ::tk::dialog::file::Config {dataName type argList} {
upvar ::tk::dialog::file::$dataName data
set data(type) $type
# 0: Delete all variable that were set on data(selectPath) the
# last time the file dialog is used. The traces may cause troubles
# if the dialog is now used with a different -parent option.
foreach trace [trace info variable data(selectPath)] {
trace remove variable data(selectPath) {*}$trace
}
# 1: the configuration specs
#
set specs {
{-defaultextension "" "" ""}
{-filetypes "" "" ""}
{-initialdir "" "" ""}
{-initialfile "" "" ""}
{-parent "" "" "."}
{-title "" "" ""}
{-typevariable "" "" ""}
}
# The "-multiple" option is only available for the "open" file dialog.
#
if {$type eq "open"} {
lappend specs {-multiple "" "" "0"}
}
# The "-confirmoverwrite" option is only for the "save" file dialog.
#
if {$type eq "save"} {
lappend specs {-confirmoverwrite "" "" "1"}
}
# 2: default values depending on the type of the dialog
#
if {![info exists data(selectPath)]} {
# first time the dialog has been popped up
set data(selectPath) [pwd]
set data(selectFile) ""
}
# 3: parse the arguments
#
tclParseConfigSpec ::tk::dialog::file::$dataName $specs "" $argList
if {$data(-title) eq ""} {
if {$type eq "open"} {
set data(-title) [mc "Open"]
} else {
set data(-title) [mc "Save As"]
}
}
# 4: set the default directory and selection according to the -initial
# settings
#
if {$data(-initialdir) ne ""} {
# Ensure that initialdir is an absolute path name.
if {[file isdirectory $data(-initialdir)]} {
set old [pwd]
cd $data(-initialdir)
set data(selectPath) [pwd]
cd $old
} else {
set data(selectPath) [pwd]
}
}
set data(selectFile) $data(-initialfile)
# 5. Parse the -filetypes option
#
set data(origfiletypes) $data(-filetypes)
set data(-filetypes) [::tk::FDGetFileTypes $data(-filetypes)]
if {![winfo exists $data(-parent)]} {
return -code error -errorcode [list TK LOOKUP WINDOW $data(-parent)] \
"bad window path name \"$data(-parent)\""
}
# Set -multiple to a one or zero value (not other boolean types like
# "yes") so we can use it in tests more easily.
if {$type eq "save"} {
set data(-multiple) 0
} elseif {$data(-multiple)} {
set data(-multiple) 1
} else {
set data(-multiple) 0
}
}
proc ::tk::dialog::file::Create {w class} {
set dataName [lindex [split $w .] end]
upvar ::tk::dialog::file::$dataName data
variable ::tk::Priv
global tk_library
toplevel $w -class $class
if {[tk windowingsystem] eq "x11"} {wm attributes $w -type dialog}
pack [ttk::frame $w.contents] -expand 1 -fill both
#set w $w.contents
# f1: the frame with the directory option menu
#
set f1 [ttk::frame $w.contents.f1]
bind [::tk::AmpWidget ttk::label $f1.lab -text [mc "&Directory:"]] \
<> [list focus $f1.menu]
set data(dirMenuBtn) $f1.menu
if {![info exists data(selectPath)]} {
set data(selectPath) ""
}
set data(dirMenu) $f1.menu.menu
ttk::menubutton $f1.menu -menu $data(dirMenu) -direction flush \
-textvariable [format %s(selectPath) ::tk::dialog::file::$dataName]
menu $data(dirMenu) -tearoff 0
$data(dirMenu) add radiobutton -label "" -variable \
[format %s(selectPath) ::tk::dialog::file::$dataName]
set data(upBtn) [ttk::button $f1.up]
$data(upBtn) configure -image $Priv(updirImage)
$f1.menu configure -takefocus 1;# -highlightthickness 2
pack $data(upBtn) -side right -padx 3p -fill both
pack $f1.lab -side left -padx 3p -fill both
pack $f1.menu -expand yes -fill both -padx 3p
# data(icons): the IconList that list the files and directories.
#
if {$class eq "TkFDialog"} {
if { $data(-multiple) } {
set fNameCaption [mc "File &names:"]
} else {
set fNameCaption [mc "File &name:"]
}
set fTypeCaption [mc "Files of &type:"]
set iconListCommand [list ::tk::dialog::file::OkCmd $w]
} else {
set fNameCaption [mc "&Selection:"]
set iconListCommand [list ::tk::dialog::file::chooseDir::DblClick $w]
}
set data(icons) [::tk::IconList $w.contents.icons \
-command $iconListCommand -multiple $data(-multiple)]
bind $data(icons) <> \
[list ::tk::dialog::file::ListBrowse $w]
# f2: the frame with the OK button, cancel button, "file name" field
# and file types field.
#
set f2 [ttk::frame $w.contents.f2]
bind [::tk::AmpWidget ttk::label $f2.lab -text $fNameCaption -anchor e]\
<> [list focus $f2.ent]
# -pady 0
set data(ent) [ttk::entry $f2.ent]
# The font to use for the icons. The default Canvas font on Unix is just
# deviant.
set ::tk::$w.contents.icons(font) [$data(ent) cget -font]
# Make the file types bits only if this is a File Dialog
if {$class eq "TkFDialog"} {
set data(typeMenuLab) [::tk::AmpWidget ttk::label $f2.lab2 \
-text $fTypeCaption -anchor e]
# -pady [$f2.lab cget -pady]
set data(typeMenuBtn) [ttk::menubutton $f2.menu \
-menu $f2.menu.m]
# -indicatoron 1
set data(typeMenu) [menu $data(typeMenuBtn).m -tearoff 0]
# $data(typeMenuBtn) configure -takefocus 1 -relief raised -anchor w
bind $data(typeMenuLab) <> [list \
focus $data(typeMenuBtn)]
}
# The hidden button is displayed when ::tk::dialog::file::showHiddenBtn is
# true. Create it disabled so the binding doesn't trigger if it isn't
# shown.
if {$class eq "TkFDialog"} {
set text [mc "Show &Hidden Files and Directories"]
} else {
set text [mc "Show &Hidden Directories"]
}
set data(hiddenBtn) [::tk::AmpWidget ttk::checkbutton $f2.hidden \
-text $text -state disabled \
-variable ::tk::dialog::file::showHiddenVar \
-command [list ::tk::dialog::file::UpdateWhenIdle $w]]
# -anchor w -padx 2p
# the okBtn is created after the typeMenu so that the keyboard traversal
# is in the right order, and add binding so that we find out when the
# dialog is destroyed by the user (added here instead of to the overall
# window so no confusion about how much gets called; exactly
# once will do). [Bug 987169]
set data(okBtn) [::tk::AmpWidget ttk::button $f2.ok \
-text [mc "&OK"] -default active];# -pady 2p]
bind $data(okBtn) [list ::tk::dialog::file::Destroyed $w]
set data(cancelBtn) [::tk::AmpWidget ttk::button $f2.cancel \
-text [mc "&Cancel"] -default normal];# -pady 2p]
# grid the widgets in f2
#
grid $f2.lab $f2.ent $data(okBtn) -padx 3p -pady 2p -sticky ew
grid configure $f2.ent -padx 1.5p
if {$class eq "TkFDialog"} {
grid $data(typeMenuLab) $data(typeMenuBtn) $data(cancelBtn) \
-padx 3p -sticky ew
grid configure $data(typeMenuBtn) -padx 0
grid $data(hiddenBtn) -columnspan 2 -padx 3p -sticky ew
} else {
grid $data(hiddenBtn) - $data(cancelBtn) -padx 3p -sticky ew
}
grid columnconfigure $f2 1 -weight 1
# Pack all the frames together. We are done with widget construction.
#
pack $f1 -side top -fill x -pady 3p
pack $f2 -side bottom -pady 3p -fill x
pack $data(icons) -expand yes -fill both -padx 3p -pady 1p
# Set up the event handlers that are common to Directory and File Dialogs
#
wm protocol $w WM_DELETE_WINDOW [list ::tk::dialog::file::CancelCmd $w]
$data(upBtn) configure -command [list ::tk::dialog::file::UpDirCmd $w]
$data(cancelBtn) configure -command [list ::tk::dialog::file::CancelCmd $w]
bind $w [list $data(cancelBtn) invoke]
bind $w [list tk::AltKeyInDialog $w %A]
# Set up event handlers specific to File or Directory Dialogs
#
if {$class eq "TkFDialog"} {
bind $data(ent) [list ::tk::dialog::file::ActivateEnt $w]
$data(okBtn) configure -command [list ::tk::dialog::file::OkCmd $w]
bind $w [format {
if {[%s cget -state] eq "normal"} {
focus %s
}
} $data(typeMenuBtn) $data(typeMenuBtn)]
} else {
set okCmd [list ::tk::dialog::file::chooseDir::OkCmd $w]
bind $data(ent) $okCmd
$data(okBtn) configure -command $okCmd
bind $w [list focus $data(ent)]
bind $w [list $data(okBtn) invoke]
}
bind $w [list $data(hiddenBtn) invoke]
bind $data(ent) [list ::tk::dialog::file::CompleteEnt $w]
# Build the focus group for all the entries
#
::tk::FocusGroup_Create $w
::tk::FocusGroup_BindIn $w $data(ent) [list \
::tk::dialog::file::EntFocusIn $w]
::tk::FocusGroup_BindOut $w $data(ent) [list \
::tk::dialog::file::EntFocusOut $w]
}
# ::tk::dialog::file::SetSelectMode --
#
# Set the select mode of the dialog to single select or multi-select.
#
# Arguments:
# w The dialog path.
# multi 1 if the dialog is multi-select; 0 otherwise.
#
# Results:
# None.
proc ::tk::dialog::file::SetSelectMode {w multi} {
set dataName __tk_filedialog
upvar ::tk::dialog::file::$dataName data
if { $multi } {
set fNameCaption [mc "File &names:"]
} else {
set fNameCaption [mc "File &name:"]
}
set iconListCommand [list ::tk::dialog::file::OkCmd $w]
::tk::SetAmpText $w.contents.f2.lab $fNameCaption
$data(icons) configure -multiple $multi -command $iconListCommand
return
}
# ::tk::dialog::file::UpdateWhenIdle --
#
# Creates an idle event handler which updates the dialog in idle time.
# This is important because loading the directory may take a long time
# and we don't want to load the same directory for multiple times due to
# multiple concurrent events.
#
proc ::tk::dialog::file::UpdateWhenIdle {w} {
upvar ::tk::dialog::file::[winfo name $w] data
if {[info exists data(updateId)]} {
return
}
set data(updateId) [after idle [list ::tk::dialog::file::Update $w]]
}
# ::tk::dialog::file::Update --
#
# Loads the files and directories into the IconList widget. Also sets up
# the directory option menu for quick access to parent directories.
#
proc ::tk::dialog::file::Update {w} {
# This proc may be called within an idle handler. Make sure that the
# window has not been destroyed before this proc is called
if {![winfo exists $w]} {
return
}
set class [winfo class $w]
if {($class ne "TkFDialog") && ($class ne "TkChooseDir")} {
return
}
set dataName [winfo name $w]
upvar ::tk::dialog::file::$dataName data
variable ::tk::Priv
variable showHiddenVar
global tk_library
unset -nocomplain data(updateId)
set folder $Priv(folderImage)
set file $Priv(fileImage)
set appPWD [pwd]
if {[catch {
cd $data(selectPath)
}]} then {
# We cannot change directory to $data(selectPath). $data(selectPath)
# should have been checked before ::tk::dialog::file::Update is
# called, so we normally won't come to here. Anyways, give an error
# and abort action.
tk_messageBox -type ok -parent $w -icon warning -message [mc \
"Cannot change to the directory \"%1\$s\".\nPermission denied."\
$data(selectPath)]
cd $appPWD
return
}
# Turn on the busy cursor. BUG?? We haven't disabled X events, though,
# so the user may still click and cause havoc ...
#
set entCursor [$data(ent) cget -cursor]
set dlgCursor [$w cget -cursor]
$data(ent) configure -cursor watch
$w configure -cursor watch
update idletasks
$data(icons) deleteall
set showHidden $showHiddenVar
# Make the dir list. Note that using an explicit [pwd] (instead of '.') is
# better in some VFS cases.
$data(icons) add $folder [GlobFiltered [pwd] d 1]
if {$class eq "TkFDialog"} {
# Make the file list if this is a File Dialog, selecting all but
# 'd'irectory type files.
#
$data(icons) add $file [GlobFiltered [pwd] {f b c l p s}]
}
# Update the Directory: option menu
#
set list ""
set dir ""
foreach subdir [file split $data(selectPath)] {
set dir [file join $dir $subdir]
lappend list $dir
}
$data(dirMenu) delete 0 end
set var [format %s(selectPath) ::tk::dialog::file::$dataName]
foreach path $list {
$data(dirMenu) add command -label $path -command [list set $var $path]
}
# Restore the PWD to the application's PWD
#
cd $appPWD
if {$class eq "TkFDialog"} {
# Restore the Open/Save Button if this is a File Dialog
#
if {$data(type) eq "open"} {
::tk::SetAmpText $data(okBtn) [mc "&Open"]
} else {
::tk::SetAmpText $data(okBtn) [mc "&Save"]
}
}
# turn off the busy cursor.
#
$data(ent) configure -cursor $entCursor
$w configure -cursor $dlgCursor
}
# ::tk::dialog::file::SetPathSilently --
#
# Sets data(selectPath) without invoking the trace procedure
#
proc ::tk::dialog::file::SetPathSilently {w path} {
upvar ::tk::dialog::file::[winfo name $w] data
set cb [list ::tk::dialog::file::SetPath $w]
trace remove variable data(selectPath) write $cb
set data(selectPath) $path
trace add variable data(selectPath) write $cb
}
# This proc gets called whenever data(selectPath) is set
#
proc ::tk::dialog::file::SetPath {w name1 name2 op} {
if {[winfo exists $w]} {
upvar ::tk::dialog::file::[winfo name $w] data
UpdateWhenIdle $w
# On directory dialogs, we keep the entry in sync with the currentdir.
if {[winfo class $w] eq "TkChooseDir"} {
$data(ent) delete 0 end
$data(ent) insert end $data(selectPath)
}
}
}
# This proc gets called whenever data(filter) is set
#
proc ::tk::dialog::file::SetFilter {w type} {
upvar ::tk::dialog::file::[winfo name $w] data
set data(filterType) $type
set data(filter) [lindex $type 1]
$data(typeMenuBtn) configure -text [lindex $type 0] ;#-indicatoron 1
# If we aren't using a default extension, use the one suppled by the
# filter.
if {![info exists data(extUsed)]} {
if {[string length $data(-defaultextension)]} {
set data(extUsed) 1
} else {
set data(extUsed) 0
}
}
if {!$data(extUsed)} {
# Get the first extension in the list that matches {^\*\.\w+$} and
# remove all * from the filter.
set index [lsearch -regexp $data(filter) {^\*\.\w+$}]
if {$index >= 0} {
set data(-defaultextension) \
[string trimleft [lindex $data(filter) $index] "*"]
} else {
# Couldn't find anything! Reset to a safe default...
set data(-defaultextension) ""
}
}
$data(icons) see 0
UpdateWhenIdle $w
}
# tk::dialog::file::ResolveFile --
#
# Interpret the user's text input in a file selection dialog. Performs:
#
# (1) ~ substitution
# (2) resolve all instances of . and ..
# (3) check for non-existent files/directories
# (4) check for chdir permissions
# (5) conversion of environment variable references to their
# contents (once only)
#
# Arguments:
# context: the current directory you are in
# text: the text entered by the user
# defaultext: the default extension to add to files with no extension
# expandEnv: whether to expand environment variables (yes by default)
#
# Return vaue:
# [list $flag $directory $file]
#
# flag = OK : valid input
# = PATTERN : valid directory/pattern
# = PATH : the directory does not exist
# = FILE : the directory exists by the file doesn't exist
# = CHDIR : Cannot change to the directory
# = ERROR : Invalid entry
#
# directory : valid only if flag = OK or PATTERN or FILE
# file : valid only if flag = OK or PATTERN
#
# directory may not be the same as context, because text may contain a
# subdirectory name
#
proc ::tk::dialog::file::ResolveFile {context text defaultext {expandEnv 1}} {
set appPWD [pwd]
set path [JoinFile $context $text]
# If the file has no extension, append the default. Be careful not to do
# this for directories, otherwise typing a dirname in the box will give
# back "dirname.extension" instead of trying to change dir.
if {
![file isdirectory $path] && ([file ext $path] eq "") &&
![string match {$*} [file tail $path]]
} then {
set path "$path$defaultext"
}
if {[catch {file exists $path}]} {
# This "if" block can be safely removed if the following code stop
# generating errors.
#
# file exists ~nonsuchuser
#
return [list ERROR $path ""]
}
if {[file exists $path]} {
if {[file isdirectory $path]} {
if {[catch {cd $path}]} {
return [list CHDIR $path ""]
}
set directory [pwd]
set file ""
set flag OK
cd $appPWD
} else {
if {[catch {cd [file dirname $path]}]} {
return [list CHDIR [file dirname $path] ""]
}
set directory [pwd]
set file [file tail $path]
set flag OK
cd $appPWD
}
} else {
set dirname [file dirname $path]
if {[file exists $dirname]} {
if {[catch {cd $dirname}]} {
return [list CHDIR $dirname ""]
}
set directory [pwd]
cd $appPWD
set file [file tail $path]
# It's nothing else, so check to see if it is an env-reference
if {$expandEnv && [string match {$*} $file]} {
set var [string range $file 1 end]
if {[info exist ::env($var)]} {
return [ResolveFile $context $::env($var) $defaultext 0]
}
}
if {[regexp {[*?]} $file]} {
set flag PATTERN
} else {
set flag FILE
}
} else {
set directory $dirname
set file [file tail $path]
set flag PATH
# It's nothing else, so check to see if it is an env-reference
if {$expandEnv && [string match {$*} $file]} {
set var [string range $file 1 end]
if {[info exist ::env($var)]} {
return [ResolveFile $context $::env($var) $defaultext 0]
}
}
}
}
return [list $flag $directory $file]
}
# Gets called when the entry box gets keyboard focus. We clear the selection
# from the icon list . This way the user can be certain that the input in the
# entry box is the selection.
#
proc ::tk::dialog::file::EntFocusIn {w} {
upvar ::tk::dialog::file::[winfo name $w] data
if {[$data(ent) get] ne ""} {
$data(ent) selection range 0 end
$data(ent) icursor end
} else {
$data(ent) selection clear
}
if {[winfo class $w] eq "TkFDialog"} {
# If this is a File Dialog, make sure the buttons are labeled right.
if {$data(type) eq "open"} {
::tk::SetAmpText $data(okBtn) [mc "&Open"]
} else {
::tk::SetAmpText $data(okBtn) [mc "&Save"]
}
}
}
proc ::tk::dialog::file::EntFocusOut {w} {
upvar ::tk::dialog::file::[winfo name $w] data
$data(ent) selection clear
}
# Gets called when user presses Return in the "File name" entry.
#
proc ::tk::dialog::file::ActivateEnt {w} {
upvar ::tk::dialog::file::[winfo name $w] data
set text [$data(ent) get]
if {$data(-multiple)} {
foreach t $text {
VerifyFileName $w $t
}
} else {
VerifyFileName $w $text
}
}
# Verification procedure
#
proc ::tk::dialog::file::VerifyFileName {w filename} {
upvar ::tk::dialog::file::[winfo name $w] data
set list [ResolveFile $data(selectPath) $filename $data(-defaultextension)]
foreach {flag path file} $list {
break
}
switch -- $flag {
OK {
if {$file eq ""} {
# user has entered an existing (sub)directory
set data(selectPath) $path
$data(ent) delete 0 end
} else {
SetPathSilently $w $path
if {$data(-multiple)} {
lappend data(selectFile) $file
} else {
set data(selectFile) $file
}
Done $w
}
}
PATTERN {
set data(selectPath) $path
set data(filter) $file
}
FILE {
if {$data(type) eq "open"} {
tk_messageBox -icon warning -type ok -parent $w \
-message [mc "File \"%1\$s\" does not exist." \
[file join $path $file]]
$data(ent) selection range 0 end
$data(ent) icursor end
} else {
SetPathSilently $w $path
if {$data(-multiple)} {
lappend data(selectFile) $file
} else {
set data(selectFile) $file
}
Done $w
}
}
PATH {
tk_messageBox -icon warning -type ok -parent $w -message \
[mc "Directory \"%1\$s\" does not exist." $path]
$data(ent) selection range 0 end
$data(ent) icursor end
}
CHDIR {
tk_messageBox -type ok -parent $w -icon warning -message \
[mc "Cannot change to the directory\
\"%1\$s\".\nPermission denied." $path]
$data(ent) selection range 0 end
$data(ent) icursor end
}
ERROR {
tk_messageBox -type ok -parent $w -icon warning -message \
[mc "Invalid file name \"%1\$s\"." $path]
$data(ent) selection range 0 end
$data(ent) icursor end
}
}
}
# Gets called when user presses the Alt-s or Alt-o keys.
#
proc ::tk::dialog::file::InvokeBtn {w key} {
upvar ::tk::dialog::file::[winfo name $w] data
if {[$data(okBtn) cget -text] eq $key} {
$data(okBtn) invoke
}
}
# Gets called when user presses the "parent directory" button
#
proc ::tk::dialog::file::UpDirCmd {w} {
upvar ::tk::dialog::file::[winfo name $w] data
if {$data(selectPath) ne "/"} {
set data(selectPath) [file dirname $data(selectPath)]
}
}
# Join a file name to a path name. The "file join" command will break if the
# filename begins with ~
#
proc ::tk::dialog::file::JoinFile {path file} {
if {[string match {~*} $file] && [file exists $path/$file]} {
return [file join $path ./$file]
} else {
return [file join $path $file]
}
}
# Gets called when user presses the "OK" button
#
proc ::tk::dialog::file::OkCmd {w} {
upvar ::tk::dialog::file::[winfo name $w] data
set filenames {}
foreach item [$data(icons) selection get] {
lappend filenames [$data(icons) get $item]
}
if {
([llength $filenames] && !$data(-multiple)) ||
($data(-multiple) && ([llength $filenames] == 1))
} then {
set filename [lindex $filenames 0]
set file [JoinFile $data(selectPath) $filename]
if {[file isdirectory $file]} {
ListInvoke $w [list $filename]
return
}
}
ActivateEnt $w
}
# Gets called when user presses the "Cancel" button
#
proc ::tk::dialog::file::CancelCmd {w} {
upvar ::tk::dialog::file::[winfo name $w] data
variable ::tk::Priv
bind $data(okBtn) {}
set Priv(selectFilePath) ""
}
# Gets called when user destroys the dialog directly [Bug 987169]
#
proc ::tk::dialog::file::Destroyed {w} {
upvar ::tk::dialog::file::[winfo name $w] data
variable ::tk::Priv
set Priv(selectFilePath) ""
}
# Gets called when user browses the IconList widget (dragging mouse, arrow
# keys, etc)
#
proc ::tk::dialog::file::ListBrowse {w} {
upvar ::tk::dialog::file::[winfo name $w] data
set text {}
foreach item [$data(icons) selection get] {
lappend text [$data(icons) get $item]
}
if {[llength $text] == 0} {
return
}
if {$data(-multiple)} {
set newtext {}
foreach file $text {
set fullfile [JoinFile $data(selectPath) $file]
if { ![file isdirectory $fullfile] } {
lappend newtext $file
}
}
set text $newtext
set isDir 0
} else {
set text [lindex $text 0]
set file [JoinFile $data(selectPath) $text]
set isDir [file isdirectory $file]
}
if {!$isDir} {
$data(ent) delete 0 end
$data(ent) insert 0 $text
if {[winfo class $w] eq "TkFDialog"} {
if {$data(type) eq "open"} {
::tk::SetAmpText $data(okBtn) [mc "&Open"]
} else {
::tk::SetAmpText $data(okBtn) [mc "&Save"]
}
}
} elseif {[winfo class $w] eq "TkFDialog"} {
::tk::SetAmpText $data(okBtn) [mc "&Open"]
}
}
# Gets called when user invokes the IconList widget (double-click, Return key,
# etc)
#
proc ::tk::dialog::file::ListInvoke {w filenames} {
upvar ::tk::dialog::file::[winfo name $w] data
if {[llength $filenames] == 0} {
return
}
set file [JoinFile $data(selectPath) [lindex $filenames 0]]
set class [winfo class $w]
if {$class eq "TkChooseDir" || [file isdirectory $file]} {
set appPWD [pwd]
if {[catch {cd $file}]} {
tk_messageBox -type ok -parent $w -icon warning -message \
[mc "Cannot change to the directory \"%1\$s\".\nPermission denied." $file]
} else {
cd $appPWD
set data(selectPath) $file
}
} else {
if {$data(-multiple)} {
set data(selectFile) $filenames
} else {
set data(selectFile) $file
}
Done $w
}
}
# ::tk::dialog::file::Done --
#
# Gets called when user has input a valid filename. Pops up a dialog
# box to confirm selection when necessary. Sets the
# tk::Priv(selectFilePath) variable, which will break the "vwait" loop
# in ::tk::dialog::file:: and return the selected filename to the script
# that calls tk_getOpenFile or tk_getSaveFile
#
proc ::tk::dialog::file::Done {w {selectFilePath ""}} {
upvar ::tk::dialog::file::[winfo name $w] data
variable ::tk::Priv
if {$selectFilePath eq ""} {
if {$data(-multiple)} {
set selectFilePath {}
foreach f $data(selectFile) {
lappend selectFilePath [JoinFile $data(selectPath) $f]
}
} else {
set selectFilePath [JoinFile $data(selectPath) $data(selectFile)]
}
set Priv(selectFile) $data(selectFile)
set Priv(selectPath) $data(selectPath)
if {($data(type) eq "save") && $data(-confirmoverwrite) && [file exists $selectFilePath]} {
set reply [tk_messageBox -icon warning -type yesno -parent $w \
-message [mc "File \"%1\$s\" already exists.\nDo you want\
to overwrite it?" $selectFilePath]]
if {$reply eq "no"} {
return
}
}
if {
[info exists data(-typevariable)] && $data(-typevariable) ne ""
&& [info exists data(-filetypes)] && [llength $data(-filetypes)]
&& [info exists data(filterType)] && $data(filterType) ne ""
} then {
upvar #0 $data(-typevariable) typeVariable
set typeVariable [lindex $data(origfiletypes) \
[lsearch -exact $data(-filetypes) $data(filterType)] 0]
}
}
bind $data(okBtn) {}
set Priv(selectFilePath) $selectFilePath
}
# ::tk::dialog::file::GlobFiltered --
#
# Gets called to do globbing, returning the results and filtering them
# according to the current filter (and removing the entries for '.' and
# '..' which are never shown). Deals with evil cases such as where the
# user is supplying a filter which is an invalid list or where it has an
# unbalanced brace. The resulting list will be dictionary sorted.
#
# Arguments:
# dir Which directory to search
# type List of filetypes to look for ('d' or 'f b c l p s')
# overrideFilter Whether to ignore the filter for this search.
#
# NB: Assumes that the caller has mapped the state variable to 'data'.
#
proc ::tk::dialog::file::GlobFiltered {dir type {overrideFilter 0}} {
variable showHiddenVar
upvar 1 data(filter) filter
if {$filter eq "*" || $overrideFilter} {
set patterns [list *]
if {$showHiddenVar} {
lappend patterns .*
}
} elseif {[string is list $filter]} {
set patterns $filter
} else {
# Invalid list; assume we can use non-whitespace sequences as words
set patterns [regexp -inline -all {\S+} $filter]
}
set opts [list -tails -directory $dir -type $type -nocomplain]
set result {}
catch {
# We have a catch because we might have a really bad pattern (e.g.,
# with an unbalanced brace); even [glob -nocomplain] doesn't like it.
# Using a catch ensures that it just means we match nothing instead of
# throwing a nasty error at the user...
foreach f [glob {*}$opts -- {*}$patterns] {
if {$f eq "." || $f eq ".."} {
continue
}
# See ticket [1641721], $f might be a link pointing to a dir
if {$type != "d" && [file isdir [file join $dir $f]]} {
continue
}
lappend result $f
}
}
return [lsort -dictionary -unique $result]
}
proc ::tk::dialog::file::CompleteEnt {w} {
upvar ::tk::dialog::file::[winfo name $w] data
set f [$data(ent) get]
if {$data(-multiple)} {
if {![string is list $f] || [llength $f] != 1} {
return -code break
}
set f [lindex $f 0]
}
# Get list of matching filenames and dirnames
set files [if {[winfo class $w] eq "TkFDialog"} {
GlobFiltered $data(selectPath) {f b c l p s}
}]
set dirs2 {}
foreach d [GlobFiltered $data(selectPath) d] {lappend dirs2 $d/}
set targets [concat \
[lsearch -glob -all -inline $files $f*] \
[lsearch -glob -all -inline $dirs2 $f*]]
if {[llength $targets] == 1} {
# We have a winner!
set f [lindex $targets 0]
} elseif {$f in $targets || [llength $targets] == 0} {
if {[string length $f] > 0} {
bell
}
return
} elseif {[llength $targets] > 1} {
# Multiple possibles
if {[string length $f] == 0} {
return
}
set t0 [lindex $targets 0]
for {set len [string length $t0]} {$len>0} {} {
set allmatch 1
foreach s $targets {
if {![string equal -length $len $s $t0]} {
set allmatch 0
break
}
}
incr len -1
if {$allmatch} break
}
set f [string range $t0 0 $len]
}
if {$data(-multiple)} {
set f [list $f]
}
$data(ent) delete 0 end
$data(ent) insert 0 $f
return -code break
}