415 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			Tcl
		
	
	
			
		
		
	
	
			415 lines
		
	
	
		
			9.8 KiB
		
	
	
	
		
			Tcl
		
	
	
| # git-gui Tools menu dialogs
 | |
| 
 | |
| class tools_add {
 | |
| 
 | |
| field w              ; # widget path
 | |
| field w_name         ; # new remote name widget
 | |
| field w_cmd          ; # new remote location widget
 | |
| 
 | |
| field name         {}; # name of the tool
 | |
| field command      {}; # command to execute
 | |
| field add_global    0; # add to the --global config
 | |
| field no_console    0; # disable using the console
 | |
| field needs_file    0; # ensure filename is set
 | |
| field confirm       0; # ask for confirmation
 | |
| field ask_branch    0; # ask for a revision
 | |
| field ask_args      0; # ask for additional args
 | |
| 
 | |
| constructor dialog {} {
 | |
| 	global repo_config use_ttk NS
 | |
| 
 | |
| 	make_dialog top w
 | |
| 	wm title $top [mc "%s (%s): Add Tool" [appname] [reponame]]
 | |
| 	if {$top ne {.}} {
 | |
| 		wm geometry $top "+[winfo rootx .]+[winfo rooty .]"
 | |
| 		wm transient $top .
 | |
| 	}
 | |
| 
 | |
| 	${NS}::label $w.header -text [mc "Add New Tool Command"] \
 | |
| 		-font font_uibold -anchor center
 | |
| 	pack $w.header -side top -fill x
 | |
| 
 | |
| 	${NS}::frame $w.buttons
 | |
| 	${NS}::checkbutton $w.buttons.global \
 | |
| 		-text [mc "Add globally"] \
 | |
| 		-variable @add_global
 | |
| 	pack $w.buttons.global -side left -padx 5
 | |
| 	${NS}::button $w.buttons.create -text [mc Add] \
 | |
| 		-default active \
 | |
| 		-command [cb _add]
 | |
| 	pack $w.buttons.create -side right
 | |
| 	${NS}::button $w.buttons.cancel -text [mc Cancel] \
 | |
| 		-command [list destroy $w]
 | |
| 	pack $w.buttons.cancel -side right -padx 5
 | |
| 	pack $w.buttons -side bottom -fill x -pady 10 -padx 10
 | |
| 
 | |
| 	${NS}::labelframe $w.desc -text [mc "Tool Details"]
 | |
| 
 | |
| 	${NS}::label $w.desc.name_cmnt -anchor w\
 | |
| 		-text [mc "Use '/' separators to create a submenu tree:"]
 | |
| 	grid x $w.desc.name_cmnt -sticky we -padx {0 5} -pady {0 2}
 | |
| 	${NS}::label $w.desc.name_l -text [mc "Name:"]
 | |
| 	set w_name $w.desc.name_t
 | |
| 	${NS}::entry $w_name \
 | |
| 		-width 40 \
 | |
| 		-textvariable @name \
 | |
| 		-validate key \
 | |
| 		-validatecommand [cb _validate_name %d %S]
 | |
| 	grid $w.desc.name_l $w_name -sticky we -padx {0 5}
 | |
| 
 | |
| 	${NS}::label $w.desc.cmd_l -text [mc "Command:"]
 | |
| 	set w_cmd $w.desc.cmd_t
 | |
| 	${NS}::entry $w_cmd \
 | |
| 		-width 40 \
 | |
| 		-textvariable @command
 | |
| 	grid $w.desc.cmd_l $w_cmd -sticky we -padx {0 5} -pady {0 3}
 | |
| 
 | |
| 	grid columnconfigure $w.desc 1 -weight 1
 | |
| 	pack $w.desc -anchor nw -fill x -pady 5 -padx 5
 | |
| 
 | |
| 	${NS}::checkbutton $w.confirm \
 | |
| 		-text [mc "Show a dialog before running"] \
 | |
| 		-variable @confirm -command [cb _check_enable_dlg]
 | |
| 
 | |
| 	${NS}::labelframe $w.dlg -labelwidget $w.confirm
 | |
| 
 | |
| 	${NS}::checkbutton $w.dlg.askbranch \
 | |
| 		-text [mc "Ask the user to select a revision (sets \$REVISION)"] \
 | |
| 		-variable @ask_branch -state disabled
 | |
| 	pack $w.dlg.askbranch -anchor w -padx 15
 | |
| 
 | |
| 	${NS}::checkbutton $w.dlg.askargs \
 | |
| 		-text [mc "Ask the user for additional arguments (sets \$ARGS)"] \
 | |
| 		-variable @ask_args -state disabled
 | |
| 	pack $w.dlg.askargs -anchor w -padx 15
 | |
| 
 | |
| 	pack $w.dlg -anchor nw -fill x -pady {0 8} -padx 5
 | |
| 
 | |
| 	${NS}::checkbutton $w.noconsole \
 | |
| 		-text [mc "Don't show the command output window"] \
 | |
| 		-variable @no_console
 | |
| 	pack $w.noconsole -anchor w -padx 5
 | |
| 
 | |
| 	${NS}::checkbutton $w.needsfile \
 | |
| 		-text [mc "Run only if a diff is selected (\$FILENAME not empty)"] \
 | |
| 		-variable @needs_file
 | |
| 	pack $w.needsfile -anchor w -padx 5
 | |
| 
 | |
| 	bind $w <Visibility> [cb _visible]
 | |
| 	bind $w <Key-Escape> [list destroy $w]
 | |
| 	bind $w <Key-Return> [cb _add]\;break
 | |
| 	tkwait window $w
 | |
| }
 | |
| 
 | |
| method _check_enable_dlg {} {
 | |
| 	if {$confirm} {
 | |
| 		$w.dlg.askbranch configure -state normal
 | |
| 		$w.dlg.askargs configure -state normal
 | |
| 	} else {
 | |
| 		$w.dlg.askbranch configure -state disabled
 | |
| 		$w.dlg.askargs configure -state disabled
 | |
| 	}
 | |
| }
 | |
| 
 | |
| method _add {} {
 | |
| 	global repo_config
 | |
| 
 | |
| 	if {$name eq {}} {
 | |
| 		error_popup [mc "Please supply a name for the tool."]
 | |
| 		focus $w_name
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	set item "guitool.$name.cmd"
 | |
| 
 | |
| 	if {[info exists repo_config($item)]} {
 | |
| 		error_popup [mc "Tool '%s' already exists." $name]
 | |
| 		focus $w_name
 | |
| 		return
 | |
| 	}
 | |
| 
 | |
| 	set cmd [list git config]
 | |
| 	if {$add_global} { lappend cmd --global }
 | |
| 	set items {}
 | |
| 	if {$no_console} { lappend items "guitool.$name.noconsole" }
 | |
| 	if {$needs_file} { lappend items "guitool.$name.needsfile" }
 | |
| 	if {$confirm} {
 | |
| 		if {$ask_args}   { lappend items "guitool.$name.argprompt" }
 | |
| 		if {$ask_branch} { lappend items "guitool.$name.revprompt" }
 | |
| 		if {!$ask_args && !$ask_branch} {
 | |
| 			lappend items "guitool.$name.confirm"
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if {[catch {
 | |
| 		eval $cmd [list $item $command]
 | |
| 		foreach citem $items { eval $cmd [list $citem yes] }
 | |
| 	    } err]} {
 | |
| 		error_popup [mc "Could not add tool:\n%s" $err]
 | |
| 	} else {
 | |
| 		set repo_config($item) $command
 | |
| 		foreach citem $items { set repo_config($citem) yes }
 | |
| 
 | |
| 		tools_populate_all
 | |
| 	}
 | |
| 
 | |
| 	destroy $w
 | |
| }
 | |
| 
 | |
| method _validate_name {d S} {
 | |
| 	if {$d == 1} {
 | |
| 		if {[regexp {[~?*&\[\0\"\\\{]} $S]} {
 | |
| 			return 0
 | |
| 		}
 | |
| 	}
 | |
| 	return 1
 | |
| }
 | |
| 
 | |
| method _visible {} {
 | |
| 	grab $w
 | |
| 	$w_name icursor end
 | |
| 	focus $w_name
 | |
| }
 | |
| 
 | |
| }
 | |
| 
 | |
| class tools_remove {
 | |
| 
 | |
| field w              ; # widget path
 | |
| field w_names        ; # name list
 | |
| 
 | |
| constructor dialog {} {
 | |
| 	global repo_config global_config system_config use_ttk NS
 | |
| 
 | |
| 	load_config 1
 | |
| 
 | |
| 	make_dialog top w
 | |
| 	wm title $top [mc "%s (%s): Remove Tool" [appname] [reponame]]
 | |
| 	if {$top ne {.}} {
 | |
| 		wm geometry $top "+[winfo rootx .]+[winfo rooty .]"
 | |
| 		wm transient $top .
 | |
| 	}
 | |
| 
 | |
| 	${NS}::label $w.header -text [mc "Remove Tool Commands"] \
 | |
| 		-font font_uibold -anchor center
 | |
| 	pack $w.header -side top -fill x
 | |
| 
 | |
| 	${NS}::frame $w.buttons
 | |
| 	${NS}::button $w.buttons.create -text [mc Remove] \
 | |
| 		-default active \
 | |
| 		-command [cb _remove]
 | |
| 	pack $w.buttons.create -side right
 | |
| 	${NS}::button $w.buttons.cancel -text [mc Cancel] \
 | |
| 		-command [list destroy $w]
 | |
| 	pack $w.buttons.cancel -side right -padx 5
 | |
| 	pack $w.buttons -side bottom -fill x -pady 10 -padx 10
 | |
| 
 | |
| 	${NS}::frame $w.list
 | |
| 	set w_names $w.list.l
 | |
| 	slistbox $w_names \
 | |
| 		-height 10 \
 | |
| 		-width 30 \
 | |
| 		-selectmode extended \
 | |
| 		-exportselection false
 | |
| 	pack $w.list.l -side left -fill both -expand 1
 | |
| 	pack $w.list -fill both -expand 1 -pady 5 -padx 5
 | |
| 
 | |
| 	set local_cnt 0
 | |
| 	foreach fullname [tools_list] {
 | |
| 		# Cannot delete system tools
 | |
| 		if {[info exists system_config(guitool.$fullname.cmd)]} continue
 | |
| 
 | |
| 		$w_names insert end $fullname
 | |
| 		if {![info exists global_config(guitool.$fullname.cmd)]} {
 | |
| 			$w_names itemconfigure end -foreground blue
 | |
| 			incr local_cnt
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if {$local_cnt > 0} {
 | |
| 		${NS}::label $w.colorlbl -foreground blue \
 | |
| 			-text [mc "(Blue denotes repository-local tools)"]
 | |
| 		pack $w.colorlbl -fill x -pady 5 -padx 5
 | |
| 	}
 | |
| 
 | |
| 	bind $w <Visibility> [cb _visible]
 | |
| 	bind $w <Key-Escape> [list destroy $w]
 | |
| 	bind $w <Key-Return> [cb _remove]\;break
 | |
| 	tkwait window $w
 | |
| }
 | |
| 
 | |
| method _remove {} {
 | |
| 	foreach i [$w_names curselection] {
 | |
| 		set name [$w_names get $i]
 | |
| 
 | |
| 		catch { git config --remove-section guitool.$name }
 | |
| 		catch { git config --global --remove-section guitool.$name }
 | |
| 	}
 | |
| 
 | |
| 	load_config 0
 | |
| 	tools_populate_all
 | |
| 
 | |
| 	destroy $w
 | |
| }
 | |
| 
 | |
| method _visible {} {
 | |
| 	grab $w
 | |
| 	focus $w_names
 | |
| }
 | |
| 
 | |
| }
 | |
| 
 | |
| class tools_askdlg {
 | |
| 
 | |
| field w              ; # widget path
 | |
| field w_rev        {}; # revision browser
 | |
| field w_args       {}; # arguments
 | |
| 
 | |
| field is_ask_args   0; # has arguments field
 | |
| field is_ask_revs   0; # has revision browser
 | |
| 
 | |
| field is_ok         0; # ok to start
 | |
| field argstr       {}; # arguments
 | |
| 
 | |
| constructor dialog {fullname} {
 | |
| 	global M1B use_ttk NS
 | |
| 
 | |
| 	set title [get_config "guitool.$fullname.title"]
 | |
| 	if {$title eq {}} {
 | |
| 		regsub {/} $fullname { / } title
 | |
| 	}
 | |
| 
 | |
| 	make_dialog top w -autodelete 0
 | |
| 	wm title $top "[mc "%s (%s):" [appname] [reponame]] $title"
 | |
| 	if {$top ne {.}} {
 | |
| 		wm geometry $top "+[winfo rootx .]+[winfo rooty .]"
 | |
| 		wm transient $top .
 | |
| 	}
 | |
| 
 | |
| 	set prompt [get_config "guitool.$fullname.prompt"]
 | |
| 	if {$prompt eq {}} {
 | |
| 		set command [get_config "guitool.$fullname.cmd"]
 | |
| 		set prompt [mc "Run Command: %s" $command]
 | |
| 	}
 | |
| 
 | |
| 	${NS}::label $w.header -text $prompt -font font_uibold -anchor center
 | |
| 	pack $w.header -side top -fill x
 | |
| 
 | |
| 	set argprompt [get_config "guitool.$fullname.argprompt"]
 | |
| 	set revprompt [get_config "guitool.$fullname.revprompt"]
 | |
| 
 | |
| 	set is_ask_args [expr {$argprompt ne {}}]
 | |
| 	set is_ask_revs [expr {$revprompt ne {}}]
 | |
| 
 | |
| 	if {$is_ask_args} {
 | |
| 		if {$argprompt eq {yes} || $argprompt eq {true} || $argprompt eq {1}} {
 | |
| 			set argprompt [mc "Arguments"]
 | |
| 		}
 | |
| 
 | |
| 		${NS}::labelframe $w.arg -text $argprompt
 | |
| 
 | |
| 		set w_args $w.arg.txt
 | |
| 		${NS}::entry $w_args \
 | |
| 			-width 40 \
 | |
| 			-textvariable @argstr
 | |
| 		pack $w_args -padx 5 -pady 5 -fill both
 | |
| 		pack $w.arg -anchor nw -fill both -pady 5 -padx 5
 | |
| 	}
 | |
| 
 | |
| 	if {$is_ask_revs} {
 | |
| 		if {$revprompt eq {yes} || $revprompt eq {true} || $revprompt eq {1}} {
 | |
| 			set revprompt [mc "Revision"]
 | |
| 		}
 | |
| 
 | |
| 		if {[is_config_true "guitool.$fullname.revunmerged"]} {
 | |
| 			set w_rev [::choose_rev::new_unmerged $w.rev $revprompt]
 | |
| 		} else {
 | |
| 			set w_rev [::choose_rev::new $w.rev $revprompt]
 | |
| 		}
 | |
| 
 | |
| 		pack $w.rev -anchor nw -fill both -expand 1 -pady 5 -padx 5
 | |
| 	}
 | |
| 
 | |
| 	${NS}::frame $w.buttons
 | |
| 	if {$is_ask_revs} {
 | |
| 		${NS}::button $w.buttons.visualize \
 | |
| 			-text [mc Visualize] \
 | |
| 			-command [cb _visualize]
 | |
| 		pack $w.buttons.visualize -side left
 | |
| 	}
 | |
| 	${NS}::button $w.buttons.ok \
 | |
| 		-text [mc OK] \
 | |
| 		-command [cb _start]
 | |
| 	pack $w.buttons.ok -side right
 | |
| 	${NS}::button $w.buttons.cancel \
 | |
| 		-text [mc "Cancel"] \
 | |
| 		-command [cb _cancel]
 | |
| 	pack $w.buttons.cancel -side right -padx 5
 | |
| 	pack $w.buttons -side bottom -fill x -pady 10 -padx 10
 | |
| 
 | |
| 	bind $w <$M1B-Key-Return> [cb _start]
 | |
| 	bind $w <Key-Return> [cb _start]
 | |
| 	bind $w <Key-Escape> [cb _cancel]
 | |
| 	wm protocol $w WM_DELETE_WINDOW [cb _cancel]
 | |
| 
 | |
| 	bind $w <Visibility> [cb _visible]
 | |
| 	return $this
 | |
| }
 | |
| 
 | |
| method execute {} {
 | |
| 	tkwait window $w
 | |
| 	set rv $is_ok
 | |
| 	delete_this
 | |
| 	return $rv
 | |
| }
 | |
| 
 | |
| method _visible {} {
 | |
| 	grab $w
 | |
| 	if {$is_ask_args} {
 | |
| 		focus $w_args
 | |
| 	} elseif {$is_ask_revs} {
 | |
| 		$w_rev focus_filter
 | |
| 	}
 | |
| }
 | |
| 
 | |
| method _cancel {} {
 | |
| 	wm protocol $w WM_DELETE_WINDOW {}
 | |
| 	destroy $w
 | |
| }
 | |
| 
 | |
| method _rev {} {
 | |
| 	if {[catch {$w_rev commit_or_die}]} {
 | |
| 		return {}
 | |
| 	}
 | |
| 	return [$w_rev get]
 | |
| }
 | |
| 
 | |
| method _visualize {} {
 | |
| 	global current_branch
 | |
| 	set rev [_rev $this]
 | |
| 	if {$rev ne {}} {
 | |
| 		do_gitk [list --left-right "$current_branch...$rev"]
 | |
| 	}
 | |
| }
 | |
| 
 | |
| method _start {} {
 | |
| 	global env
 | |
| 
 | |
| 	if {$is_ask_revs} {
 | |
| 		set name [_rev $this]
 | |
| 		if {$name eq {}} {
 | |
| 			return
 | |
| 		}
 | |
| 		set env(REVISION) $name
 | |
| 	}
 | |
| 
 | |
| 	if {$is_ask_args} {
 | |
| 		set env(ARGS) $argstr
 | |
| 	}
 | |
| 
 | |
| 	set is_ok 1
 | |
| 	_cancel $this
 | |
| }
 | |
| 
 | |
| }
 |