#!/usr/bin/wish # # xorriso-tcltk # Copyright (C) 2012, Thomas Schmitt , libburnia project. # Provided under GNU GPL version 2 or later. # # This is mainly a proof of concept for xorriso serving under a frontend. # It exercises several fundamental gestures of communication: # - connecting via two named pipes # - sending commands # - receiving replies # - inquiring the xorriso message sieve # - using the xorriso parsing service # Note that any other language than Tcl/Tk could be used, if it only can # do i/o via standard input and standard output or via named pipes. # Further it has to perform integer arithmetics and string manipulations. # And, well, a graphical widget set would be nice. set own_version "1.2.5" proc print_usage {argv0} { puts stderr "Usage:" puts stderr " $argv0 \[options\]" puts stderr "Options:" puts stderr " All options must be given with two dashes (\"--option\") in" puts stderr " order to distinguish them from any options of the Tcl shell." puts stderr " --help" puts stderr " Print this text and exit." puts stderr " --stdio" puts stderr " Establishes connection to xorriso via stdin and stdout." puts stderr " E.g. when letting xorriso start this frontend program:" puts stderr " xorriso -launch_frontend \$(which xorriso-tcltk) --stdio --" puts stderr " --named_pipes cmd_fifo reply_fifo" puts stderr " Establishes connection to a xorriso process started by:" puts stderr " xorriso -dialog on reply_fifo" puts stderr " which is then ready for a run of:" puts stderr " xorriso-tcltk --named_pipes cmd_fifo reply_fifo" puts stderr " It is important that the parent of xorriso and of this" puts stderr " tcl/tk frontend opens the named pipe for commands before" puts stderr " it opens the named pipe for replies. This avoids deadlock." puts stderr " --silent_start" puts stderr " Do not issue the start message xorriso-tcltk-version." puts stderr " This works only if --silent_start is the first argument." puts stderr " --no_bwidget" puts stderr " Do not try to load the Tcl/Tk package BWidget which is" puts stderr " a prerequisite for the \"/\" file browser buttons." puts stderr " --geometry {+|-}X{+|-}Y" puts stderr " Sets the position of the main window." puts stderr " --click_to_focus" puts stderr " Chooses that input fields and list boxes get the keyboard" puts stderr " focus only when being clicked by the mouse." puts stderr " --auto_focus" puts stderr " Chooses that the keyboard focus is where the mouse" puts stderr " pointer is. (Default)" puts stderr " --log_file path" puts stderr " Set a file address for logging of xorriso commands and" puts stderr " reply messages. The log lines will be appended." puts stderr "" puts stderr "If neither --stdio nor --named_pipes is given, then this script" puts stderr "will try to locate itself in the filesystem and start a xorriso" puts stderr "run that launches it again." puts stderr "" puts stderr "In the running GUI, click with the rightmost mouse button on" puts stderr "any GUI element to get its particular help text." puts stderr "" } # ------------------------------- the frontend ---------------------------- # # Starts xorriso, connects to it, sends commands, receives replies, # prepares replies for GUI # Connection to xorriso set cmd_conn "" set reply_conn "" # The addresses of the named pipes, if such are used (see option -named_pipe) set cmd_pipe_adr "" set reply_pipe_adr "" # The command to send (resp. the command most recently sent) set cmdline "" # Wether to clear the cmdline after sending set cmdline_clear true # Command counter set cmd_sent 0 # Current -mark synchronization text set mark_count 0 # Results of most recent await_all_replies set info_list "" set info_count 0 set emerging_info "" set result_list "" set result_count 0 set emerging_result "" # Highest severities encountered in total and with most recent command set highest_total_sev ALL set highest_total_sev_msg "" set highest_cmd_sev ALL set highest_cmd_sev_msg "" # This one registers like highest_cmd_sev with threshold ALL set highest_seen_cmd_sev ALL # State of last read_sieve command set sieve_ret 0 # Mode for parsing replies with multiple words of arbitrary characters # 0= single -msg_op parse commands # 1= -msg_op parse_bulk (less problems with connection latency) set bulk_parse_mode 1 # How many texts to pass with one parse_bulk command set bulk_parse_max_chunk 200 # Parse parameters set bulk_parse_prefix "" set bulk_parse_separators "" set bulk_parse_max_words "" set bulk_parse_flag "" # The announced number of texts to parse set bulk_parse_num_texts "" # Whether to complain on stderr about broken pipes. # This may be expected when xorriso is being shut down by this frontend. set expect_broken_pipes "0" # Local copies of xorriso state # Addresses of drives (or image files) set outdev_adr "" set indev_adr "" # Whether the medium is blank, appendable, closed, missing set indev_medium_status "missing" set outdev_medium_status "missing" # What kind of medium is presented by the drive set indev_profile "" set outdev_profile "" # List of known drive addresses set devlist "" # Intermediate storage for messages until the GUI is up with .msglist box set pre_msglist "" # Whether overwriting of files in the ISO model is allowed set overwrite_iso_files 1 # If overwrite_iso_files is 1: Wether overwriting of ISO directories is allowed set overwrite_iso_dirs 0 # Whether overwriting of files on disk is allowed set overwrite_disk_files 0 # The file where to log commands and replies for debugging purposes set log_file "" set log_conn stderr # Whether to log all commands and replies to the log_file set logging 0 # The result of the most recent isofs_ls run set isofs_ls_result "" # xorriso specific constants # List of severities (gets later overridden by -msg_op list_sev -) set xorriso_severity_list { ALL DEBUG UPDATE NOTE HINT WARNING SORRY MISHAP FAILURE FATAL ABORT NEVER } set scan_event_threshold HINT # --------- Communication between frontend and xorriso ---------- # Open the connection to a pair of named pipes. Program option -named_pipes # proc init_frontend_named_pipes {cmd_pipe reply_pipe} { global cmd_conn global reply_conn set cmd_conn [open "$cmd_pipe" w] set reply_conn [open "$reply_pipe" r] # Note: disencouraged flags would be necessary for opening idle fifo # set reply_conn [open "$reply_pipe" {RDONLY NONBLOCK}] } # Send a command line to the xorriso process. Do not wait for reply. # proc send_async_cmd {cmd} { global cmd_sent cmd_conn logging log_conn display_busy 1 log_puts " ==============================================================" log_puts " $cmd" display_msg "======> $cmd" incr cmd_sent puts $cmd_conn "$cmd" flush $cmd_conn } # Send a command line and a -mark command to xorriso. Wait until the # mark message confirms that all command output has been received. # proc send_marked_cmd {cmd} { global cmd_conn mark_count send_async_cmd "$cmd" incr mark_count set mark_cmd "-mark $mark_count" log_puts " $mark_cmd" puts $cmd_conn "$mark_cmd" flush $cmd_conn await_all_replies } # Wait for the currently pending mark message to arrive. # Buffer all received result lines and info messages. # proc await_all_replies {} { global reply_conn mark_count result_count result_list global info_count info_list expect_broken_pipes global .busy_text clear_reply_lists while {1} { set ret [gets "$reply_conn" line] if {"$ret" < 0} { if {"$expect_broken_pipes" != 1} { puts stderr "EOF at reply pipe" } break } log_puts "$line" if {[string range "$line" 0 0] == "M"} { if {[string range "$line" 5 end] == "$mark_count"} { break } else { # outdated mark message continue } } de_pkt_line "$line" } display_busy 0 } # Decode -pkt_output format to complete lines and buffer them. # proc de_pkt_line {line} { global info_list global info_count global emerging_info global result_list global result_count global emerging_result # Distinguish R and I set ch [string range "$line" 0 0] set payload [string range "$line" 5 end] if {"$ch" == "R"} { set emerging_result "$emerging_result$payload" } else { if {"$ch" == "I"} { set emerging_info "$emerging_info$payload" } else { return "" }} # if line end : add to list if {[string range "$line" 2 2] == "1"} { if {"$ch" == "R"} { lappend result_list "$emerging_result" incr result_count display_msg "$emerging_result" set emerging_result "" } else { lappend info_list "$emerging_info" incr info_count display_msg "$emerging_info" scan_info_for_event "$emerging_info" set emerging_info "" } } } # Search in the decoded info messages for the most severe event reports. # proc scan_info_for_event {line} { global highest_total_sev highest_total_sev_msg global highest_cmd_sev highest_cmd_sev_msg highest_seen_cmd_sev global scan_event_threshold global display_msg_enabled # check for word : CAPS : text ... set ret [regexp {[a-z][a-z]*[ ]*: [A-Z][A-Z]* :} "$line"] if {"$ret" != 1} {return ""} # retrieve severity set pos [string first ":" "$line"] set sev [string range "$line" [expr $pos+2] end] set pos [string first ":" "$sev"] set sev [string range "$sev" 0 [expr $pos-2]]; if {[compare_sev "$sev" "$highest_seen_cmd_sev"] > 0} { set highest_seen_cmd_sev "$sev" } if {[compare_sev "$sev" "$scan_event_threshold"] < 0} {return ""} if {"$display_msg_enabled" == 0} { set display_msg_enabled 1 display_msg "$line" set display_msg_enabled 0 } if {[compare_sev "$sev" "$highest_total_sev"] >= 0} { set highest_total_sev "$sev" set highest_total_sev_msg "$line" } if {[compare_sev "$sev" "$highest_cmd_sev"] >= 0} { set highest_cmd_sev "$sev" set highest_cmd_sev_msg "$line" } } # Unpack the output format of -msg_op read_sieve into a result_list # of strings which each hold one parsed word. # proc de_sieve {} { global result_list global sieve_ret set sieve_ret [lindex "$result_list" 0] set sieve_result_count [lindex "$result_list" 1] set payload "" set sieve_result_count 0 for {set i 2} {$i < [llength "$result_list"]} {incr i} { set line "" set num_lines [lindex "$result_list" $i] for {set j 0} {$j < "$num_lines"} {incr j} { incr i set line "$line[lindex "$result_list" $i]" if {$j < "$num_lines" - 1} { set line "$line\n" } else { lappend payload "$line" incr sieve_result_count } } } set result_list "$payload" set result_count "$sieve_result_count" } # Alternative to proc await_all_replies. It waits for a line at one of the # three channels and displays all lines which it receives before that line. # Used before this frontend had the opportunity to set up xorriso by commands # like -pkt_output "on". # proc wait_for_msg {prefix channel} { global reply_conn if {"$channel" == "M"} { set channel_prefix "M:0: " } else { set channel_prefix "$channel:1: " } set prefix_l [string length "$prefix"] while {1} { # >>> Have a timeout set ret [gets "$reply_conn" line] if {"$ret" < 0} { break } log_puts "$line" if {[string length "$line"] < "$prefix_l"} { display_msg "$line" continue } if {[string range "$line" 0 [expr "$prefix_l - 1"]] == "$prefix"} { return [string range "$line" "$prefix_l" end] } if {[string length "$line"] >= [expr "$prefix_l + 5"]} { if {[string range "$line" 0 4] == "$channel_prefix"} { if {[string range "$line" 5 [expr "$prefix_l + 4"]] == "$prefix"} { return [string range "$line" [expr "$prefix_l + 5"] end] } } } display_msg "$line" } } # Reset the buffer for result lines and info messages. # proc clear_reply_lists {} { global info_list global info_count global emerging_info global result_list global result_count global emerging_result set info_list "" set info_count 0 set emerging_info "" set result_list "" set result_count 0 set emerging_result "" } # Reset the register of the most severe event for command sequences. # Typically this is done before executing the commands of a procedure # that is triggered by the user interface. # proc reset_highest_cmd_sev {} { global highest_cmd_sev highest_cmd_sev_msg highest_seen_cmd_sev set highest_cmd_sev ALL set highest_cmd_sev_msg "" set highest_seen_cmd_sev ALL } # Clear the recordings of the xorriso message sieve. # proc clear_sieve {} { set disp_en_mem [set_display_msg 0] send_marked_cmd "-msg_op clear_sieve -" set_display_msg "$disp_en_mem" } # Obtain a recorded item from the xorriso message sieve. # proc read_sieve {name} { set disp_en_mem [set_display_msg 0] send_marked_cmd "-msg_op read_sieve '$name'" set_display_msg "$disp_en_mem" de_sieve } # ------- Inquiring xorriso status ------- # Get more information about drive that was inquired by recent -toc_of. # proc obtain_drive_info {dev} { global result_list global sieve_ret global indev_medium_status outdev_medium_status global indev_profile outdev_profile set line "" if {"$dev" == "in"} { set indev_medium_status "missing" set indev_profile "" } else { set outdev_medium_status "missing" set outdev_profile "" } read_sieve "Media status :" if {"$sieve_ret" > 0} { set reply [lindex "$result_list" 0] foreach i {blank appendable closed} { if {[string first "$i" "$reply"] != -1} { set line "$i " if {"$dev" == "in"} { set indev_medium_status "$i" } else { set outdev_medium_status "$i" } break } } } read_sieve "Media current:" if {"$sieve_ret" > 0} { set profile [lindex "$result_list" 0] if {"$profile" == "is not recognizable"} { set profile "no recognizable medium" set line "$line$profile" return "$line" } else { set line "$line$profile, " if {"$dev" == "in"} { set indev_profile "$profile" } else { set outdev_profile "$profile" } } } read_sieve "Media summary:" if {"$sieve_ret" > 0} { set line "$line[lindex "$result_list" 0] sessions, " if {"$dev" == "in"} { set line "$line[lindex "$result_list" 2] used" } else { set line "$line[lindex "$result_list" 3] free" } } return "$line" } # Inquire whether changes of the ISO image are pending. # proc changes_are_pending {} { global result_count result_list set disp_en_mem [set_display_msg 0] send_marked_cmd "-changes_pending show_status" set_display_msg "$disp_en_mem" if {"$result_count" >= 1} { if {[lindex "$result_list" 0] == "-changes_pending no"} { return "0" } return "1" } return "" } # Inquire the file type of an address in the xorriso ISO image tree. # This is a precondition for writing the session. Vice versa pending changes # block a change of the input drive or the program end. # proc get_iso_filetype {adr} { global result_list result_count scan_event_threshold set scan_event_mem "$scan_event_threshold" set scan_event_threshold "SORRY" set disp_en_mem [set_display_msg 0] send_marked_cmd "-lsdl [make_text_shellsafe "$adr"] --" set_display_msg "$disp_en_mem" set scan_event_threshold "$scan_event_mem" if {"$result_count" <= 0} { return "" } return [string range [lindex "$result_list" 0] 0 0] } # Inquire whether an ISO image model has been created inside xorriso. # This is a precondition for inserting files into the ISO tree model. # proc assert_iso_image {with_msg} { global highest_seen_cmd_sev scan_event_threshold set disp_en_mem [set_display_msg 0] set highest_seen_cmd_sev "" set set_mem "$scan_event_threshold" set scan_event_threshold "FATAL" send_marked_cmd "-lsd / --" set scan_event_threshold "$set_mem" set_display_msg "$disp_en_mem" if {[compare_sev "$highest_seen_cmd_sev" "FAILURE"] >= 0} { if {"$with_msg" == 1} { xorriso_tcltk_errmsg "xorriso-tcltk : SORRY : First you need to create or load an ISO image by selecting a drive or an image file" } return "0" } return "1" } # Obtain the list of possible event severity names, sorted in ascending order # proc inquire_severity_list {} { global xorriso_severity_list global result_list set disp_en_mem [set_display_msg 0] send_marked_cmd "-msg_op list_sev -" set_display_msg "$disp_en_mem" if {[lindex "$result_list" 0] != ""} { set xorriso_severity_list [split [lindex "$result_list" 0] " "] } } # Parse-by-xorriso handler function for proc inquire_dev # proc set_inquired_dev {} { global result_list indev_adr outdev_adr if {[llength "$result_list"] < 2} {return ""} set what [lindex "$result_list" 0] if {"$what" == "-dev" || "$what" == "-indev"} { set indev_adr [lindex "$result_list" 1] } if {"$what" == "-dev" || "$what" == "-outdev"} { set outdev_adr [lindex "$result_list" 1] } } # Inquire -indev and -outdev from xorriso and install in indev_adr # and outdev_adr. Usually called when the user messed up the text fields. # (This could be done by -toc_of like in proc refresh_indev. But here # i demonstrate the use of command -status and parsing its result by # help of xorriso.) # proc inquire_dev {} { set disp_en_mem [set_display_msg 0] send_marked_cmd "-status -dev" handle_result_list set_inquired_dev "''" "''" 2 0 set_display_msg "$disp_en_mem" update idletasks return "" } # Parse-by-xorriso handler function for proc isofs_ls # proc isofs_ls_handler {} { global result_list isofs_ls_result if {[lindex "$result_list" 0] == "total"} {return ""} lappend isofs_ls_result \ "[string range [lindex "$result_list" 0] 0 0] [lindex "$result_list" 8]" } # Produce a list of all files in a directory of the ISO model # proc isofs_ls {dir} { global isofs_ls_result set isofs_ls_result "" set disp_en_mem [set_display_msg 0] send_marked_cmd "-lsl [make_text_shellsafe "$dir"]" handle_result_list isofs_ls_handler "''" "''" 0 0 set_display_msg "$disp_en_mem" return "$isofs_ls_result" } # Tells the file type of an absolute path in the ISO model. # Indicator characters like with ls -l. Empty text means non existing file. # proc isofs_filetype {path} { global result_list set disp_en_mem [set_display_msg 0] send_marked_cmd "-lsdl [make_text_shellsafe "$path"]" set_display_msg "$disp_en_mem" if {[llength "$result_list"] < 1} {return ""} return [string range [lindex "$result_list" 0] 0 0] } # Verify that the connected process runs a xorriso program that is modern # enough. This is done before sending xorriso the setup commands. # proc check_xorriso_version {} { global sieve_ret result_list pre_msglist xorriso_version global reply_conn set min_version "1.2.5" set version "0.0.0 (= unknown)" set disp_en_mem [set_display_msg 0] # In order to see the pre-frontend messages of xorriso # set an individual -mark and use send_async_cmd set mark_text "xorriso-tcltk-version-check-[clock seconds]" send_async_cmd "-mark [make_text_shellsafe "$mark_text"]" set_display_msg "$disp_en_mem" wait_for_msg "$mark_text" "M" set_display_msg 0 send_async_cmd "-version" set xorriso_version [wait_for_msg "xorriso version : " "R"] if {"$xorriso_version" < "$min_version"} { puts stderr "xorriso-tcltk: xorriso-$xorriso_version is too old." puts stderr "xorriso-tcltk: Need at least version $min_version" window_ack \ "xorriso-$xorriso_version is too old. Need at least version $min_version" \ "red" "embedded" central_exit 2 } set_display_msg "$disp_en_mem" } # Commands which bring the connected xorriso process into the state that # is expected by this frontend. # proc setup_xorriso {} { set cmd "" # Invalidate possible -mark 1 set cmd "$cmd -mark 0" # Make replies digestible for await_all_replies set cmd "$cmd -pkt_output on" # Report version early set cmd "$cmd -version" # This frontend relies heavily on the message sieve set cmd "$cmd -msg_op start_sieve -" # -reassure questions from xorriso would not be properly handled by # this frontend set cmd "$cmd -reassure off" set cmd "$cmd -for_backup" # There is a performance problem in xorriso with -hardlinks on and # image manipulations. So for now -hardlinks is set to off. set cmd "$cmd -hardlinks off" set cmd "$cmd -abort_on NEVER" set cmd "$cmd -return_with ABORT 32" set cmd "$cmd -report_about UPDATE" set cmd "$cmd -osirrox on" set cmd "$cmd -iso_rr_pattern off" set cmd "$cmd -disk_pattern off" set cmd "$cmd -follow mount:limit=100" send_marked_cmd "$cmd" inquire_severity_list } # ------ Parsing by help of xorriso ------ # Parsing by xorriso takes from the frontend the burden to understand # and implement the quoting rules of xorriso input and output. # Lines which are supposed to consist of several words get sent to # xorriso command -msg_op. The result lines of this command encode # the words unambigously in one or more text lines. # This is supposed to be safe for even the weirdest file names. # Only NUL characters cannot be part of names. # If enabled: Start a bulk parser job by which xorriso shall split the output # of e.g. -lsl into single words from which this frontend can pick information. # proc start_bulkparse {prefix separators max_words flag num_lines} { global bulk_parse_mode bulk_parse_prefix bulk_parse_separators global bulk_parse_max_words bulk_parse_flag bulk_parse_num_texts if {"$num_lines" <= 0} {return ""} set bulk_parse_prefix "$prefix" set bulk_parse_separators "$separators" set bulk_parse_max_words "$max_words" set bulk_parse_flag "$flag" set bulk_parse_num_texts "$num_lines" if {"$bulk_parse_mode" == 1} { set cmd "-msg_op parse_bulk \"$prefix $separators $max_words $flag $num_lines\"" send_async_cmd "$cmd" # Do not wait for mark } } # Submit a new input line to the xorriso parser. If no bulk parser job was # started then submit a single line parser command. # proc submit_bulkparse {text} { global cmd_conn reply_conn global result_list result_count global bulk_parse_mode bulk_parse_prefix bulk_parse_separators global bulk_parse_max_words bulk_parse_flag if {"$bulk_parse_mode" != 1} { clear_reply_lists } set disp_en_mem [set_display_msg 0] set num_lines [expr [count_newlines "$text"] + 1] if {"$bulk_parse_mode" == 0} { set cmd "-msg_op parse \"$bulk_parse_prefix $bulk_parse_separators $bulk_parse_max_words $bulk_parse_flag $num_lines\"" send_async_cmd "$cmd" } else { log_puts ">>>>> $num_lines" puts $cmd_conn "$num_lines" } log_puts ">>>>> $text" puts $cmd_conn "$text" flush $cmd_conn if {"$bulk_parse_mode" != 1} { set loop_limit 2 while {"$result_count" < "$loop_limit"} { set ret [gets "$reply_conn" line] if {"$ret" < 0} { return ""} log_puts "$line" de_pkt_line "$line" if {"$result_count" == 1} { set parse_ret [lindex "$result_list" 0] } if {"$result_count" == 2} { set num_replies [lindex "$result_list" 1] set loop_limit [expr "$num_replies * 2 + 2"] } } de_sieve } set_display_msg "$disp_en_mem" } # If a bulk parsing job was started, then read the expected number of # replies into the result buffer and call handler_proc to inspect them. # Each input line of the parser yields one reply buffer full of parsed words. # proc read_bulkparse {handler_proc num_texts} { global bulk_parse_mode if {"$bulk_parse_mode" != 1} { return ""} set disp_en_mem [set_display_msg 0] for {set i 0} {"$i" < "$num_texts"} {incr i} { clear_reply_lists read_parse_reply $handler_proc } set_display_msg "$disp_en_mem" } # Read and decode the xorriso parser reply for one input line. # proc read_parse_reply {} { global reply_conn global result_list result_count set loop_limit 2 while {"$result_count" < "$loop_limit"} { set ret [gets "$reply_conn" line] if {"$ret" < 0} { return ""} log_puts "$line" de_pkt_line "$line" if {"$result_count" == 1} { set parse_ret [lindex "$result_list" 0] } if {"$result_count" == 2} { set num_replies [lindex "$result_list" 1] set loop_limit [expr "$num_replies * 2 + 2"] } } de_sieve } # Let xorriso parse the lines in the result buffer and call handler_proc # with the parse result of each line. # This is used to split the result lines of -lsl into words from which # handler proc isolist_parse_handler picks the info which it displays # in .stbox isolist . # Note that all parameters must be xorriso words. E.g. empty prefix or # separator have to be submitted as tcl string "''" rather than "". # proc handle_result_list {handler_proc \ prefix separators max_words flag } { global result_list global bulk_parse_mode bulk_parse_max_chunk set raw_list "$result_list" set raw_line_count [expr [llength "$raw_list"]] if {"$raw_line_count" > "$bulk_parse_max_chunk"} { set chunk_size "$bulk_parse_max_chunk" } else { set chunk_size "$raw_line_count" } start_bulkparse "$prefix" "$separators" "$max_words" "$flag" "$chunk_size" set submit_count 0 set submit_in_chunk_count 0 foreach i "$raw_list" { submit_bulkparse "$i" incr submit_count incr submit_in_chunk_count if {"$bulk_parse_mode" != 1} { $handler_proc } if {"$bulk_parse_mode" == 1 && "$submit_in_chunk_count" == "$chunk_size"} { read_bulkparse "$handler_proc" "$chunk_size" set todo [expr "$raw_line_count" - "$submit_count"] if {"$todo" <= 0} { break } if {"$todo" > "$bulk_parse_max_chunk"} { set chunk_size "$bulk_parse_max_chunk" } else { set chunk_size "$todo" } start_bulkparse "$prefix" "$separators" "$max_words" "$flag" \ "$chunk_size" set submit_in_chunk_count 0 } } display_busy 0 } # ------------------------------- the GUI ---------------------------- # ------ State variables ------ # Whether to display messages in .msglist set display_msg_enabled 1 # Whether a device list is already displayed set devices_scanned 0 # Currently displayed ISO directory set isodir_adr "" set isodir_is_pwd 0 # The plain names and types matching listbox .isolist set isolist_names "" set isolist_types "" # The name which to select after isodir_return set isodir_return_name "" # The address where to move selected ISO files set isomanip_move_target "" # Memorized isolist selection set memorized_isolist_selection "" # Image file address for .burn_write_image set burn_write_image_adr "" # Whether to close medium after writing set burn_write_close 0 # Whether to force CD TAO, DVD-R Inremental, DVD+R/BD-R open ended track set burn_write_tao 0 # Whether to engage Defect Management on formatted BD media set burn_write_defect_mgt 0 # Answer of yes/no window set answer_of_yesno "" # The hard disk filesystem address to be mapped into isodir_adr set insert_from_adr "" # Whether to insert with leafname of insert_from_adr underneath isodir_adr # (else: -map $insert_from_adr $isodir_adr) set insert_underneath 1 # Whether to insert at or under the selected .isolist item # rather than isodir_adr set insert_at_selected 0 # The hard disk filesystem address to which to extract from isodir_adr set extract_to_adr "" # Whether to insert with leafname of insert_from_adr underneath isodir_adr # (else: -map $insert_from_adr $isodir_adr) set extract_underneath 1 # Whether to insert at or under the selected .isolist item set extract_from_selected 0 # Whether to temporarily enforce rwx permissions for target directories on disk set extract_auto_chmod 0 # Whether the display label .busy_text is already usable set busy_text_exists 0 # Whether to demand a click before focus goes to entry or listbox set click_to_focus "0" # Whether .ack_window , .yesno_window , .help_window is already displayed # (>>> Better would be to block user interaction until they are gone) set ack_window_is_active 0 set yesno_window_is_active 0 set help_window_is_active 0 # Whether the help window already has a scroll bar set help_window_has_scroll 0 # Whether there is the BWidget package available: 0=unknown, 1=yes, -1=banned # set have_bwidget 0 set bwidget_version "" # Whether the .browse_disk_window is already displayed set browse_disk_window_is_active 0 set browse_disk_window_var "" # Whether the .browse_iso_window is already displayed set browse_iso_window_is_active 0 set browse_iso_window_var "" # Whether to bring the selected browser item directly into the text field set browse_select_is_setvar 1 # ------ GUI callback procedures ---- # Called when the Return key is hit in commandline. # proc cmdline_return {} { global cmdline cmdline_clear global .cmdline .cmdline_text .cmdline_entry global highest_cmd_sev global highest_cmd_sev_msg reset_highest_cmd_sev set_display_msg 1 send_marked_cmd "$cmdline" set cmdline "" # To force display of GUI changes now and not some time later update idletasks } # Called when the input drive address shall be brought into effect with # xorriso. # proc indev_return {} { global indev_adr global .indev_entry global .outdev_entry if {[assert_no_changes] == 0} { inquire_dev return "0" } reset_highest_cmd_sev send_marked_cmd "-indev [make_text_shellsafe $indev_adr]" set indev_mem_adr "$indev_adr" .indev_entry icursor 0 refresh_indev return "1" } # Called when the "Eject" button for the input drive is hit. # proc eject_indev {} { if {[assert_no_changes] == 0} {return ""} reset_highest_cmd_sev send_marked_cmd "-eject indev" refresh_outdev refresh_indev } # Called when the output drive address shall be brought into effect with # xorriso. # proc outdev_return {} { global outdev_adr indev_adr global .outdev_entry reset_highest_cmd_sev send_marked_cmd "-outdev [make_text_shellsafe $outdev_adr]" set outdev_mem_adr "$outdev_adr" .outdev_entry icursor 0 refresh_outdev return "1" } # Called when the "Eject" button for the output drive is hit. # proc eject_outdev {} { global outdev_adr indev_adr if {"$outdev_adr" == "$indev_adr"} { if {[assert_no_changes] == 0} {return ""} } reset_highest_cmd_sev send_marked_cmd "-eject outdev" refresh_outdev refresh_indev } # Called when both drive addresses shall be brought into effect with xorriso. # proc dev_return {} { global outdev_adr indev_adr global .outdev_entry .indev_entry if {"$outdev_adr" != "$indev_adr"} { if {[indev_return] == 0} {return "0"} outdev_return } else { if {[assert_no_changes] == 0} { inquire_dev return "0" } reset_highest_cmd_sev send_marked_cmd "-dev [make_text_shellsafe $outdev_adr]" .outdev_entry icursor 0 refresh_outdev .indev_entry icursor 0 refresh_indev } } # Obtain and display the input drive status. # Called after the input drive address may have changed. # proc refresh_indev {} { global result_list global indev_adr global sieve_ret global .indev_summary .indev_summary configure -text "" set indev_adr "" update idletasks set disp_en_mem [set_display_msg 0] clear_sieve send_marked_cmd "-toc_of in:short" read_sieve "Drive current:" set_display_msg "$disp_en_mem" if {"$sieve_ret" > 0} { set cmd [lindex "$result_list" 0] if {"$cmd" == "-indev" || "$cmd" == "-dev"} { set indev_adr [lindex "$result_list" 1] } set line [obtain_drive_info in] .indev_summary configure -text "$line" } .avail_label configure -text "" update idletasks isodir_return "refresh_indev" } # Obtain and display the output drive status. # Called after the output drive address may have changed. # proc refresh_outdev {} { global result_list global outdev_adr global sieve_ret .outdev_summary configure -text "" set outdev_adr "" update idletasks set disp_en_mem [set_display_msg 0] clear_sieve send_marked_cmd "-toc_of out:short" read_sieve "Drive current:" set_display_msg "$disp_en_mem" if {"$sieve_ret" > 0} { set cmd [lindex "$result_list" 0] if {"$cmd" == "-outdev" || "$cmd" == "-dev"} { set outdev_adr [lindex "$result_list" 1] } set line [obtain_drive_info out] .outdev_summary configure -text "$line" } .avail_label configure -text "" update idletasks } # Scan the system for optical drives with rw permission # Called when the "Scan for drives button" is hit. # proc scan_for_drives {} { global .drivelist global sieve_ret result_list devlist devices_scanned indev_adr outdev_adr if {[assert_no_changes] == 0} {return ""} if {"$indev_adr" != "" || "$outdev_adr" != ""} { if {[window_yesno \ "Really give up aquired drives for scanning a new drive list ?"] \ != 1} { return "" } } set max_idx [.drivelist index end] .drivelist delete 0 [expr "$max_idx-1"] set devlist "" reset_highest_cmd_sev clear_sieve send_marked_cmd "-devices" set max_idx 0 while {1} { read_sieve "? -dev" if {"$sieve_ret" > 0} { .drivelist insert end "[lindex "$result_list" 0] : [lindex "$result_list" 2] [lindex "$result_list" 3]" lappend devlist [lindex "$result_list" 0] } else { break } } while {1} { read_sieve "?? -dev" if {"$sieve_ret" > 0} { .drivelist insert end "[lindex "$result_list" 0] : [lindex "$result_list" 2] [lindex "$result_list" 3]" lappend devlist [lindex "$result_list" 0] } else { break } } set devices_scanned 1 update idletasks # Command -devices drops all aquired drives refresh_outdev refresh_indev } # Refresh the display after some xorriso may have changed the status # Called by the "Refresh state" display button and others. # proc refresh_state {} { refresh_indev refresh_outdev } # Reset the recorded Worst problem message. # Called when the "Clear" button is hit. # proc clear_total_errmsg {} { global highest_total_sev global highest_total_sev_msg set highest_total_sev ALL set highest_total_sev_msg "" update idletasks } # Called when the "Pick input drive button" is hit. # proc pick_indev {} { pick_drive indev } # Called when the "Pick output drive button" is hit. # proc pick_outdev {} { pick_drive outdev } # Called when the "Pick drive for both roles" button is hit. # or when an item in the scanned drive list is double-clicked. # proc pick_dev {} { pick_drive dev } # Perform the actual work of pick_dev, pick_indev, and pick_outdev # proc pick_drive {role} { global .drivelist global devlist global highest_cmd_sev_msg outdev_adr indev_adr devices_scanned set selected [.drivelist curselection] if {[llength "$selected"] != 1} { set must_scan "" if {"$devices_scanned" == 0} { set must_scan " scan and"} xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : First you must$must_scan select a single drive" return "" } set drive_idx [lindex "$selected" 0] if {"$role" == "dev"} { set outdev_adr [lindex "$devlist" "$drive_idx"] set indev_adr [lindex "$devlist" "$drive_idx"] dev_return } if {"$role" == "outdev"} { set outdev_adr [lindex "$devlist" "$drive_idx"] outdev_return } if {"$role" == "indev"} { set indev_adr [lindex "$devlist" "$drive_idx"] indev_return } .drivelist selection clear 0 end } # Called when the "Give up drives" button is hit. # proc give_up_dev {} { global outdev_adr indev_adr if {[assert_no_changes] == 0} {return ""} set outdev_adr "" outdev_return set indev_adr "" indev_return } # Obtain and display the content of the current ISO directory. # Called when the Return key is hit in the .isodir_entry text field # and by many others which change variable isodir_adr or the # content of the directory in xorriso's tree model. # proc isodir_return {caller} { global isodir_adr result_list isolist_names isolist_types isodir_return_name global isodir_is_pwd highest_cmd_sev highest_cmd_sev_msg global indev_adr outdev_adr global .isolist global bulk_parse_mode global bulk_parse_max_chunk set chunk_size 0 set max_idx [.isolist index end] .isolist delete 0 [expr "$max_idx-1"] update idletasks set isolist_names "" set isolist_types "" if {"$indev_adr" == "" && "$outdev_adr" == ""} { if {[changes_are_pending] == "0"} {return ""} } normalize_isodir_adr set file_type [isofs_filetype "$isodir_adr"] if {"$file_type" != "d" && "$file_type" != ""} { .isolist insert end "@@@ exists but is not a directory @@@" set isodir_is_pwd 0 return "" } set disp_en_mem [set_display_msg 0] set highest_cmd_sev_mem "$highest_cmd_sev" set highest_cmd_sev_msg_mem "$highest_cmd_sev_msg" reset_highest_cmd_sev send_marked_cmd "-cd [make_text_shellsafe "$isodir_adr"]" if {[compare_sev "$highest_cmd_sev" "WARNING"] < 0} { send_marked_cmd "-lsl --" set isodir_is_pwd 1 } else { send_marked_cmd "-lsl [make_text_shellsafe "$isodir_adr"] --" set isodir_is_pwd 0 } handle_result_list isolist_parse_handler "''" "''" 0 0 set_display_msg "$disp_en_mem" set highest_cmd_sev "$highest_cmd_sev_mem" set highest_cmd_sev_msg "$highest_cmd_sev_msg_mem" if {"$isodir_return_name" != ""} { set idx [lsearch -exact "$isolist_names" "$isodir_return_name"] if {"$idx" != -1} { .isolist see "$idx" .isolist selection set "$idx" } set isodir_return_name "" } update idletasks } # The handler procedure that is submitted to proc handle_result_list # and will be called for every parsed line. # It records file names and types in separate lists and displays them # in the .isolist box. # proc isolist_parse_handler {} { global result_list isolist_names isolist_types global .isolist if {[lindex "$result_list" 0] == "total"} {return ""} set name [lindex "$result_list" 8] set ftype [string range [lindex "$result_list" 0] 0 0] lappend isolist_names "$name" lappend isolist_types "$ftype" .isolist insert end "$ftype $name" } # Make current the ISO directory that was selected from the .isolist box. # Called when an item in the .isolist box is double-clicked. # proc pick_isodir {} { global isolist_names isolist_types isodir_adr isodir_return_name global .isolist set selected [.isolist curselection] if {[llength "$selected"] != 1} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : First you must select a single directory" return "" } set idx [lindex "$selected" 0] if {[lindex "$isolist_types" "$idx"] != "d"} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : You may only double-click a directory" return "" } if {"$isodir_adr" == "/"} { set isodir_adr "" } set isodir_adr "$isodir_adr/[lindex $isolist_names "$idx"]" set isodir_return_name "" isodir_return "pick_isodir" } # Make current the parent directory of the current ISO directory. # Called when the "Up" button is hit. # proc isodir_up {} { global isodir_adr isodir_return_name set isodir_return_name "" set idx [string last "/" "$isodir_adr"] set l [string length "$isodir_adr"] if {"$idx" == -1} { set isodir_return_name "$isodir_adr" set isodir_adr "/" } else { if {"$idx" > 0} { if {"$idx" < [expr "$l" - 1]} { set isodir_return_name \ [string range "$isodir_adr" [expr "$idx" + 1] end] } set isodir_adr [string range "$isodir_adr" 0 [expr "$idx" - 1]] } else { if {"$l" > 1} { set isodir_return_name [string range "$isodir_adr" 1 end] } set isodir_adr "/" } } isodir_return "isodir_up" } # Rename resp. move the files which are selected in the .isolist box. # The target is defined by the .isomanip_move_target text field. # Called when the "Rename to:" button is hit. # proc isomanip_mv {} { global .isolist global isomanip_move_target isolist_names isodir_is_pwd isodir_adr global isodir_return_name if {"$isomanip_move_target" == ""} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : First you must enter a target address" return "" } set selected [.isolist curselection] set num_selected [llength "$selected"] if {"$num_selected" < 1} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : First you must select one or more ISO files" return "" } set target "$isomanip_move_target" if {"$isodir_is_pwd" == 0 && [string range "$target" 0 0] != "/"} { set target [combine_dir_and_name "$isodir_adr" "$target"] } set target_ftype [get_iso_filetype "$target"] # If more than one selected : target must be directory if {"$num_selected" > 1} { if {"$target_ftype" != "d" && "$target_ftype" != ""} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : If multiple files are selected then the target must be a directory" return "" } if {"$target_ftype" == ""} { set isomanip_move_target_mem "$isomanip_move_target" set isomanip_move_target "$target" isomanip_mkdir set isomanip_move_target "$isomanip_move_target_mem" } } enforce_overwrite_settings "isofs" reset_highest_cmd_sev foreach i "$selected" { set name [lindex "$isolist_names" "$i"] if {"$isodir_is_pwd" == 0} { set name [combine_dir_and_name "$isodir_adr" "$name"] } # Ask for confirmation if overwriting is about to happen if {"$target_ftype" == "d"} { set eff_target [combine_dir_and_name "$target" "$name"] set eff_target_ftype [get_iso_filetype "$eff_target"] if {[handle_iso_overwriting \ "$eff_target" "$eff_target_ftype" 0 "" ""] == "0"} { return "" } } else { if {[handle_iso_overwriting \ "$target" "$target_ftype" 0 "" ""] == "0"} { return "" } } send_marked_cmd "-mv [make_text_shellsafe "$name"] [make_text_shellsafe "$target"] --" } if {[llength "$selected"] == 1} { set isodir_return_name [path_touches_isodir "$target"] } browse_iso_refresh isodir_return "isomanip_mv" } # Create an empty ISO directory with address given by variable # isomanip_move_target. # Called when the "Make directory" button is hit or by other functions. # proc isomanip_mkdir {} { global isomanip_move_target isodir_adr isodir_return_name if {"$isomanip_move_target" == ""} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : First you must enter a target address" return "" } if {[string range "$isomanip_move_target" 0 0] == "/"} { set abs_adr "$isomanip_move_target" } else { set abs_adr [combine_dir_and_name "$isodir_adr" "$isomanip_move_target"] } reset_highest_cmd_sev send_marked_cmd "-mkdir [make_text_shellsafe "$abs_adr"] --" # Refresh only if new dir in isodir_adr # or if a parent directory of new dir is created in isodir_adr set touch_name [path_touches_isodir "$abs_adr"] if {"$touch_name" != ""} { if {[llength [.isolist curselection]] != 0} { memorize_isolist_selection set selection_memorized 1 } else { set isodir_return_name "$touch_name" set selection_memorized 0 } isodir_return "isomanip_mkdir" if {"$selection_memorized" != 0} { restore_isolist_selection } } browse_iso_refresh } # Remove a file or directory tree from the ISO image. # Called when the "Delete" button is hit. # proc isomanip_rm_r {} { global .isolist global isomanip_move_target isolist_names isodir_is_pwd isodir_adr set selected [.isolist curselection] if {[llength "$selected"] < 1} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : First you must select one or more ISO files" return "" } if {[window_yesno "Really delete the selected files from ISO image ?"] \ != 1} { return "" } reset_highest_cmd_sev foreach i "$selected" { set name [lindex "$isolist_names" "$i"] if {"$isodir_is_pwd" == 0} { set name [combine_dir_and_name "$isodir_adr" "$name"] } send_marked_cmd "-rm_r [make_text_shellsafe "$name"] --" } browse_iso_refresh isodir_return "isomanip_rm_r" } # Perform a blanking run on the output drive. # Called when the "Blank" button is hit. # proc burn_blank {} { global outdev_adr outdev_profile if {[assert_outdev blanking] <= 0} {return ""} set victim "medium in" if {[string first "stdio" "$outdev_profile"] == 0} { set victim "image file" } if {[window_yesno \ "Really blank the $victim [make_text_shellsafe "$outdev_adr"] ?"] \ != 1} { return "" } reset_highest_cmd_sev send_marked_cmd "-blank as_needed" refresh_indev refresh_outdev } # Perform a formatting run on the output drive. # Called when the "Format" button is hit. # proc burn_format {} { global outdev_adr outdev_profile if {[assert_outdev formatting] <= 0} {return ""} if {[string first "stdio" "$outdev_profile"] == 0} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : Image files cannot be formatted" return "" } if {[window_yesno "Really format the medium in $outdev_adr ?"] \ != 1} { return "" } reset_highest_cmd_sev send_marked_cmd "-format as_needed" refresh_indev refresh_outdev } # Write pending changes in the xorriso ISO model as session to the output # drive. This will be an add-on session if the drive is output and input drive # and if its medium is not blank. # Else it will be a new independent ISO image. # proc burn_commit {} { global outdev_adr result_list result_count outdev_medium_status global burn_write_close burn_write_tao burn_write_defect_mgt global indev_adr outdev_adr if {[assert_outdev "writing ISO session"] <= 0} {return ""} if {"$outdev_adr" == "$indev_adr"} { if {"$outdev_medium_status" != "blank" && \ "$outdev_medium_status" != "appendable"} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : Medium in output drive is neither blank nor appendable" return "" } } else { if {"$outdev_medium_status" != "blank"} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : Medium in output drive is not blank" return "" } } if {[changes_are_pending] == "0"} { window_ack "No changes are pending. Will not write ISO session." \ "grey" "toplevel" return "" } if {"$outdev_adr" == "$indev_adr"} { if {[window_yesno "Really write ISO changes as session to $outdev_adr ?"] \ != 1} { return "" } } else { if {[window_yesno "Really write new ISO filesystem to $outdev_adr ?"] \ != 1} { return "" } } set cmd "" set cmd "$cmd -close" if {"$burn_write_close" == 1} { set cmd "$cmd on" } else { set cmd "$cmd off" } set cmd "$cmd -write_type" if {"$burn_write_tao" == 1} { set cmd "$cmd tao" } else { set cmd "$cmd auto" } set cmd "$cmd -stream_recording" if {"$burn_write_defect_mgt" == 1} { set cmd "$cmd off" } else { set cmd "$cmd data" } set cmd "$cmd -commit" reset_highest_cmd_sev send_marked_cmd "$cmd" refresh_indev refresh_outdev } # Verify the MD5 checksums of the data files in the tree underneath the # current ISO directory. # Called when the "Verify" in the "ISO directory:" line is hit. # proc isodir_verify {} { global isodir_adr reset_highest_cmd_sev send_marked_cmd "-check_md5_r sorry [make_text_shellsafe "$isodir_adr"] --" # >>> select mismatching files or directories with mismatching files } # Verify the MD5 checksums of the data files orch are selected or which # sit in the trees underneath the selected items in the isolist box. # Called when the "Verify" in the "ISO selection:" line is hit. # proc isomanip_verify {} { global .isolist global isomanip_move_target isolist_names isodir_is_pwd isodir_adr set selected [.isolist curselection] if {[llength "$selected"] < 1} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : First you must select one or more ISO files" return "" } reset_highest_cmd_sev foreach i "$selected" { set name [combine_dir_and_name "$isodir_adr" \ [lindex "$isolist_names" "$i"]] send_marked_cmd "-check_md5_r sorry [make_text_shellsafe "$name"] --" } # >>> select mismatching files or directories with mismatching files } # Slow down the spinning of the aquired optical drives. # Called when button "Calm drives" is hit. # proc calm_drives {} { reset_highest_cmd_sev send_marked_cmd "-calm_drive all" } # Burn a data file from disk as session to the output drive. # Called when the "Burn image file:" button is hit. # proc burn_write_image {} { global burn_write_image_adr burn_write_close outdev_adr outdev_medium_status global outdev_profile burn_write_tao burn_write_defect_mgt indev_adr inquire_dev if {"$indev_adr" != ""} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : You may not have an input drive open when writing an image file" return "" } if {[assert_outdev "writing an image file"] <= 0} {return ""} if {"$burn_write_image_adr" == ""} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : You have to set an image file address in the hard disk filesystem first" return "" } if {"$outdev_medium_status" != "blank"} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : You must have a blank medium in the output drive for burning an image data file" return "" } if {[file readable "$burn_write_image_adr"] == 0 || \ [file isfile "$burn_write_image_adr"] == 0 || [file exists "$burn_write_image_adr"] == 0} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : Image file '$burn_write_image_adr' is not a readable data file" return "" } if {[window_yesno "Really write '$burn_write_image_adr' as image to $outdev_adr ?"] \ != 1} { return "" } set cmd "-as cdrecord -v" if {[regexp "^CD" "$outdev_profile"] == 1 && \ ( "$outdev_medium_status" == "appendable" || "$burn_write_tao" == 1 )} { set cmd "$cmd padsize=150s" } set cmd "$cmd dev=[make_text_shellsafe "$outdev_adr"]" set cmd "$cmd [make_text_shellsafe "$burn_write_image_adr"]" if {"$burn_write_tao" == 1} { set cmd "$cmd -tao" } if {"$burn_write_close" != 1} { set cmd "$cmd -multi" } if {"$burn_write_defect_mgt" == 1} { set cmd "$cmd stream_recording=off" } else { set cmd "$cmd stream_recording=32s" } reset_highest_cmd_sev send_marked_cmd "$cmd" refresh_state } # Discard all image modifications and reload ISO image model from input drive. # Called when the "Rollback" button is hit. # proc iso_rollback {} { if {[window_yesno \ "Really discard all pending changes and reload from input drive ?"] \ != 1} { return "" } reset_highest_cmd_sev send_marked_cmd "-rollback" .avail_label configure -text "" isodir_return "iso_rollback" } # Inquire an accurate prediction of free space after writing a session with # the pending changes of the ISO image. # Called when button "Refresh avail:" is hit. # proc refresh_avail {} { global result_list highest_cmd_sev global sieve_ret if {[assert_outdev "refreshing available space count"] <= 0} {return ""} set line "n.a." reset_highest_cmd_sev clear_sieve send_marked_cmd "-tell_media_space" if {[compare_sev "$highest_cmd_sev" "FAILURE"] < 0} { set ac "" read_sieve "After commit :" if {"$sieve_ret" > 0} { set ac [lindex "$result_list" 0] set ac [string range "$ac" 0 [expr [string length "$ac"] - 2]] set line "[format "%7dm" [expr "$ac / 512"]]" } } .avail_label configure -text "$line" } # Warn and prompt the user for confirmation if there is the risk to overwrite # existing files in the ISO image model. # Called from several procedures which change the ISO tree. # proc handle_iso_overwriting {target target_ftype from_is_dir selected_adr selected_ftype} { global overwrite_iso_dirs # >>> Nicer would be: # >>> Check if any file will get overwritten. Not only the direct target. # >>> Then silently allow directories to be merged if {"$target_ftype" != ""} { if {"$target_ftype" == "d"} { if {"$from_is_dir" == 1} { if {[window_yesno \ "'$target'\n\nReally merge with existing ISO directory ?"] \ != 1} { return "0" } } else { if {"$overwrite_iso_dirs" == 1} { if {[window_yesno \ "'$target'\n\nReally overwrite existing ISO directory ?"] \ != 1} { return "0" } } else { xorriso_tcltk_errmsg "xorriso-tcltk : SORRY : You would have to enable \"Overwriting of ISO directories\"" return "0" } } } else { if {[window_yesno "'$target'\n\nReally overwrite existing ISO file ?"] \ != 1} { return "0" } } } if {"$selected_adr" != "$target" && "$selected_adr" != "" && \ "$selected_ftype" != "d" && "$selected_ftype" != ""} { if {[window_yesno \ "'$selected_adr'\n\nReally replace existing ISO file by a directory ?"] \ != 1} { return "0" } } return "1" } # Insert a file or directory tree into the ISO model tree and schedule it # for being copied when "Write ISO session" is hit. # Called when button "Insert from disk:" is hit. # proc insert_from {} { global insert_from_adr isodir_adr isolist_names isodir_return_name global insert_at_selected insert_underneath if {[assert_iso_image 1] == 0} {return ""} if {"$insert_from_adr" == ""} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : You have to set a source address in the hard disk filesystem first" return "" } set selected_ftype "" set selected_adr "" if {"$insert_at_selected" == 1} { set selected [.isolist curselection] if {[llength "$selected"] != 1} { xorriso_tcltk_errmsg "xorriso-tcltk : SORRY : You must select exactly one ISO file as insertion target" return "" } set target "[lindex "$isolist_names" [lindex "$selected" 0]]" set selected_ftype [get_iso_filetype "$target"] set selected_adr "$target" } else { set target "$isodir_adr" } set from_is_dir [localfs_isdir "$insert_from_adr"] set name [file tail "$insert_from_adr"] if {"$insert_underneath" == 1 || "$from_is_dir" == 0} { set target [combine_dir_and_name "$target" "$name"] } set target_ftype [get_iso_filetype "$target"] if {[handle_iso_overwriting "$target" "$target_ftype" "$from_is_dir" \ "$selected_adr" "$selected_ftype"] == "0"} { return "" } set preserve_selection 0 if {"$insert_underneath" + "$insert_at_selected" == 1} { set isodir_return_name "$name" } else { set preserve_selection 1 } reset_highest_cmd_sev enforce_overwrite_settings "isofs" send_marked_cmd "-map [make_text_shellsafe "$insert_from_adr"] [make_text_shellsafe "$target"]" if {"$preserve_selection" == 1} { memorize_isolist_selection } isodir_return "insert_from" if {"$preserve_selection" == 1} { restore_isolist_selection } browse_iso_refresh } # Copy a file out of the ISO image model to the hard disk filesystem. # The meta data stem from the ISO model tree. The content data are usually # read from the input drive. # Called when button "Extract to disk:" is hit. # proc extract_to {} { global extract_to_adr extract_from_selected extract_underneath global extract_auto_chmod global isodir_adr isolist_names if {[assert_iso_image 1] == 0} {return ""} if {"$extract_to_adr" == ""} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : You have to set a target address in the hard disk filesystem first" return "" } set sources "" set selected_ftype "" set selected_adr "" if {"$extract_from_selected" == 1} { set selected [.isolist curselection] if {[llength "$selected"] < 1} { xorriso_tcltk_errmsg "xorriso-tcltk : SORRY : You must select at least one ISO file as extraction source" return "" } foreach i "$selected" { set path [combine_dir_and_name "$isodir_adr" \ [lindex "$isolist_names" "$i"]] lappend sources "$path" } } else { set sources [list "$isodir_adr"] } # Warn of directory mergers. They can cause mess-up on hard disk. set merge_counter 0 set merge_source "" set merge_target "" foreach i "$sources" { if {[isofs_filetype "$i"] != "d"} { continue } if {"$extract_underneath" == 1} { set name [file tail "$i"] set target [combine_dir_and_name "$extract_to_adr" "$name"] } else { set target "$extract_to_adr" } if {[localfs_isdir "$target"] == 1} { incr merge_counter set merge_source "$i" set merge_target "$target" } } if {"$merge_counter" > 0} { if {"$sources" == "/" || "$sources" == "{}"} { if {[window_yesno "Really unpack ISO root directory into hard disk directory\n$merge_target\n?"] \ != 1} { return "" } } else { set and_others "" if {"$merge_counter" > 1} { set and_others "and do so at $merge_counter more occasions" } if {[window_yesno "Really unpack ISO directory\n [lindex "$sources" 0]\n into hard disk directory\n $merge_target\n$and_others ?"] \ != 1} { return "" } } } reset_highest_cmd_sev enforce_overwrite_settings "localfs" set disp_en_mem [set_display_msg 0] if {"$extract_auto_chmod" == 1} { send_marked_cmd "-osirrox on:sort_lba_on:auto_chmod_on" } else { send_marked_cmd "-osirrox on:sort_lba_off:auto_chmod_off" } set_display_msg "$disp_en_mem" foreach i "$sources" { if {"$extract_underneath" == 1} { set name [file tail "$i"] set target [combine_dir_and_name "$extract_to_adr" "$name"] } else { if {[llength "$sources"] != 1} { xorriso_tcltk_errmsg "xorriso-tcltk : SORRY : You must select exactly one ISO file as extraction source" return "" } set target "$extract_to_adr" } if {"$i" == ""} { set i "/" } send_marked_cmd "-extract [make_text_shellsafe "$i"] [make_text_shellsafe "$target"]" } browse_tree_populate "localfs" } # Send the currently chosen -overwrite settings of the checkbuttons # "Overwrite ISO files", "Overwrite ISO dirs", "Overwrite disk files". # Called before operations which could overwrite files in ISO model # or in the hard disk filesystem. # I.e. any -overwrite command sent via the "Command:" text field will not # be able to override the checkbuttons. # proc enforce_overwrite_settings {which_fs} { global overwrite_iso_files overwrite_iso_dirs overwrite_disk_files if {"$which_fs" == "isofs"} { if {"$overwrite_iso_files" == 0} { set mode "off" } else { if {"$overwrite_iso_dirs" == 0} { set mode "nondir" } else { set mode "on" } } } else { if {"$overwrite_disk_files" == 1} { set mode "on" } else { set mode "off" } } set disp_en_mem [set_display_msg 0] send_marked_cmd "-overwrite $mode" set_display_msg "$disp_en_mem" } # Send xorriso an appropriate end command and wait for the pipes to break. # Called when button "End" is hit. # proc end_xorriso {} { global expect_broken_pipes if {[window_yesno "Really end this program and its xorriso backend ?"] \ != 1} { return "" } if {[changes_are_pending] == 1} { if {[window_yesno \ "Changes of the ISO image are pending.\nReally discard them ?"] \ != 1} { return "" } set expect_broken_pipes "1" send_marked_cmd "-rollback_end" } else { set expect_broken_pipes "1" send_marked_cmd "-end" } central_exit 0 } # Check whether an output drive is aquired. Propose refusal if not. # Called by procedures which are about to use the output drive. # proc assert_outdev {purpose} { global outdev_adr if {"$outdev_adr" == ""} { xorriso_tcltk_errmsg \ "xorriso-tcltk : SORRY : You must choose an output drive before $purpose" return "0" } return "1" } # Check whether changes to the ISO model are pending. If so, propose refusal. # Called by procedures which are about to discard the ISO model. # proc assert_no_changes {} { if {[changes_are_pending] == 1} { window_ack "ISO image changes are pending. You have to do \"Write ISO session\" or \"Rollback\"." "grey" "toplevel" return "0" } return "1" } # ------ A primitive file tree browser for hard disk filesystem and ISO model # Write a directory content list into a Tree widget # proc browse_tree_fill_dir {tr parent children} { if {"$parent" == "/"} { set parent_name root } else { set parent_name "$parent" } if {[$tr exists "$parent_name"] == 0} {return ""} $tr delete [$tr nodes "$parent_name"] foreach i "$children" { set name [string range "$i" 2 end] set adr [combine_dir_and_name "$parent" "$name"] $tr insert end "$parent_name" "$adr" -text "$name" if {[string range "$i" 0 0] == "d"} { set dir_dummy [combine_dir_and_name "$adr" "dir_dummy"] $tr insert end "$adr" "$dir_dummy" -text "dir_dummy" } } } # The command to be executed when the user selects a node. # proc browse_tree_select {adr_var_name tr selected} { global browse_select_is_setvar if {"$browse_select_is_setvar" == 0} {return ""} browse_tree_accept "$adr_var_name" 0 "$tr" "$selected" } # The command to be executed when the user double-clicks a node. # proc browse_tree_accept {adr_var_name do_return tr selected} { global extract_to_adr insert_from_adr burn_write_image_adr isodir_adr global isomanip_move_target indev_adr outdev_adr if {[llength "$selected"] == 0} { set value "" } else { set value [lindex "$selected" 0] } if {"$adr_var_name" == "burn_write_image_adr"} { set burn_write_image_adr "$value" if {"$do_return" == 1} {burn_write_image} } if {"$adr_var_name" == "extract_to_adr"} { set extract_to_adr "$value" if {"$do_return" == 1} {extract_to} } if {"$adr_var_name" == "insert_from_adr"} { set insert_from_adr "$value" if {"$do_return" == 1} {insert_from} } if {"$adr_var_name" == "isodir_adr"} { set isodir_adr "$value" if {"$do_return" == 1} {isodir_return "browse_tree_select"} } if {"$adr_var_name" == "isomanip_move_target"} { set isomanip_move_target "$value" # No do_return because the field is shared between buttons } if {"$adr_var_name" == "indev_adr"} { set indev_adr "$value" if {"$do_return" == 1} {indev_return} } if {"$adr_var_name" == "outdev_adr"} { set outdev_adr "$value" if {"$do_return" == 1} {outdev_return} } } # Accept the single selected item of the tree browser # Called by the \"Accept\" button in the browser window. # proc browse_tree_accept_sel {adr_var_name do_return tr} { set selected [$tr selection get] if {[llength "$selected"] != 1} { xorriso_tcltk_errmsg "xorriso-tcltk : SORRY : You must select a single tree item clicking the \"Accept\" button." return "" } browse_tree_accept "$adr_var_name" "$do_return" "$tr" "$selected" } # Move up one directory level of the file browser selection # proc browse_tree_up {adr_var_name tr which_fs} { global extract_to_adr insert_from_adr burn_write_image_adr isodir_adr global isomanip_move_target indev_adr outdev_adr set selected [$tr selection get] if {[llength "$selected"] != 1} { xorriso_tcltk_errmsg "xorriso-tcltk : SORRY : You must select a single tree item clicking the \"Up\" button." return "" } set adr [file dirname [lindex "$selected" 0]] eval set mem $$adr_var_name set $adr_var_name "$adr" browse_tree_populate "$which_fs" set $adr_var_name "$mem" if {"$adr" != "/" && "$adr" != ""} { $tr selection clear $tr selection set "$adr" } } # Move down one directory level of the file browser selection # proc browse_tree_down {adr_var_name tr which_fs} { global extract_to_adr insert_from_adr burn_write_image_adr isodir_adr global isomanip_move_target indev_adr outdev_adr set selected [$tr selection get] if {[llength "$selected"] != 1} { xorriso_tcltk_errmsg "xorriso-tcltk : SORRY : You must select a single tree item clicking the \"Down\" button." return "" } set adr [lindex "$selected" 0] eval set mem $$adr_var_name set $adr_var_name "$adr" browse_tree_populate "$which_fs" set $adr_var_name "$mem" $tr selection set "$adr" } # The command to be executed when the user closes a directory node. # It replaces the directory content list by a single dummy item. # proc browse_tree_close_dir {tr name} { browse_tree_fill_dir $tr "$name" "{? dir_dummy}" } # Delete the old content of the browse window and display the freshly # obtained current state down to the current address in the field variable. # proc browse_tree_populate {which_fs} { global browse_disk_window_var browse_iso_window_var global browse_iso_window_is_active browse_disk_window_is_active global extract_to_adr insert_from_adr burn_write_image_adr isodir_adr global isomanip_move_target indev_adr outdev_adr if {"$which_fs" == "isofs"} { if {"$browse_iso_window_is_active" == 0} {return ""} set w {.browse_iso_window} set open_dir_cmd "browse_iso_open_dir" set adr_var "$browse_iso_window_var" } else { if {"$browse_disk_window_is_active" == 0} {return ""} set w {.browse_disk_window} set open_dir_cmd "browse_disk_open_dir" set adr_var "$browse_disk_window_var" } # Variable indirection eval set adr $$adr_var # Install root level $open_dir_cmd $w.tree "/" # Set $adr as current address set comps [split "$adr" "/"] # Install the stack of directories above current address set path "/" foreach i "$comps" { if {"$i" == ""} { continue } set path [combine_dir_and_name "$path" "$i"] $open_dir_cmd $w.tree "$path" catch { $w.tree opentree "$path" 0 $w.tree see "$path" } } } # The procedure to be run by mouse button 3 in the file browser. # It has to strip off the surplus parameter added by the Tree widget. # proc browse_tree_help {about_what button_color from_item} { window_help "$about_what" "$button_color" } # Destroy the hard disk browser pop-up window. # proc destroy_browse_disk {w} { global browse_disk_window_is_active if {"$w" != "" && "$browse_disk_window_is_active" == 1} { destroy "$w" } set browse_disk_window_is_active 0 } # The command to be executed when the user opens a directory node in # the hard disk filesystem. # proc browse_disk_open_dir {tr name} { if {[localfs_isdir "$name"] != 1} {return ""} set lslist [localfs_ls "$name"] browse_tree_fill_dir $tr "$name" "$lslist" } # Refresh the content of a possibly displayed tree browser for hard disk # proc browse_disk_refresh {} { browse_tree_populate "localfs" } # The command to be executed when the user opens a directory node in # the ISO model. # proc browse_iso_open_dir {tr name} { if {[isofs_filetype "$name"] != "d"} {return ""} set lslist [isofs_ls "$name"] browse_tree_fill_dir $tr "$name" "$lslist" } # Destroy the ISO browser pop-up window. # proc destroy_browse_iso {w} { global browse_iso_window_is_active if {"$w" != "" && "$browse_iso_window_is_active" == 1} { destroy "$w" } set browse_iso_window_is_active 0 } # Refresh the content of a possibly displayed tree browser for ISO model # proc browse_iso_refresh {} { browse_tree_populate "isofs" } # Open a file browser window for hard disk filesystem or ISO model # proc browse_tree {adr_var which_fs} { upvar $adr_var adr global have_bwidget browse_disk_window_is_active browse_iso_window_is_active global browse_disk_window_var browse_iso_window_var tree_window_lines set button_color "grey" set old_geometry "" if {"$which_fs" == "isofs"} { set w {.browse_iso_window} set window_is_active "$browse_iso_window_is_active" set title_name "xorriso-tcltk ISO filesystem browser" set open_dir_cmd "browse_iso_open_dir" set destroy_cmd "destroy_browse_iso" if {"$browse_iso_window_var" != "$adr_var" && "$window_is_active" == 1} { set old_geometry [wm geometry $w] destroy_browse_iso $w set window_is_active 0 } set browse_iso_window_var "$adr_var" set browse_iso_window_is_active 1 } else { set w {.browse_disk_window} set window_is_active "$browse_disk_window_is_active" set title_name "xorriso-tcltk disk filesystem browser" set open_dir_cmd "browse_disk_open_dir" set destroy_cmd "destroy_browse_disk" if {"$browse_disk_window_var" != "$adr_var" && "$window_is_active" == 1} { set old_geometry [wm geometry $w] destroy_browse_disk $w set window_is_active 0 } set browse_disk_window_var "$adr_var" set browse_disk_window_is_active 1 } set re_use_widgets 0 if {"$window_is_active" == 0} { toplevel $w -borderwidth 10 -class Browse wm title $w "$title_name" if {"$old_geometry" != ""} { wm geometry $w "$old_geometry" } } else { set re_use_widgets 1 } if {"$re_use_widgets" == 0} { # BWidget Tree frame $w.tree_frame frame $w.tree_frame_y Tree $w.tree -width 80 -height "$tree_window_lines" \ -opencmd "$open_dir_cmd $w.tree" \ -closecmd "browse_tree_close_dir $w.tree" \ -selectcommand "browse_tree_select $adr_var" \ -selectfill 1 \ -yscrollcommand "$w.treescroll_y set" \ -xscrollcommand "$w.treescroll_x set" # ??? why doesn't work ? # $w.tree bindText "browse_tree_accept $adr_var 1 $w.tree" # At least double-click does work $w.tree bindText "browse_tree_accept $adr_var 1 $w.tree" $w.tree bindText {browse_tree_help "Browse tree" grey} scrollbar $w.treescroll_y -command "$w.tree yview" scrollbar $w.treescroll_x -orient horizontal -command "$w.tree xview " pack $w.tree -in $w.tree_frame_y -side left -expand 1 -fill both pack $w.treescroll_y -in $w.tree_frame_y -side left -fill y pack $w.tree_frame_y $w.treescroll_x -in $w.tree_frame \ -side top -expand 1 -fill both frame $w.button_line set button_width 10 button $w.accept -text "Accept" -width "$button_width" \ -command "browse_tree_accept_sel $adr_var 1 $w.tree" bind_help $w.accept "Accept (browse tree)" button $w.to_field -text "To field" -width "$button_width" \ -command "browse_tree_accept_sel $adr_var 0 $w.tree" bind_help $w.to_field "To field (browse tree)" button $w.up -text "Up" -width "$button_width" \ -command "browse_tree_up $adr_var $w.tree $which_fs" bind_help $w.up "Up (browse tree)" button $w.down -text "Down" -width "$button_width" \ -command "browse_tree_down $adr_var $w.tree $which_fs" bind_help $w.down "Down (browse tree)" button $w.help -text "Help" -width "$button_width" \ -command {window_help "Browse tree" grey} bind_help $w.help "Browse tree" button $w.close -text "Close" -width "$button_width" \ -command "$destroy_cmd $w" \ -background "$button_color" bind_help $w.close "Close (browse tree)" pack $w.accept $w.to_field $w.up $w.down $w.help $w.close \ -in $w.button_line -side left -expand 1 -fill both pack $w.tree_frame $w.button_line -side top focus $w.tree } else { raise $w } browse_tree_populate "$which_fs" } # ------ GUI display procedures ---- # Display a message of xorriso or of this frontend in the .msglist box # proc display_msg {msg} { global .msglist global msglist_max_fill msglist_running pre_msglist display_msg_enabled if {"$display_msg_enabled" == 0} {return ""} if {"$msg" == "============================" || \ "$msg" == "==============================================================" || \ "$msg" == "enter option and arguments :"} {return ""} if {"$msglist_running" == 0} { lappend pre_msglist "$msg" } else { if {[.msglist index end] > "$msglist_max_fill"} { .msglist delete 0 0 } .msglist insert end "$msg" .msglist see [expr "[.msglist index end]-1"] update idletasks } } # Set whether messages submitted to proc display_message shall really show up # This is used by callback procedures to hide auxilliary commands and lengthy # reply messages from the user display. # proc set_display_msg {mode} { global display_msg_enabled set old "$display_msg_enabled" if {"$mode" == "0"} { set display_msg_enabled 0 } else { set display_msg_enabled "1" } return "$old" } # Display a frontend error message in the .msglist box and by a pop-up window. # >>> It would be nice to be able to wait for user confirmation. # proc xorriso_tcltk_errmsg {msg} { global highest_cmd_sev_msg set highest_cmd_sev_msg "$msg" display_msg "$msg" window_ack "$msg" "grey" "toplevel" update idletasks } # Memorize the current selection in the .isolist box. # proc memorize_isolist_selection {} { global memorized_isolist_selection isolist_names global .isolist set memorized_isolist_selection "" set selected [.isolist curselection] foreach i "$selected" { lappend memorized_isolist_selection [lindex $isolist_names "$i"] } } # Restore the memorized selection in the .isolist box as far as the # names have survived in the meantime. # proc restore_isolist_selection {} { global memorized_isolist_selection isolist_names global .isolist .isolist selection clear 0 end foreach i "$memorized_isolist_selection" { set idx [lsearch -exact "$isolist_names" "$i"] if {"$idx" > -1} { .isolist selection set "$idx" "$idx" } } set memorized_isolist_selection "" } # Receive the answer of the yes/no window and destroy it. # proc destroy_yesno {w answer} { global yesno_window_is_active answer_of_yesno if {"$w" != ""} { destroy "$w" } set yesno_window_is_active 0 set answer_of_yesno "$answer" } # Pop-up a window which asks for yes or no. Return 1 if answer is yes. # proc window_yesno {question} { global answer_of_yesno yesno_window_is_active set w {.yesno_window} if {"$yesno_window_is_active" == 1} { raise $w xorriso_tcltk_errmsg "xorriso-tcltk : SORRY : You still need to answer an older yes/no question" return "0" } set yesno_window_is_active 1 set answer_of_yesno "" toplevel $w -borderwidth 20 -class Dialog wm title $w "xorriso-tcltk yes/no" # wm geometry $w -0+0 label $w.question -text "$question" button $w.yes -text "yes" -command "destroy_yesno $w 1" \ -borderwidth 10 -padx 20 -pady 20 -relief ridge button $w.no -text "no" -command "destroy_yesno $w 0" \ -borderwidth 10 -padx 20 -pady 20 -relief ridge pack $w.yes $w.question $w.no -side left -expand 1 -fill both update idletasks tkwait variable answer_of_yesno return "$answer_of_yesno" } # Destroy the notification pop-up window. # proc destroy_ack {w} { global ack_window_is_active if {"$w" != ""} { destroy "$w" } set ack_window_is_active 0 } # Pop-up a window which notifies of a problem and asks for a button click. # proc window_ack {question button_color where} { global answer_of_yesno ack_window_is_active set re_use_widgets 0 if {"$where" == "embedded"} { set w "" set destroy_cmd "" } else { set w {.ack_window} if {"$ack_window_is_active" == 0} { toplevel $w -borderwidth 20 -class Dialog wm title $w "xorriso-tcltk acknowledge" set ack_window_is_active 1 } else { set re_use_widgets 1 } # wm geometry $w +0+0 set destroy_cmd "destroy_ack $w" } if {"$re_use_widgets" == 1} { $w.question configure -text "$question" raise $w } else { label $w.question -text "$question" button $w.ok -text "Continue" -command "$destroy_cmd" \ -background "$button_color" pack $w.question -side top -expand 1 -fill both pack $w.ok -side top } # >>> How to block any event but the "Continue" button ? } # Destroy the help pop-up window. # proc destroy_help {w} { global help_window_is_active help_window_has_scroll if {"$w" != ""} { destroy "$w" } set help_window_is_active 0 set help_window_has_scroll 0 } proc surround_text {text} { return "\n\n [string map {\n "\n "} "$text"]\n" } # Pop-up a window which shows a help text and a Close button. # proc window_help {about_what button_color} { global help_window_is_active help_window_lines help_window_has_scroll global help_window_border_width global .help_window set w {.help_window} # Giving the help text some distance from the border decorations set line_width 82 set helptext "\n\n [string map {\n "\n "} [tell_helptext "$about_what"]]\n" if {[count_newlines "$helptext"] >= "$help_window_lines"} { if {"$help_window_is_active" == 1 && "$help_window_has_scroll" == 0} { destroy_help $w } set help_window_has_scroll 1 } set re_use_widgets 0 if {"$help_window_is_active" == 0} { toplevel $w -borderwidth "$help_window_border_width" -class Help wm title $w "xorriso-tcltk help text" set help_window_is_active 1 } else { set re_use_widgets 1 } if {"$re_use_widgets" == 1} { $w.text configure -state normal $w.text delete 1.0 end $w.text insert end "$helptext" raise $w } else { # wm geometry $w +0+0 set destroy_cmd "destroy_help $w" frame $w.text_frame text $w.text -width "$line_width" -height "$help_window_lines" \ -relief flat -borderwidth 0 $w.text insert end "$helptext" pack $w.text -in $w.text_frame -side left -expand 1 -fill both if {"$help_window_has_scroll" == 1} { scrollbar $w.scroll_y -command "$w.text yview" $w.text configure -yscrollcommand "$w.scroll_y set" bind_listbox_keys $w.text "$help_window_lines" "text" pack $w.scroll_y -in $w.text_frame -side left -fill y } button $w.close -text "Close" -command "$destroy_cmd" \ -background "$button_color" pack $w.text_frame -side top -expand 1 -fill both frame $w.middle_spacer -height 6 frame $w.bottom_spacer -height 6 pack $w.middle_spacer $w.close $w.bottom_spacer -side top } $w.text configure -state disabled } # Display the busy/ready state of xorriso. # Called with 1 by sender of commands and with 0 by receivers of replies . # proc display_busy {state} { global busy_text_exists global .busy_text if {"$busy_text_exists" == 0} {return ""} if {"$state" == 0} { .busy_text configure -text "ready" .busy_text configure -background "#D0D0D0" } else { .busy_text configure -text "busy" .busy_text configure -background "#808080" } update idletasks } # Tries to make use of the BWidget package for getting its Tree widget # proc check_for_bwidget {} { global have_bwidget bwidget_version if {"$have_bwidget" == 0} { catch { set bwidget_version [package require BWidget] set have_bwidget 1 } } } proc browser_dummy {} { window_ack \ "The file browser cannot be used because Tcl/Tk package \"BWidget\" is not loaded" "grey" "toplevel" } # ------ Building GUI components ------ # ------ GUI layout parameters ------ # The default position of the main window set main_window_geometry "" # How to mark the borders of the main grouping frames set main_framerelief ridge set main_borderwidth 4 # How to mark the borders of the second level grouping frames set borderwidth 1 # Number of lines in msglist display set msglist_lines 8 set msglist_max_fill 1000 set msglist_running 0 # Number of lines in drivelist display set drivelist_lines 2 # Number of lines in ISO directory content display set isolist_lines 8 # Whether the message box shall export its selection to the whole X display set export_msg_selection true # Wether the item lists shall export their selection set export_selection false # The number of lines in the display of the help texts set help_window_lines 18 # The distance of the help text from the help window border set help_window_border_width 0 # The number of items to display in a tree browser window set tree_window_lines 12 # -------- GUI definition procedures # Overall definition of the GUI # proc init_gui {} { global .input .cmdline_entry .msgbox .errmsg .dev .drivebox global .isobox .localfs global main_framerelief main_borderwidth check_for_bwidget # Main grouping frames frame .connection_block \ -relief "$main_framerelief" -borderwidth "$main_borderwidth" frame .drive_block \ -relief "$main_framerelief" -borderwidth "$main_borderwidth" frame .iso_block \ -relief "$main_framerelief" -borderwidth "$main_borderwidth" init_input init_msgbox init_errmsg init_dev init_drivebox init_isobox init_isomanip init_burn init_localfs pack .input .msgbox .errmsg -in .connection_block \ -side top -expand 1 -fill both pack .drivebox .dev .burn -in .drive_block \ -side top -expand 1 -fill both pack .localfs .isobox .isomanip -in .iso_block \ -side top -expand 1 -fill both pack .connection_block .drive_block .iso_block \ -side top -expand 1 -fill both focus .cmdline_entry } # The xorriso headline with End button, xorriso version, busy/ready indicator, # command line, and "Refresh state display" button. # proc init_input {} { global borderwidth busy_text_exists xorriso_version logging global .input .input_line1 .xorriso_version .busy .busy_text global .refresh_state .end_button .cmdline .log_pipes_switch frame .input -borderwidth $borderwidth frame .input_line1 -borderwidth 0 pack .input_line1 -in .input \ -side top -anchor w -expand 1 -fill both button .end_button -text "End" -command "end_xorriso" bind_help .end_button "End" if {[string length "$xorriso_version"] > 10} { set xorriso_version [string range "$xorriso_version" 0 9] } label .xorriso_version -text "xorriso-$xorriso_version" bind_help .xorriso_version "version" frame .busy -relief ridge -borderwidth 2 label .busy_text -width 5 -text "busy" bind_help .busy_text "ready/busy" set busy_text_exists 1 pack .busy_text -in .busy button .refresh_state -text "Refresh state display" \ -command "refresh_state" bind_help .refresh_state "Refresh state display" checkbutton .log_pipes_switch -text "Log pipes" \ -indicatoron 1 -selectcolor "" \ -relief ridge -borderwidth 2 \ -variable logging \ -onvalue 1 -offvalue 0 bind_help .log_pipes_switch "Log pipes" button .help -text "Help" -command {window_help "Help" "grey"} bind_help .help "Help" init_cmdline pack .end_button .xorriso_version .busy -in .input_line1 -side left pack .cmdline \ -in .input_line1 -side left -expand 1 -fill both pack .refresh_state .log_pipes_switch .help -in .input_line1 -side left } # The combination of "Command:" label and command line # proc init_cmdline {} { global cmdline borderwidth global .cmdline .cmdline_text .cmdline_entry frame .cmdline -borderwidth 0 label .cmdline_text -width 10 -text "Command:" bind_help .cmdline_text "Command:" entry .cmdline_entry -width 60 -relief sunken -bd 1 \ -textvariable cmdline bind_entry_keys .cmdline_entry {cmdline_return} bind_help .cmdline_entry "Command:" # >>> is there a chance to get a history on an entry ? pack .cmdline_text -in .cmdline -side left pack .cmdline_entry -in .cmdline -side left -expand 1 -fill both } # The listbox where to display commands and reply messages unless this is # disabled for auxiliary commands which shall not clutter the display. # proc init_msgbox {} { global borderwidth global msglist_lines export_msg_selection msglist_running pre_msglist global .msgbox .msglist .msgscroll frame .msgbox -borderwidth $borderwidth listbox .msglist -height $msglist_lines -selectmode extended \ -yscrollcommand ".msgscroll set" \ -exportselection $export_msg_selection bind_listbox_keys ".msglist" "$msglist_lines" "listbox" bind_help .msglist "message box" set msglist_running 1 foreach i "$pre_msglist" { display_msg "$i" } scrollbar .msgscroll -command ".msglist yview" pack .msglist -in .msgbox -side left -expand 1 -fill both pack .msgscroll -in .msgbox -side right -fill y set pre_msglist "" } # Two display lines for most severe event messages. One gets reset with # each important command. The other one stays until the user clears it. # proc init_errmsg {} { global borderwidth global .errmsg .total_errmsg .cmd_errmsg frame .errmsg -borderwidth $borderwidth init_total_errmsg init_cmd_errmsg pack .cmd_errmsg .total_errmsg -in .errmsg \ -side top -anchor w -expand 1 -fill both } # The most severe message display which gets reset automatically. # proc init_cmd_errmsg {} { global borderwidth global .cmd_errmsg .cmd_errmsg_label .cmd_errmsg_msg frame .cmd_errmsg -borderwidth "$borderwidth" label .cmd_errmsg_label -width 14 -text "Recent problem:" -anchor w bind_help .cmd_errmsg_label "Recent problem:" label .cmd_errmsg_msg -width 80 -relief ridge -bd 2 \ -anchor w \ -textvariable highest_cmd_sev_msg # (no keys, no focus) bind_help .cmd_errmsg_msg "Recent problem:" pack .cmd_errmsg_label -in .cmd_errmsg -side left pack .cmd_errmsg_msg -in .cmd_errmsg -side left -expand 1 -fill both } # The persistent most severe message display that is to be reset by the user. # proc init_total_errmsg {} { global borderwidth global .total_errmsg .total_errmsg_label .total_errmsg_msg global .total_errmsg_clear frame .total_errmsg -borderwidth "$borderwidth" label .total_errmsg_label -text "Worst problem:" -width 14 -anchor w bind_help .total_errmsg_label "Worst problem:" button .total_errmsg_clear -text "Clear" \ -width 5 \ -command "clear_total_errmsg" bind_help .total_errmsg_clear "Clear" label .total_errmsg_msg -width 80 -relief ridge -bd 2 \ -anchor w \ -textvariable highest_total_sev_msg # (no keys, no focus) bind_help .total_errmsg_msg "Worst problem:" pack .total_errmsg_label -in .total_errmsg -side left pack .total_errmsg_msg -in .total_errmsg -side left -expand 1 -fill both pack .total_errmsg_clear -in .total_errmsg -side left } # The list of drives which were found by scanning, the Scan button, and # buttons for picking a drive from the list, for giving them up, for # calming them down, and for reloading the ISO image from the input drive. # proc init_drivebox {} { global borderwidth drivelist_lines export_selection global .drivebox .drivelistbox .drivelist .drivescroll .drive_scan global .drive_picker .drive_scan .drive_pick_in .drive_pick_out global .drive_pick_both .drive_drop_both .drive_calm .iso_rollback_button frame .drivebox -borderwidth $borderwidth frame .drivelistbox -borderwidth $borderwidth listbox .drivelist -height $drivelist_lines -selectmode extended \ -yscrollcommand ".drivescroll set" \ -exportselection $export_selection bind_listbox_keys ".drivelist" "$drivelist_lines" "listbox" bind_help .drivelist "drivelist" scrollbar .drivescroll -command ".drivelist yview" pack .drivelist -in .drivelistbox -side left -expand 1 -fill both pack .drivescroll -in .drivelistbox -side right -fill y frame .drive_picker -borderwidth $borderwidth button .drive_scan -text "Scan for drives" \ -command "scan_for_drives" bind_help .drive_scan "Scan for drives" button .drive_pick_in -text "Pick input drive" \ -command "pick_indev" bind_help .drive_pick_in "Pick input drive" button .drive_pick_out -text "Pick output drive" \ -command "pick_outdev" bind_help .drive_pick_out "Pick output drive" button .drive_pick_both -text "Pick drive for both roles" \ -command "pick_dev" bind_help .drive_pick_both "Pick drive for both roles" button .drive_drop_both -text "Give up drives" \ -command "give_up_dev" bind_help .drive_drop_both "Give up drives" button .drive_calm -text "Calm drives" \ -command "calm_drives" bind_help .drive_calm "Calm drives" button .iso_rollback_button -text "Rollback" -command {iso_rollback} bind_help .iso_rollback_button "Rollback" pack .drive_scan \ .drive_pick_in .drive_pick_out .drive_pick_both \ .drive_drop_both .drive_calm .iso_rollback_button \ -in .drive_picker -side left -expand 1 -fill none pack .drive_picker -in .drivebox \ -side top -expand 1 -fill none pack .drivelistbox -in .drivebox \ -side top -expand 1 -fill both bind .drivelist { pick_dev } } # The text fields for setting and display of the current input and output # drives. With Eject button and a short text description of the medium status. # proc init_dev {} { global borderwidth global .dev .indev .outdev frame .dev -borderwidth $borderwidth init_indev init_outdev pack .indev .outdev -in .dev \ -side top -anchor w -expand 1 -fill both } # Set and display the current input drive. # proc init_indev {} { global borderwidth indev_adr global .indev .indev_eject .indev_label .indev_entry .indev_summary frame .indev -borderwidth $borderwidth button .indev_eject -text "Eject" -command {eject_indev} bind_help .indev_eject "Eject (indev)" label .indev_label -width 22 -text "Input drive or image " bind_help .indev_label "Input drive or image" entry .indev_entry -width 40 -relief sunken -bd 1 \ -textvariable indev_adr bind_entry_keys .indev_entry {indev_return} bind_help .indev_entry "Input drive or image" label .indev_summary -width 65 -text "" -relief ridge -borderwidth 2 bind_help .indev_summary "input drive info" create_browser_button .indev_browse_button \ "indev_adr" "localfs" "Browse disk (indev)" pack .indev_eject .indev_label .indev_entry \ -in .indev -side left -expand 1 -fill both pack .indev_browse_button -in .indev -side left pack .indev_summary \ -in .indev -side left -expand 1 -fill both } # Set and display the current output drive. # proc init_outdev {} { global .outdev .outdev_eject .outdev_label .outdev_entry .outdev_summary global borderwidth outdev_adr frame .outdev -borderwidth $borderwidth button .outdev_eject -text "Eject" -command {eject_outdev} bind_help .outdev_eject "Eject (outdev)" label .outdev_label -width 22 -text "Output drive or image" bind_help .outdev_label "Output drive or image" entry .outdev_entry -width 40 -relief sunken -bd 1 \ -textvariable outdev_adr bind_entry_keys .outdev_entry {outdev_return} bind_help .outdev_entry "Output drive or image" create_browser_button .outdev_browse_button \ "outdev_adr" "localfs" "Browse disk (outdev)" label .outdev_summary -width 65 -text "" -relief ridge -borderwidth 2 bind_help .outdev_summary "output drive info" pack .outdev_eject .outdev_label .outdev_entry \ -in .outdev -side left -expand 1 -fill both pack .outdev_browse_button -in .outdev -side left pack .outdev_summary \ -in .outdev -side left -expand 1 -fill both } # The button panel for blanking, formatting, and writing to the output drive. # proc init_burn {} { global borderwidth burn_write_image_adr burn_write_close burn_write_tao global burn_write_defect_mgt global .burn .burn_blank_button .burn_format_button .burn_commit_button global .burn_write_image .burn_write_image_entry .burn_write_close global .burn_write_tao .burn_write_defect_mgt frame .burn -borderwidth $borderwidth button .burn_blank_button -text "Blank" \ -command {burn_blank} bind_help .burn_blank_button "Blank" button .burn_format_button -text "Format" \ -command {burn_format} bind_help .burn_format_button "Format" button .burn_commit_button -text "Write ISO session" \ -command {burn_commit} bind_help .burn_commit_button "Write ISO session" button .burn_write_image -text "Burn image file:" \ -command {burn_write_image} bind_help .burn_write_image "Burn image file:" entry .burn_write_image_entry -width 40 -relief sunken -bd 1 \ -textvariable burn_write_image_adr bind_entry_keys .burn_write_image_entry {burn_write_image} bind_help .burn_write_image_entry "Burn image file:" create_browser_button .burn_image_browse_button \ "burn_write_image_adr" "localfs" "Browse disk (burn image)" checkbutton .burn_write_close -text "Close" \ -indicatoron 1 -selectcolor "" \ -relief ridge -borderwidth 2 \ -variable burn_write_close \ -onvalue 1 -offvalue 0 bind_help .burn_write_close "Close" checkbutton .burn_write_tao -text "TAO" \ -indicatoron 1 -selectcolor "" \ -relief ridge -borderwidth 2 \ -variable burn_write_tao \ -onvalue 1 -offvalue 0 bind_help .burn_write_tao "TAO" checkbutton .burn_write_defect_mgt -text "Defect Mgt" \ -indicatoron 1 -selectcolor "" \ -relief ridge -borderwidth 2 \ -variable burn_write_defect_mgt \ -onvalue 1 -offvalue 0 bind_help .burn_write_defect_mgt "Defect Mgt" pack .burn_blank_button .burn_format_button \ .burn_commit_button .burn_write_close .burn_write_tao \ .burn_write_defect_mgt \ .burn_write_image .burn_write_image_entry \ -in .burn -side left -expand 1 -fill both pack .burn_image_browse_button -in .burn -side left } # Set and display the current ISO directory and its content. # proc init_isobox {} { global borderwidth isolist_lines export_selection global .isobox .isodir .isolist .isodir_entry .isodir_up .isodir_up2 global .isodir_label .isodir_verify .isolistbox .isoscroll_y .isoscroll_x frame .isobox -borderwidth $borderwidth frame .isodir -borderwidth 0 label .isodir_label -text "ISO directory:" \ -width 14 bind_help .isodir_label "ISO directory:" entry .isodir_entry -width 60 -relief sunken -bd 1 \ -textvariable isodir_adr bind_entry_keys .isodir_entry {isodir_return "isodir_entry"} bind_help .isodir_entry "ISO directory:" create_browser_button .isodir_browse_button \ "isodir_adr" "isofs" "Browse ISO (isodir)" button .isodir_verify -text "Verify" -command {isodir_verify} bind_help .isodir_verify "Verify" button .isodir_up -text "Up" -command {isodir_up} bind_help .isodir_up "Up" button .isodir_up2 -text "Up" -command {isodir_up} bind_help .isodir_up2 "Up" pack .isodir_label .isodir_up \ -in .isodir -side left pack .isodir_entry \ -in .isodir -side left -expand 1 -fill both pack .isodir_browse_button .isodir_up2 .isodir_verify \ -in .isodir -side left frame .isolistbox -borderwidth 0 listbox .isolist -height $isolist_lines -selectmode extended \ -yscrollcommand ".isoscroll_y set" \ -xscrollcommand ".isoscroll_x set" \ -exportselection $export_selection bind_help .isolist "isolist" bind_listbox_keys ".isolist" "$isolist_lines" "listbox" scrollbar .isoscroll_y -command ".isolist yview" scrollbar .isoscroll_x -orient horizontal -command ".isolist xview" pack .isolist -in .isolistbox -side left -expand 1 -fill both bind .isolist { pick_isodir } pack .isoscroll_y -in .isolistbox -side right -fill y pack .isodir .isolistbox .isoscroll_x \ -in .isobox -side top -expand 1 -fill both } # The ISO-internal manipulation buttons for the ISO directory or its content. # Plus a text field where to set an ISO path as target for renaming or # directory making. # proc init_isomanip {} { global borderwidth isomanip_move_target global .isomanip .isomanip_move .isomanip_prefix .isomanip_verify_button global .isomanip_move_target .isomanip_rm_r_button .isomanip_move_button global .isomanip_mkdir_button .isomanip_move_target frame .isomanip -borderwidth $borderwidth frame .isomanip_move -borderwidth 0 label .isomanip_prefix -text "ISO selection:" bind_help .isomanip_prefix "ISO selection:" button .isomanip_verify_button -width 8 -text "Verify" \ -command {isomanip_verify} bind_help .isomanip_verify_button "Verify (selection)" button .isomanip_rm_r_button -width 8 -text "Delete" \ -command {isomanip_rm_r} bind_help .isomanip_rm_r_button "Delete" button .isomanip_move_button -text "Rename to:" \ -command {isomanip_mv} bind_help .isomanip_move_button "Rename to:" button .isomanip_mkdir_button -width 8 -text "Make dir" \ -command {isomanip_mkdir} bind_help .isomanip_mkdir_button "Make dir" entry .isomanip_move_target -width 60 -relief sunken -bd 1 \ -textvariable isomanip_move_target # bind_entry_keys .isomanip_move_target {isomanip_mv} bind_entry_keys .isomanip_move_target "" bind_help .isomanip_move_target "rename and mkdir target" create_browser_button .isomanip_move_target_button \ "isomanip_move_target" "isofs" "Browse ISO (move target)" pack .isomanip_prefix .isomanip_verify_button .isomanip_rm_r_button \ .isomanip_move_button .isomanip_move_target \ -in .isomanip_move -side left -expand 1 -fill both pack .isomanip_move_target_button -in .isomanip_move -side left pack .isomanip_mkdir_button \ -in .isomanip_move -side left -expand 1 -fill both pack .isomanip_move \ -in .isomanip -side top -expand 1 -fill both } # The means for interaction of ISO image and hard disk filesystem. # proc init_localfs {} { global borderwidth global .localfs .extract_frame .localfs_aux_frame .insert_frame frame .localfs -borderwidth $borderwidth init_extract init_localfs_aux init_insert pack .extract_frame .localfs_aux_frame .insert_frame \ -in .localfs -side top -expand 1 -fill both } # The means for extracting files from ISO image to disk # proc init_extract {} { global borderwidth extract_to_adr extract_from_selected extract_underneath global .extract_button .extract_frame .extract_entry .extract_from_selected global .extract_underneath frame .extract_frame -borderwidth 0 button .extract_button -text "Extract to disk:" \ -width 17 \ -command {extract_to} bind_help .extract_button "Extract to disk:" entry .extract_entry -width 40 -relief sunken -bd 1 \ -textvariable "extract_to_adr" bind_entry_keys .extract_entry {extract_to} bind_help .extract_entry "Extract to disk:" create_browser_button .extract_browse_button \ "extract_to_adr" "localfs" "Browse disk (extract)" checkbutton .extract_underneath -text "Underneath" \ -indicatoron 1 -selectcolor "" \ -relief ridge -borderwidth 2 \ -variable extract_underneath \ -onvalue 1 -offvalue 0 bind_help .extract_underneath "Underneath (extract)" checkbutton .extract_from_selected -text "Selected" \ -indicatoron 1 -selectcolor "" \ -relief ridge -borderwidth 2 \ -variable extract_from_selected \ -onvalue 1 -offvalue 0 bind_help .extract_from_selected "Selected (extract)" pack .extract_button -in .extract_frame -side left pack .extract_entry \ -in .extract_frame -side left -expand 1 -fill both pack .extract_from_selected .extract_underneath \ -in .extract_frame -side right pack .extract_browse_button -in .extract_frame -side right } # Some controls which apply to insertion, extraction, or both. # proc init_localfs_aux {} { global borderwidth have_bwidget global .localfs_aux_frame global .overwrite_iso_files_button .overwrite_dir_button .extract_auto_chmod global .avail_label .avail_label_frame .avail_button frame .localfs_aux_frame -borderwidth 0 checkbutton .overwrite_iso_files_button -text "Overwrite ISO files" \ -indicatoron 1 -selectcolor "" \ -relief ridge -borderwidth 2 \ -variable overwrite_iso_files \ -onvalue 1 -offvalue 0 bind_help .overwrite_iso_files_button "Overwrite ISO files" checkbutton .overwrite_dir_button -text "Overwrite ISO dirs" \ -indicatoron 1 -selectcolor "" \ -relief ridge -borderwidth 2 \ -variable overwrite_iso_dirs \ -onvalue 1 -offvalue 0 bind_help .overwrite_dir_button "Overwrite ISO dirs" checkbutton .extract_auto_chmod -text "Enforce disk dir write access" \ -indicatoron 1 -selectcolor "" \ -relief ridge -borderwidth 2 \ -variable extract_auto_chmod \ -onvalue 1 -offvalue 0 bind_help .extract_auto_chmod "Enforce disk dir write access" checkbutton .overwrite_disk_files_button -text "Overwrite disk files" \ -indicatoron 1 -selectcolor "" \ -relief ridge -borderwidth 2 \ -variable overwrite_disk_files \ -onvalue 1 -offvalue 0 bind_help .overwrite_disk_files_button "Overwrite disk files" pack .overwrite_iso_files_button .overwrite_dir_button .extract_auto_chmod \ .overwrite_disk_files_button \ -in .localfs_aux_frame -side left if {"$have_bwidget" == 1} { checkbutton .browse_select_is_setvar -text "File browser text field" \ -indicatoron 1 -selectcolor "" \ -relief ridge -borderwidth 2 \ -variable "browse_select_is_setvar" \ -onvalue 1 -offvalue 0 bind_help .browse_select_is_setvar "File browser text field" pack .browse_select_is_setvar -in .localfs_aux_frame -side left } button .avail_button -text "Refresh avail:" \ -command {refresh_avail} bind_help .avail_button "Refresh avail:" frame .avail_label_frame -relief ridge -borderwidth 2 label .avail_label -width 10 -text "" bind_help .avail_label "Refresh avail:" pack .avail_label -in .avail_label_frame pack .avail_label_frame .avail_button -in .localfs_aux_frame -side right } # The means for inserting files from disk into the ISO image # proc init_insert {} { global borderwidth insert_from_adr insert_at_selected insert_underneath global .insert_button .insert_from_frame .insert_entry .insert_at_selected global .insert_underneath .insert_frame frame .insert_frame -borderwidth 0 frame .insert_from_frame -borderwidth 0 button .insert_button -text "Insert from disk:" \ -width 17 \ -command {insert_from} bind_help .insert_button "Insert from disk:" entry .insert_entry -width 40 -relief sunken -bd 1 \ -textvariable "insert_from_adr" bind_entry_keys .insert_entry {insert_from} bind_help .insert_entry "Insert from disk:" create_browser_button .insert_browse_button \ "insert_from_adr" "localfs" "Browse disk (insert)" checkbutton .insert_underneath -text "Underneath" \ -indicatoron 1 -selectcolor "" \ -relief ridge -borderwidth 2 \ -variable insert_underneath \ -onvalue 1 -offvalue 0 bind_help .insert_underneath "Underneath (insert)" checkbutton .insert_at_selected -text "Selected" \ -indicatoron 1 -selectcolor "" \ -relief ridge -borderwidth 2 \ -variable insert_at_selected \ -onvalue 1 -offvalue 0 bind_help .insert_at_selected "Selected (insert)" pack .insert_button -in .insert_from_frame -side left pack .insert_entry \ -in .insert_from_frame -side left -expand 1 -fill both pack .insert_browse_button -in .insert_from_frame -side left pack .insert_at_selected .insert_underneath \ -in .insert_from_frame -side right pack .insert_from_frame -in .insert_frame -side left -expand 1 -fill both } # Set common behavior of listboxes in respect to focus and navigation keys. # proc bind_listbox_keys {box height what_widget} { global click_to_focus if {"$click_to_focus" == 1} { bind $box <1> "focus \"$box\"" bind $box <2> "focus \"$box\"" bind $box <3> "focus \"$box\"" } else { bind $box "focus \"$box\"" } # No underlining if {"$what_widget" == "listbox"} { $box configure -activestyle "none" } # Need to evaluate all $box and $height at bind-time. Thus "-quotes. bind $box " if {\"%K\" == \"Up\"} { $box yview scroll \"-1\" units } if {\"%K\" == \"Down\"} { $box yview scroll 1 units } if {\"%K\" == \"Prior\"} { $box yview scroll -[expr \"$height\" - 1] units # $box yview scroll -1 pages } if {\"%K\" == \"Next\"} { $box yview scroll [expr \"$height\" - 1] units # $box yview scroll 1 pages } if {\"%K\" == \"Home\"} { $box yview 0 } if {\"%K\" == \"End\"} { $box yview end } # >>> Do i need this ? # >>> For now: yes. It prevents double scrolling by PgUp PgDown # Prevent other bindings from being performed break " } # Set common behavior of entries in respect to focus and Return key. # proc bind_entry_keys {entry return_cmd} { global click_to_focus if {"$click_to_focus" != 1} { bind $entry "focus \"$entry\"" } if {"$return_cmd" != ""} { bind $entry "$return_cmd" } } # Bind a help text to a widget. # proc bind_help {to_what help_name} { bind $to_what "window_help \"$help_name\" grey" } # Create a "/" button and wire it with variable, fileystem type, and help text. # proc create_browser_button {button_name var_name which_fs help_name} { global have_bwidget if {"$have_bwidget" == 1} { button $button_name -text "/" -command "browse_tree $var_name $which_fs" bind_help $button_name "$help_name" } else { button $button_name -text "/" -command "browser_dummy" if {"$which_fs" == "localfs"} { bind_help $button_name "Browse disk (dummy)" } else { bind_help $button_name "Browse ISO (dummy)" } } } # The central storage for help texts. # proc tell_helptext {what} { global own_version argv0 bwidget_version if {"$what" == "Help"} { return \ "For getting particular help texts: Click the rightmost mouse button on any button, list box, or text field. For a help text about startup options of this frontend, execute in a shell: $argv0 --help For background info about xorriso and its commands, execute in a shell: man xorriso ----------------------------------------------------------------------------- The GUI window is separated into three main areas: - The area for connection to xorriso - shows xorriso messages, - offers some general activities, - displays the \"ready/busy\" state of the connection, - and allows to toggle xorriso commands into the \"Command:\" field. - The area for management of drives and ISO image data files - allows to scan for optical drives, - to aquire them and load their ISO directory tree, - to aquire ISO image files from hard disk as pseudo drives like DVD+RW, - to blank CD-RW, DVD=RW, DVD+RW, BD-RE and format DVD-RW, BD-R, - to trigger writing of ISO sessions (which get defined in the third area), - and to burn image data files from hard disk to optical media. - The area for inspection, manipulation, and exploitation of the ISO model - allows to insert directories and files from hard disk into the ISO model, - to delete and rename file objects in the ISO model, - to verify data files of loaded ISO directory trees by MD5, - to extract directories and files from ISO filesystem to hard disk. ----------------------------------------------------------------------------- Some Use Cases ----------------------------------------------------------------------------- - Burn a directory as only content onto a CD, DVD or BD - Write a directory as only content to an ISO image data file on hard disk - Burn an image data file from hard disk onto CD, DVD or BD - Add more data to an appendable medium or to an ISO image data file - Extract a directory tree from an ISO filesystem to hard disk ----------------------------------------------------------------------------- Burn a directory as only content onto a CD, DVD or BD - Click the \"Scan for drives\" button in the middle area. - Select a drive and click the \"Pick output drive\" button. - If the information field in the \"Output drive or Image\" line begins by \"appendable\" or \"closed\" and if the medium is CD-RW, DVD-RW, DVD+RW, or BD-RE then click the \"Blank\" button to erase the old data. (Blanking of \"DVD-RW sequential recording\" will last very long.) - Go to the \"Insert from disk:\" line in the lower area. Either toggle in the address of the hard disk directory, or click on the \"/\" button to the right of the text field to get a file browser. - Hit the Return key or click on a name in the browser to schedule the disk directory for writing to the medium. (You may of course insert several directories or files that way.) - Click the \"Write ISO session\" button in the middle area. Confirm in the \"yes/no\" window that pops up. Burning will begin (or refuse on unsuitable medium status). - When the \"busy\" field displays \"ready\" again, you may click \"Eject\". Desktop drives should then put out the tray with the medium. ----------------------------------------------------------------------------- Write a directory as only content to an ISO image data file on hard disk - Toggle the name of the intended data file in the text field of the \"Output drive or image\" line and hit the Return key. - If the information field in the \"Output drive or Image\" line begins by \"appendable\" or \"closed\" then you addressed an existing data file. Warning: Applying the \"Blank\" button to it would damage its content ! You probably do not want this in this special use case. - Go to the \"Insert from disk:\" line in the lower area. Continue like in the above description for CD, DVD, and BD media. ----------------------------------------------------------------------------- Burn an image data file from hard disk onto CD, DVD or BD - Click the \"Scan for drives\" button in the middle area. - Select a drive and click the \"Pick output drive\" button. - If the information field in the \"Output drive or Image\" line begins by \"appendable\" or \"closed\" and if the medium is CD-RW, DVD-RW, DVD+RW, or BD-RE then click the \"Blank\" button to erase the old data. (Blanking of \"DVD-RW sequential recording\" will last very long.) - Go to the text field beneath the \"Burn image file:\" button and toggle the address of the image file. Or click on the \"/\" button to the right of the field to get a file browser. - Hit the Return key in the text field or click on a name in the browser to initiate the burn run. Confirm in the \"yes/no\" window that pops up. - When the \"busy\" field displays \"ready\" again, you may click \"Eject\". Desktop drives should then put out the tray with the medium. ----------------------------------------------------------------------------- Add more data to an appendable medium or to an ISO image data file - Like above, \"Scan for drives\" but click button \"Pick drive for both roles\" in order to load the directory tree of the existing ISO filesystem. For an ISO image data file, bring its name into the input fields of both lines \"Input drive or image\" and \"Output drive or image\". You should now see in both info fields texts which begin by \"appendable\". - Go to the \"Insert from disk:\" line in the lower area. Use the means described in the first use case to add more directories or data files. - If you are interested in \"Delete\" or \"Rename to:\" buttons in the bottom line of the GUI: Click them by the rightmost mouse button to see their help texts. - When all intended changes are done: Click \"Write ISO session\" and confirm in the \"yes/no\" window. ----------------------------------------------------------------------------- Extract a directory tree from an ISO filesystem to hard disk - Like above, \"Scan for drives\" but click button \"Pick input drive\" in order to load the directory tree of the existing ISO filesystem. For an ISO image data file, bring its name into the input field of the lines \"Input drive or image\" and \"Output drive or image\". You should now see in both info fields texts which begin by \"appendable\". - Go to the \"ISO directory:\" line in the lower area. Either toggle in the address of the directory you want to extract or click the \"/\" button to get a file browser. - Hit the Return key or click on a name in the browser to determine the ISO directory for writing to the medium. - Go to the \"Extract to disk:\" line in the lower area. Either toggle in the address of the hard disk directory, or click on the \"/\" button to the right of the text field to get a file browser. - Hit the Return key or click on a name in the browser to initiate the extraction run. If a \"yes/no\" window pops up, consider well whether you are up to to shooting your own foot right now. Enable the \"Overwrite disk files\" switch only if you are really sure that the files from ISO are better than the ones on hard disk. ----------------------------------------------------------------------------- xorriso-tcltk is mainly a proof of concept for a frontend that operates xorriso in dialog mode. It demonstrates some of xorriso's multi-session features with ISO 9660 filesystems on optical media (CD, DVD, BD) or in disk files. Dependencies: xorriso, Tcl language, Tk GUI toolkit, optionally Tcl/Tk package BWidget Copyright (C) 2012, Thomas Schmitt , libburnia-project.org Provided under GNU GPL version 2 or later." } if {"$what" == "End"} { return \ "The \"End\" button leads to the end of frontend and xorriso process." } if {"$what" == "version"} { return \ "The field between \"End\" button and ready/busy field displays the version of the serving xorriso program. xorriso is a program which copies file objects from POSIX compliant filesystems into Rock Ridge enhanced ISO 9660 filesystems and allows session-wise manipulation of such filesystems. It can load the management information of existing ISO images and it writes the session results to optical media or to filesystem objects. Vice versa xorriso is able to restore file objects from ISO 9660 filesystems. xorriso-tcltk-$own_version is mainly a proof of concept for a frontend that operates xorriso in dialog mode. It exercises several fundamental gestures of communication: - connecting via two pipes - sending commands - receiving replies - inquiring the xorriso message sieve - using the xorriso parsing service Note that any other language than Tcl/Tk could be used, if it only can do i/o via standard input and standard output or via named pipes. Further it has to perform integer arithmetics and string manipulations. And, well, a graphical widget set would be nice." } if {"$what" == "Refresh state display"} { return \ "The \"Refresh state display\" button causes several text fields and list boxes to update their display after manually transmitted commands may have changed the state of drives or ISO model." } if {"$what" == "ready/busy"} { return \ "The ready/busy field indicates whether a xorriso command is being executed and the frontend is still waiting for its reply messages." } if {"$what" == "Command:"} { return \ "The \"Command:\" field can be used to send commands to xorriso. See the manual page of xorriso for its concepts and commands." } if {"$what" == "Refresh state display"} { return \ "The \"Refresh state display\" button causes the other two main areas to update their display after manually transmitted commands may have changed the state of drives or ISO model." } if {"$what" == "Log pipes"} { return \ "The \"Log pipes\" switch controls whether all xorriso commands and replies shall be logged to standard error resp. to the file that has been given with program argument --log_file. Caution: This log is verbous." } if {"$what" == "message box"} { return \ "The message box displays commands sent to xorriso and messages received from xorriso. Many commands which are emitted by the GUI will hide themselves and their replies from this display. All event messages with severity WARNING or higher will show up, nevertheless." } if {"$what" == "Recent problem:"} { return \ "The \"Recent problem:\" field shows the most severe event message that occured during the execution of the most recent command. It also displays the most recent problem message from the frontend program itself. Several commands emitted by the GUI will not clear this display. But any manually transmitted command and the major GUI gestures will do. " } if {"$what" == "Worst problem:"} { return \ "The \"Worst problem:\" field shows the most severe event message that occured since last time the \"Clear\" button was hit. It will not clear automatically." } if {"$what" == "Clear"} { return \ "The \"Clear\" button removes the message from the \"Worst problem:\" field." } if {"$what" == "Scan for drives"} { return \ "The \"Scan for drives\" button executes command -devices and puts the list of found optical drives into the box underneath the button. Scanning should be done before any ISO image manipulations because xorriso has to give up its aquired drives in order to perform the scan run. To become visible and to be usable, the drives have to offer rw-permission to the user of this program. If drives do not show up, then consider to become superuser and to execute xorriso -devices Then apply chmod a+rw to the listed device files. (Consider to use finer means of permission granting for a permanent solution.)" } if {"$what" == "Pick input drive"} { return \ "The \"Pick input drive\" button executes command -indev and obtains some information about the medium status. This info is displayed in the \"Input drive or image\" line. Further it causes the display of the ISO image model to be updated. The medium in the input drive must be blank or contain a valid ISO 9660 filesystem. Choosing an input drive causes a root directory to be created in the ISO model of xorriso. If there is a valid ISO filesystem in the input drive then its directory tree gets loaded underneath that model root directory. The input drive may also be a data file on hard disk if that file contains an ISO 9660 filesystem image. See field \"Input drive or image\"." } if {"$what" == "Pick output drive"} { return \ "The \"Pick output drive\" button executes command -outdev and obtains some information about the medium status. This info is displayed in the \"Output drive or image\" line. The output drive may be empty or loaded with a medium, that may be blank, appendable or closed. It is usable for writing only if there is a medium inserted which is either blank or appendable. Button \"Blank\" can bring appendable or closed media into blank state. The output drive may also be a data file on hard disk. See field \"Output drive or image\"." It is considered appendable if it contains an ISO 9660 filesystem image. It is considered blank if it is empty or marked as blank by button \"Blank\". It is considered closed if it contains other data." } if {"$what" == "Pick drive for both roles"} { return \ "The \"Pick drive for both roles\" button executes command -dev and obtains some information about the medium status. This info is displayed in the \"Input drive or image\" line and in the \"Output drive or image\" line. Further it causes the display of the ISO image model to be updated. The medium in the drive must be blank or contain a valid ISO 9660 filesystem. Else the drive will only be aquired as output drive. This drive configuration is the most usual one with xorriso. It loads an eventual ISO image, allows to manipulate it by insertion, deletion, and renaming. When this is done, the changes get written to the drive via button \"Write ISO session\". The drive may also be a data file on hard disk. See fields \"Input drive or image\" and \"Output drive or image\". It is considered appendable if it contains an ISO 9660 filesystem image. It is considered blank if it is empty or marked as blank by button \"Blank\". It is considered closed if it contains other data." } if {"$what" == "Give up drives"} { return \ "The \"Give up drives\" button executes commands -indev \"\" -outdev \"\" and clears both \"... drive or image\" lines, as well as the ISO model." } if {"$what" == "Calm drives"} { return \ "The \"Calm drives\" button executes command -calm_drives which tells the aquired optical drives to stop spinning until the next drive activity gets triggered." } if {"$what" == "Rollback"} { return \ "The \"Rollback\" button executes command -rollback which drops all pending changes of the ISO model and reloads it from the input drive, if one is aquired." } if {"$what" == "drivelist"} { return \ "The box underneath the \"Scan for drives\" button shows the optical drives which were found by the most recent scan run. A double-click on a drive item has the same effect as button \"Pick drive for both roles\". " } if {"$what" == "Input drive or image"} { return \ "The \"Input drive or image\" field displays the address of the input drive. Editing the drive address and pressing the Return key causes the execution of command -indev with the field content as drive address. Use this to load the model from an ISO image data file on hard disk. It is of course permissible that input image and output image are the same file. " } if {"$what" == "input drive info"} { return \ "The text field beneath the \"Input drive or image\" field displays the medium status of the input drive. It tells about the writability, the medium type, the number of ISO sessions, and the amount of readable data." } if {"$what" == "Eject (indev)"} { return \ "The \"Eject\" button beneath the \"Input drive or image\" field excutes command -eject \"in\"." } if {"$what" == "Output drive or image"} { return \ "The \"Output drive or image\" field displays the address of the output drive. Editing the drive address and pressing the Return key causes the execution of command -outdev with the field content as drive address. Use this to direct writing to an ISO image data file on hard disk. It is of course permissible that input image and output image are the same file. " } if {"$what" == "output drive info"} { return \ "The text beneath the \"Output drive or image\" field displays the medium status of the output drive. It tells about the writability, the medium type, the number of ISO sessions, and the amount of free space." } if {"$what" == "Eject (outdev)"} { return \ "The \"Eject\" button beneath the \"Output drive or image\" field excutes command -eject \"out\"." } if {"$what" == "Blank"} { return \ "The \"Blank\" button executes command -blank \"as_needed\" on the output drive. This also applies to ISO image data files and overwritable media like DVD+RW, which will get invalidated by a small write operation." } if {"$what" == "Format"} { return \ "The \"Format\" button executes -format \"as_needed\". This only applies to real optical drives and is of interest only with DVD-RW or BD-R media, which both can be used formatted and unformatted. Other media types which mandatorily need formatting will be formatted by the write commands. Formatted DVD-RW media have the advantage of being overwritable and thus being quickly blankable while maintaining the capability for multi-session. Formatted BD-R can perform Defect Management, which is of questionable value." } if {"$what" == "Write ISO session"} { return \ "The \"Write ISO session\" executes command -commit, which writes a session with all pending changes to the output drive. The output drive must be either blank or it must be the same as the input drive. Writing the session is the last step in the course of creating a new ISO filesystem or an add-on session that expands or changes the ISO filesystem on the medium of the output drive. So first choose a drive, then insert files from hard disk or do other manipulations, and then click \"Write ISO session\" to let xorriso write the data to medium or ISO image file. " } if {"$what" == "Close"} { return \ "The \"Close\" switch controls whether command -close \"on\" is emitted with \"Write ISO session\" or whether -as cdrecord option -multi is omitted with \"Burn image file:\". Closed optical media cannot be written any more unless they get blanked, which is not possible with CD-R, DVD-R, DVD+R, and BD-R. " } if {"$what" == "TAO"} { return \ "The \"TAO\" switch controls whether an incremental MMC write type shall be enforced with write commands. Normally xorriso will decide by medium status and job parameters which MMC write type to choose. Some drives at the edge of failure might work with the one write type while already failing with the other." } if {"$what" == "Defect Mgt"} { return \ "The \"Defect Mgt\" switch controls whether slow and error-prone drive internal check-reading shall be enabled when writing to formatted BD-R or BD-RE. " } if {"$what" == "Burn image file:"} { return \ "The \"Burn image file:\" button executes command -as \"cdrecord\" to burn a data file from hard disk onto the output drive. The address of the disk file is taken from the neighboring text field. The medium in the drive must be blank. (It is well possible to burn image files to appendable media. But the image needs to be prepared for the address offset. Who can do that can as well use one of the command line tools for burning the result. E.g. xorriso -as cdrecord -v dev=/dev/sr0 -multi stream_recording=32s image.iso )" } if {"$what" == "Extract to disk:"} { return \ "The \"Extract to disk:\" button executes command -extract with the whole tree of the current ISO directory or with the selected items of the box underneath \"ISO directory:\". This copies the selected files or directory trees from the input drive to the address on hard disk which is given by the text field right of the button." } if {"$what" == "Browse tree"} { return "[tell_file_browser_help 0]" } if {"$what" == "Close (browse tree)"} { return \ "The \"Close\" button in the file browser closes the browser window without performing other actions." } if {"$what" == "Up (browse tree)"} { return \ "The \"Up\" button in the file browser brings you to the parent directory of the currently selected file tree item. The parent directory will be opened and become the selected item. All opened directory trees underneath the parent will be closed." } if {"$what" == "Down (browse tree)"} { return \ "The \"Down\" button in the file browser opens the directory underneath the currently selected file tree item. It has the same effect as clicking the \"+\" node of the selected item." } if {"$what" == "Accept (browse tree)"} { return \ "The \"Accept\" button in the file browser brings the single selected item from the file browser tree into effect with the associated text field. I.e. it hits the Return key of the field. It works as if the item had been double clicked." } if {"$what" == "To field (browse tree)"} { return \ "The \"To field\" button in the file browser brings the single selected item from the file browser tree into the associated text field. It does not hit the Return key of the field. It works as if the switch \"File browser text field\" was enabled." } if {"$what" == "Browse disk (extract)"} { return \ "The \"/\" button in the \"Extract to disk:\" line pops up a file tree browser to select a target address in the hard disk filesystem. [tell_file_browser_help 1]" } if {"$what" == "Browse disk (burn image)"} { return \ "The \"/\" button beneath the \"Burn image file\" field pops up a file tree browser to select a source address in the hard disk filesystem. [tell_file_browser_help 1]" } if {"$what" == "Browse disk (insert)"} { return \ "The \"/\" button beneath the \"Insert from disk\" field pops up a file tree browser to select a source address in the hard disk filesystem. [tell_file_browser_help 1]" } if {"$what" == "Browse disk (indev)"} { return \ "The \"/\" button in the \"Input drive or image\" line pops up a file tree browser to select a source address in the hard disk filesystem. [tell_file_browser_help 1]" } if {"$what" == "Browse disk (outdev)"} { return \ "The \"/\" button in the \"Output drive or image\" line pops up a file tree browser to select a source address in the hard disk filesystem. [tell_file_browser_help 1]" } if {"$what" == "Browse ISO (isodir)"} { return \ "The \"/\" button in the \"ISO directory\" line pops up a file tree browser to select the current directory in the ISO filesystem model. [tell_file_browser_help 1]" } if {"$what" == "Browse ISO (move target)"} { return \ "The \"/\" button in the \"ISO selection:\" line pops up a file tree browser to select the current directory in the ISO filesystem model. [tell_file_browser_help 1]" } if {"$what" == "Browse disk (dummy)"} { return \ "Normally this button would start a file browser to select a file or directory on hard disk. But the browser cannot be displayed because Tcl/Tk package \"BWidget\" is not loaded." } if {"$what" == "Browse ISO (dummy)"} { return \ "Normally this button would start a file browser to select a file or directory in the ISO model. But the browser cannot be displayed because Tcl/Tk package \"BWidget\" is not loaded." } if {"$what" == "Underneath (extract)"} { return \ "The \"Underneath\" switch controls the effective hard disk target address of an item if the address in the \"Extract to disk:\" field points to a directory. If \"Underneath\" is enabled, then the file object from the ISO filesystem will be copied to its name underneath the hard disk directory. If \"Underneath\" is disabled then an ISO directory tree item will be merged with the disk directory tree at the given address. Example: Selected are \"/iso_dir\" and \"/iso_file\". Address for hard disk is \"/tmp/from_iso\". Switch \"Selected\" is enabled. \"Underneath\" enabled causes commands: -extract /iso_dir /tmp/from_iso/iso_dir -extract /iso_file /tmp/from_iso/iso_file \"Underneath\" disabled: -extract /iso_dir /tmp/from_iso -extract /iso_file /tmp/from_iso The last command will fail because /tmp/from_iso already exists as directory." } if {"$what" == "Selected (extract)"} { return \ "The \"Selected\" switch controls whether the whole current ISO directory, or only the selected items shall be copied to hard disk. " } if {"$what" == "Overwrite ISO files"} { return \ "The \"Overwrite ISO files\" switch controls whether existing files may be overwritten in the ISO image. The frontend program will only detect the most obvious name collisions, but xorriso will reliably refuse to overwrite files if this is banned." } if {"$what" == "Overwrite ISO dirs"} { return \ "The \"Overwrite ISO dirs\" switch controls whether it is allowed to replace an ISO directory by a non-directory file. If a directory is copied to a directory, then both directory trees will be merged. So this switch applies only to situations where non-directories hit directories." } if {"$what" == "Enforce disk dir write access"} { return \ "The \"Enforce disk dir write access\" switch enables the -osirrox options \"auto_chmod_on\" and \"sort_lba_on\" which influence file extraction. \"auto_chmod_on\" allows xorriso to give itself temporariy w-permission to all disk directories which are owned by the xorriso user. This is DANGEROUS, of course, but comes in handy with restoring of backups. Option \"sort_lba_on\" reduces head-moves of optical drives and thus can speed up extraction substantially. It is bound to \"auto_chmod_on\" because else it might get in trouble when restoring ISO directories which offer no w-permission." } if {"$what" == "Overwrite disk files"} { return \ "The \"Overwrite disk files\" switch controls whether existing files may be overwritten by extraction on hard disk. This is DANGEROUS, of course, but comes in handy with restoring of backups. The frontend program will only detect the most obvious name collisions, but xorriso will reliably refuse to overwrite files if this is banned." } if {"$what" == "File browser text field"} { return \ "The \"File browser text field\" switch controls whether a single click or cursor movement in the file browser shall bring the selected file address into the associated text input field. If the switch is disabled, then the address gets written into the field only if double clicked. In any case, double clicked addresses get treated as if the Return key had been hit in the text field." } if {"$what" == "Refresh avail:"} { return \ "The \"Refresh avail:\" button triggers a time consuming exact prediction of the free space on the medium in the output drive. For this purpose, the size of an ISO session with the pending changes is computed. With image files rather than real optical drives, the free space of the hosting filesystem is displayed." } if {"$what" == "Insert from disk:"} { return \ "The \"Insert from disk:\" button executes command -map with the disk file address that is given by the text field right to the button. This inserts files or directory trees into the ISO image model and schedules them for being copied with the next \"Write ISO session\" run. The switches \"Underneath\" and \"Selected\" control what ISO address the inserted files shall have. You may use buttons \"Delete\" and \"Rename to:\" for further adjustments. " } if {"$what" == "Underneath (insert)"} { return \ "The \"Underneath\" switch controls the effective ISO target address if the address in the \"Insert from disk:\" field points to a hard disk directory. If \"Underneath\" is enabled, a directory from disk will not be unpacked to its single files but be put underneath the target address by its own leaf name." If \"Underneath\" is disabled then the directory itself will not show up in the ISO image but only its files and sub directories will do." } if {"$what" == "Selected (insert)"} { return \ "If the switch \"Selected\" is enabled, then the given disk file or tree will be inserted at or underneath the only selected item in the box underneath \"ISO directory:\"." } if {"$what" == "ISO directory:"} { return \ "The current ISO directory shall be used to navigate in the ISO image model of xorriso. By default it is the target of file insertions and the source of file extractions. The text field in the \"ISO directory:\" line displays the current ISO directory and can be used to toggle its path directly. Hitting the Return key causes the current directory to change and the display in the box underneath to be refreshed. It is possible to choose the ISO directory by double-clicking an item in the box underneath the \"ISO directory:\" line. " } if {"$what" == "Up"} { return \ "The \"Up\" buttons move the current ISO directory one directory level up." } if {"$what" == "Verify"} { return \ "The \"Verify\" button executes -md5_check_r \"SORRY\" with the current ISO directory. This reads the content of all data files which are underneath the current ISO directory and which have MD5 checksums in the ISO image. ISO images bear MD5 checksums for each data file if they were produced by xorriso with -md5 \"on\" resp. -for_backup. This frontend enables this feature on startup." } if {"$what" == "isolist"} { return \ "The list box underneath the \"ISO directory:\" line displays the files in the current ISO directory. One or more item can be selected and play a role with extraction or insertion of files. Most of the buttons underneath the box operate on the selected items unconditionally." } if {"$what" == "ISO selection:"} { return \ "The ISO selection consists of the items which are selected in the list box above the \"ISO selection:\" line. If the respective \"Selected\" switches are enabled, then the ISO selection is source of file extraction and target of file insertion. In any case it is the old nome of the \"Rename to:\" button, the victim of the \"Delete\" button, and the subject of the \"Verify\" button." } if {"$what" == "Verify (selection)"} { return \ "The \"Verify\" button in the \"ISO selection:\" line executes command -md5_check_r \"SORRY\" with each of the selected items. This reads the content of all data files which are selected or underneath selected directories and which have MD5 checksums in the ISO image. ISO images bear MD5 checksums for each data file if they were produced by xorriso with -md5 \"on\" resp. -for_backup. This frontend enables this feature on startup." } if {"$what" == "Delete"} { return \ "The \"Delete\" button executes command -rm_r with each of the selected items. This removes the affected files and directory trees from the ISO model. They will not show up in the directory tree of the next session that is written via \"Write ISO session\". Nevertheless they will stay present in earlier sessions if they were not freshly inserted after loading the ISO model from the input drive. " } if {"$what" == "Rename to:"} { return \ "The \"Rename to:\" button uses command -mv to move each of the selected items to the address that is given by the text field right to the button. If this address points to an existing ISO directory, then the items will be moved underneath that directory and keep their leaf names. Else there may be only one selected item which will be renamed to the given address." } if {"$what" == "Make dir"} { return \ "The \"Make dir\" button executes command -mkdir with the address in the text field to its left (the same as used by \"Rename to:\"). Useful to create a target directory before moving the selection." } if {"$what" == "rename and mkdir target"} { return \ "The text field between the \"Rename to:\" button and the \"Make dir\" button serves both buttons by providing the target address for renaming resp. directory creation." } return "--- No help text found for topic '$what'" } # Tell the general help text of the file browser. # proc tell_file_browser_help {with_separator} { set sep "" if {"$with_separator" == 1} { set sep \ "-------------------------------------------------------------------------\n\n" } return \ "${sep}The file tree browser presents to you a directory tree and lets you bring into effect one of the file addresses in that tree. Click on the \"+\" resp. \"-\" nodes to open resp. close directories. Double click on an item to bring it into effect with the associated text field. I.e. double clicking also hits the Return key in that field. The \"Accept\" button does the same with the selected item. The \"To field\" button brings the selected item into the text field without hitting the Return key. So you may edit the name before hitting Return yourself. The \"Up\" button brings you to the parent directory of the selected item. The \"Down\" button works like clicking the \"+\" node of the selected item. The \"Help\" button displays this help text window. The \"Close\" button closes the browser window. If the \"File browser text field\" switch is enabled then a single click or a cursor movement by the arrow keys brings the selected item into the associated text field, but does not hit the Return key. " } # ------- Misc helper procedures ------- # Equip a text with quotation marks so that xorriso will consider it as # a single word. # proc make_text_shellsafe {text} { set result "'" set rest "$text" while {[string length "$rest"]} { set idx [string first "'" "$rest"] if {"$idx" == -1} { set result "$result$rest" break } else { if {"$idx" > 0} { set result "$result[string range "$rest" 0 [expr "$idx" - 1]]" } set result "$result'\"'\"'" if {"$idx" == [expr [string length "$rest"] - 1]} { break } set rest [string range "$rest" [expr "$idx" + 1] end] } } set result "$result'" } # Count the number of newline chracters in text. # proc count_newlines {text} { set rest "$text" set count 0 while {[string length "$rest"]} { set idx [string first "\n" "$rest"] if {"$idx" == -1} { break } else { set count [expr "$count" + 1] if {"$idx" == [expr [string length "$rest"] - 1]} { break } set rest [string range "$rest" [expr "$idx" + 1] end] } } return "$count" } # Append name to dir so that the result is a path to name under dir. # proc combine_dir_and_name {dir name} { set has_slash 0 if {"$name" == ""} { return "$dir" } if {[string range "$name" 0 0] == "/"} { incr has_slash } if {[string last "/" "$dir"] == [expr [string length "$dir"] - 1] && "$dir" != ""} { incr has_slash 1 } if {"$has_slash" == 2} { return "$dir[string range $name" 1 end]" } if {"$has_slash" == 1} { return "$dir$name" } return "$dir/$name" } # Force the content of variable isodir_adr to be an absolute address # proc normalize_isodir_adr {} { global isodir_adr if {"$isodir_adr" == ""} { set isodir_adr "/" } if {[string range "$isodir_adr" 0 0] != "/"} { set isodir_adr "/$isodir_adr" } } # Inspect path whether one of its components is in isodir_adr # proc path_touches_isodir {path} { global isodir_adr normalize_isodir_adr set cmp_start 0 if {"$isodir_adr" == "/"} { set cmp_start 1 } if {[string range "$path" 0 0] != "/"} { if {[string first "/" "$path"] == -1} { return "$path" } else { return [file dirname "$path"] } } set l [expr {[string length "$isodir_adr"] - $cmp_start}] if {[string length "$path"] < [expr {$l + 2}]} { return "" } if {$l > 0} { if {[string range "$path" $cmp_start [expr {$l - 1}]] != \ [string range "$isodir_adr" $cmp_start end]} { return "" } } if {[string range "$path" "$l" "$l"] != "/"} { return "" } set subpath [string range "$path" [expr {$l + 1}] end] set slash [string first "/" "$subpath"] if {"$slash" == -1} { return "$subpath" } if {"$slash" == 0} { return "" } return [string range "$subpath" 0 [expr {$slash - 1}]] } # Compare two severity names by help of the severity list that was obtained # from xorriso via proc inquire_severity_list. # proc compare_sev {sev1 sev2} { global xorriso_severity_list set idx1 [lsearch -exact "$xorriso_severity_list" "$sev1"] set idx2 [lsearch -exact "$xorriso_severity_list" "$sev2"] if {$idx1 < $idx2} {return -1} if {$idx1 > $idx2} {return 1} return 0 } # Write a text to the pipe log # proc log_puts {text} { global logging log_conn if {"$logging" == 1} { puts $log_conn "$text" } } # End program and return the given exit value. # proc central_exit {value} { exit $value } # Start a xorriso process which will in return launch another frontend # process. This is necessary until i learned how to create a pair of pipes # and to fork in Tcl. # proc start_xorriso {} { global argv0 argv set self "" if {[string first "/" "$argv0"] != -1} { set self "$argv0" } if {"$self" == ""} { set self "/usr/bin/xorriso-tcltk" if {[file executable "$self"] == 0} {set self ""} } if {"$self" == ""} { set self "/usr/local/bin/xorriso-tcltk" if {[file executable "$self"] == 0} {set self ""} } if {"$self" == ""} { catch { set conn [open "|which xorriso-tcltk" r] set self [gets $conn] close $conn } } if {"$self" == ""} { catch { set conn [open "|sh -c \"type -p xorriso-tcltk\"" r] set self [gets $conn] close $conn } } if {"$self" == ""} { puts stderr "$argv0 :\n Cannot locate address of script xorriso-tcltk in filesystem.\n" puts stderr "You will have to use --stdio or --named_pipes." puts stderr "See $argv0 --help\n" central_exit 1 } # eval is used to split $argv into single words eval exec xorriso -launch_frontend "\"$self\"" --silent_start --stdio $argv -- 2>@stderr central_exit 0 } # Print a startup message to stderr if not the first argument is --silent_start # proc yell_xorriso_tcltk {} { global argv own_version if {[llength "$argv"] > 0} { if {[lindex "$argv" 0] == "--silent_start"} {return ""} } puts stderr "xorriso-tcltk $own_version : Proof of concept for GUI frontends of xorriso\n" } # Tells whether an absolute path leads to a directory on hard disk # proc localfs_isdir {path} { catch {file lstat "$path" stbuf} if {[info exists stbuf] == 1} { if {$stbuf(type) == "directory"} {return 1} } return 0 } # Return the list of files of a hard disk filesystem directory # proc localfs_ls {dir} { set result "" if {[localfs_isdir "$dir"] == 0} {return ""} set conn [open "|ls {$dir}" r] while {1} { set ret [gets $conn line] if {"$ret" == -1} { break } set adr [combine_dir_and_name "$dir" "$line"] # >>> Obtain all types, not only "d" and "?" if {[localfs_isdir "$adr"] == 1} { lappend result "d $line" } else { lappend result "? $line" } } catch {close $conn} return "$result" } # -------- start living proc setup_by_args {argv0 argv} { global cmd_pipe_adr reply_pipe_adr main_window_geometry click_to_focus global log_file log_conn logging have_bwidget global cmd_conn reply_conn global geometry stdout stdin # wish normally eats the -geometry option and puts the result into $geometry catch {set main_window_geometry "$geometry"} set connection_defined 0 set loop_limit [llength "$argv"] for {set i 0} {"$i" < "$loop_limit"} {incr i} { set ok "0" set opt [lrange "$argv" "$i" "$i"] if {"$opt" == "--help"} { set ok "1" print_usage "$argv0" central_exit 0 } if {"$opt" == "--silent_start"} { set ok "1" } if {"$opt" == "--stdio"} { set ok "1" set connection_defined 1 } if {"$opt" == "--named_pipes"} { set ok "1" incr i set cmd_pipe_adr [lrange "$argv" "$i" "$i"] incr i set reply_pipe_adr [lrange "$argv" "$i" "$i"] if {"$cmd_pipe_adr" != "" && "$reply_pipe_adr" != "" && "$cmd_pipe_adr" != "-" && "$reply_pipe_adr" != "-"} { init_frontend_named_pipes "$cmd_pipe_adr" "$reply_pipe_adr" } set connection_defined 1 } if {"$opt" == "--geometry" || "$opt" == "-geometry"} { set ok "1" # Just in case -geometry does not get eaten by wish incr i set main_window_geometry [lrange "$argv" "$i" "$i"] set give_geometry [lrange "$argv" "$i" "$i"] } if {"$opt" == "--click_to_focus"} { set ok "1" set click_to_focus "1" } if {"$opt" == "--auto_focus"} { set ok "1" set click_to_focus "0" } if {"$opt" == "--log_file"} { set ok "1" incr i set log_file [lrange "$argv" "$i" "$i"] if {"$log_file" == "" || "$log_file" == "-"} { set log_conn stderr } else { set log_conn "" catch {set log_conn [open "$log_file" a]} if {"$log_conn" == ""} { puts stderr "$argv0 : Cannot open -log_file '$log_file' for writing" central_exit 2 } } set logging "1" } if {"$opt" == "--no_bwidget"} { set ok "1" set have_bwidget "-1" } if {"$ok" == 0} { puts stderr "$argv0 : Unknown option '$opt'" print_usage "$argv0" central_exit 1 } } if {"$connection_defined" == 0} { start_xorriso } if {"$cmd_pipe_adr" == "" || "$reply_pipe_adr" == "" || "$cmd_pipe_adr" == "-" || "$reply_pipe_adr" == "-"} { set cmd_conn stdout set reply_conn stdin } if {"$main_window_geometry" != ""} { wm geometry . "$main_window_geometry" } } yell_xorriso_tcltk setup_by_args "$argv0" "$argv" check_xorriso_version setup_xorriso init_gui display_busy 0 refresh_state