 f4d8e19123
			
		
	
	f4d8e19123
	
	
	
		
			
			* https://github.com/prati0100/git-gui: git-gui: use gray background for inactive text widgets git-gui: Fix selected text colors Makefile: conditionally include GIT-VERSION-FILE git-gui: fix colored label backgrounds when using themed widgets git-gui: ssh-askpass: add a checkbox to show the input text git-gui: update Russian translation git-gui: use commit message template git-gui: Only touch GITGUI_MSG when needed
		
			
				
	
	
		
			411 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Tcl
		
	
	
	
	
	
			
		
		
	
	
			411 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Tcl
		
	
	
	
	
	
| # Functions for supporting the use of themed Tk widgets in git-gui.
 | |
| # Copyright (C) 2009 Pat Thoyts <patthoyts@users.sourceforge.net>
 | |
| 
 | |
| 
 | |
| namespace eval color {
 | |
| 	# Variable colors
 | |
| 	# Preffered way to set widget colors is using add_option.
 | |
| 	# In some cases, like with tags in_diff/in_sel, we use these colors.
 | |
| 	variable select_bg				lightgray
 | |
| 	variable select_fg				black
 | |
| 	variable inactive_select_bg		lightgray
 | |
| 	variable inactive_select_fg		black
 | |
| 
 | |
| 	proc sync_with_theme {} {
 | |
| 		set base_bg				[ttk::style lookup . -background]
 | |
| 		set base_fg				[ttk::style lookup . -foreground]
 | |
| 		set text_bg				[ttk::style lookup Treeview -background]
 | |
| 		set text_fg				[ttk::style lookup Treeview -foreground]
 | |
| 		set select_bg			[ttk::style lookup Default -selectbackground]
 | |
| 		set select_fg			[ttk::style lookup Default -selectforeground]
 | |
| 		set inactive_select_bg	[convert_rgb_to_gray $select_bg]
 | |
| 		set inactive_select_fg	$select_fg
 | |
| 
 | |
| 		set color::select_bg $select_bg
 | |
| 		set color::select_fg $select_fg
 | |
| 		set color::inactive_select_bg $inactive_select_bg
 | |
| 		set color::inactive_select_fg $inactive_select_fg
 | |
| 
 | |
| 		proc add_option {key val} {
 | |
| 			option add $key $val widgetDefault
 | |
| 		}
 | |
| 		# Add options for plain Tk widgets
 | |
| 		# Using `option add` instead of tk_setPalette to avoid unintended
 | |
| 		# consequences.
 | |
| 		if {![is_MacOSX]} {
 | |
| 			add_option *Menu.Background $base_bg
 | |
| 			add_option *Menu.Foreground $base_fg
 | |
| 			add_option *Menu.activeBackground $select_bg
 | |
| 			add_option *Menu.activeForeground $select_fg
 | |
| 		}
 | |
| 		add_option *Text.Background $text_bg
 | |
| 		add_option *Text.Foreground $text_fg
 | |
| 		add_option *Text.selectBackground $select_bg
 | |
| 		add_option *Text.selectForeground $select_fg
 | |
| 		add_option *Text.inactiveSelectBackground $inactive_select_bg
 | |
| 		add_option *Text.inactiveSelectForeground $inactive_select_fg
 | |
| 	}
 | |
| }
 | |
| 
 | |
| proc convert_rgb_to_gray {rgb} {
 | |
| 	# Simply take the average of red, green and blue. This wouldn't be good
 | |
| 	# enough for, say, converting a photo to grayscale, but for this simple
 | |
| 	# purpose of approximating the brightness of a color it's good enough.
 | |
| 	lassign [winfo rgb . $rgb] r g b
 | |
| 	set gray [expr {($r / 256 + $g / 256 + $b / 256) / 3}]
 | |
| 	return [format "#%2.2X%2.2X%2.2X" $gray $gray $gray]
 | |
| }
 | |
| 
 | |
| proc ttk_get_current_theme {} {
 | |
| 	# Handle either current Tk or older versions of 8.5
 | |
| 	if {[catch {set theme [ttk::style theme use]}]} {
 | |
| 		set theme  $::ttk::currentTheme
 | |
| 	}
 | |
| 	return $theme
 | |
| }
 | |
| 
 | |
| proc InitTheme {} {
 | |
| 	# Create a color label style (bg can be overridden by widget option)
 | |
| 	ttk::style layout Color.TLabel {
 | |
| 		Color.Label.border -sticky news -children {
 | |
| 			Color.label.fill -sticky news -children {
 | |
| 				Color.Label.padding -sticky news -children {
 | |
| 					Color.Label.label -sticky news}}}}
 | |
| 	eval [linsert [ttk::style configure TLabel] 0 \
 | |
| 			  ttk::style configure Color.TLabel]
 | |
| 	ttk::style configure Color.TLabel \
 | |
| 		-borderwidth 0 -relief flat -padding 2
 | |
| 	ttk::style map Color.TLabel -background {{} gold}
 | |
| 	# We also need a padded label.
 | |
| 	ttk::style configure Padded.TLabel \
 | |
| 		-padding {5 5} -borderwidth 1 -relief solid
 | |
| 	# We need a gold frame.
 | |
| 	ttk::style layout Gold.TFrame {
 | |
| 		Gold.Frame.border -sticky nswe -children {
 | |
| 			Gold.Frame.fill -sticky nswe}}
 | |
| 	ttk::style configure Gold.TFrame -background gold -relief flat
 | |
| 	# listboxes should have a theme border so embed in ttk::frame
 | |
| 	ttk::style layout SListbox.TFrame {
 | |
| 		SListbox.Frame.Entry.field -sticky news -border true -children {
 | |
| 			SListbox.Frame.padding -sticky news
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	set theme [ttk_get_current_theme]
 | |
| 
 | |
| 	if {[lsearch -exact {default alt classic clam} $theme] != -1} {
 | |
| 		# Simple override of standard ttk::entry to change the field
 | |
| 		# packground according to a state flag. We should use 'user1'
 | |
| 		# but not all versions of 8.5 support that so make use of 'pressed'
 | |
| 		# which is not normally in use for entry widgets.
 | |
| 		ttk::style layout Edged.Entry [ttk::style layout TEntry]
 | |
| 		ttk::style map Edged.Entry {*}[ttk::style map TEntry]
 | |
| 		ttk::style configure Edged.Entry {*}[ttk::style configure TEntry] \
 | |
| 			-fieldbackground lightgreen
 | |
| 		ttk::style map Edged.Entry -fieldbackground {
 | |
| 			{pressed !disabled} lightpink
 | |
| 		}
 | |
| 	} else {
 | |
| 		# For fancier themes, in particular the Windows ones, the field
 | |
| 		# element may not support changing the background color. So instead
 | |
| 		# override the fill using the default fill element. If we overrode
 | |
| 		# the vista theme field element we would loose the themed border
 | |
| 		# of the widget.
 | |
| 		catch {
 | |
| 			ttk::style element create color.fill from default
 | |
| 		}
 | |
| 
 | |
| 		ttk::style layout Edged.Entry {
 | |
| 			Edged.Entry.field -sticky nswe -border 0 -children {
 | |
| 				Edged.Entry.border -sticky nswe -border 1 -children {
 | |
| 					Edged.Entry.padding -sticky nswe -children {
 | |
| 						Edged.Entry.color.fill -sticky nswe -children {
 | |
| 							Edged.Entry.textarea -sticky nswe
 | |
| 						}
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		ttk::style configure Edged.Entry {*}[ttk::style configure TEntry] \
 | |
| 			-background lightgreen -padding 0 -borderwidth 0
 | |
| 		ttk::style map Edged.Entry {*}[ttk::style map TEntry] \
 | |
| 			-background {{pressed !disabled} lightpink}
 | |
| 	}
 | |
| 
 | |
| 	if {[lsearch [bind . <<ThemeChanged>>] InitTheme] == -1} {
 | |
| 		bind . <<ThemeChanged>> +[namespace code [list InitTheme]]
 | |
| 	}
 | |
| }
 | |
| 
 | |
| # Define a style used for the surround of text widgets.
 | |
| proc InitEntryFrame {} {
 | |
| 	ttk::style theme settings default {
 | |
| 		ttk::style layout EntryFrame {
 | |
| 			EntryFrame.field -sticky nswe -border 0 -children {
 | |
| 				EntryFrame.fill -sticky nswe -children {
 | |
| 					EntryFrame.padding -sticky nswe
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		ttk::style configure EntryFrame -padding 1 -relief sunken
 | |
| 		ttk::style map EntryFrame -background {}
 | |
| 	}
 | |
| 	ttk::style theme settings classic {
 | |
| 		ttk::style configure EntryFrame -padding 2 -relief sunken
 | |
| 		ttk::style map EntryFrame -background {}
 | |
| 	}
 | |
| 	ttk::style theme settings alt {
 | |
| 		ttk::style configure EntryFrame -padding 2
 | |
| 		ttk::style map EntryFrame -background {}
 | |
| 	}
 | |
| 	ttk::style theme settings clam {
 | |
| 		ttk::style configure EntryFrame -padding 2
 | |
| 		ttk::style map EntryFrame -background {}
 | |
| 	}
 | |
| 
 | |
| 	# Ignore errors for missing native themes
 | |
| 	catch {
 | |
| 		ttk::style theme settings winnative {
 | |
| 			ttk::style configure EntryFrame -padding 2
 | |
| 		}
 | |
| 		ttk::style theme settings xpnative {
 | |
| 			ttk::style configure EntryFrame -padding 1
 | |
| 			ttk::style element create EntryFrame.field vsapi \
 | |
| 				EDIT 1 {disabled 4 focus 3 active 2 {} 1} -padding 1
 | |
| 		}
 | |
| 		ttk::style theme settings vista {
 | |
| 			ttk::style configure EntryFrame -padding 2
 | |
| 			ttk::style element create EntryFrame.field vsapi \
 | |
| 				EDIT 6 {disabled 4 focus 3 active 2 {} 1} -padding 2
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	bind EntryFrame <Enter> {%W instate !disabled {%W state active}}
 | |
| 	bind EntryFrame <Leave> {%W state !active}
 | |
| 	bind EntryFrame <<ThemeChanged>> {
 | |
| 		set pad [ttk::style lookup EntryFrame -padding]
 | |
| 		%W configure -padding [expr {$pad eq {} ? 1 : $pad}]
 | |
| 	}
 | |
| }
 | |
| 
 | |
| proc gold_frame {w args} {
 | |
| 	global use_ttk
 | |
| 	if {$use_ttk && ![is_MacOSX]} {
 | |
| 		eval [linsert $args 0 ttk::frame $w -style Gold.TFrame]
 | |
| 	} else {
 | |
| 		eval [linsert $args 0 frame $w -background gold]
 | |
| 	}
 | |
| }
 | |
| 
 | |
| proc tlabel {w args} {
 | |
| 	global use_ttk
 | |
| 	if {$use_ttk && ![is_MacOSX]} {
 | |
| 		set cmd [list ttk::label $w -style Color.TLabel]
 | |
| 		foreach {k v} $args {
 | |
| 			switch -glob -- $k {
 | |
| 				-activebackground {}
 | |
| 				default { lappend cmd $k $v }
 | |
| 			}
 | |
| 		}
 | |
| 		eval $cmd
 | |
| 	} else {
 | |
| 		eval [linsert $args 0 label $w]
 | |
| 	}
 | |
| }
 | |
| 
 | |
| # The padded label gets used in the about class.
 | |
| proc paddedlabel {w args} {
 | |
| 	global use_ttk
 | |
| 	if {$use_ttk} {
 | |
| 		eval [linsert $args 0 ttk::label $w -style Padded.TLabel]
 | |
| 	} else {
 | |
| 		eval [linsert $args 0 label $w \
 | |
| 				  -padx 5 -pady 5 \
 | |
| 				  -justify left \
 | |
| 				  -anchor w \
 | |
| 				  -borderwidth 1 \
 | |
| 				  -relief solid]
 | |
| 	}
 | |
| }
 | |
| 
 | |
| # Create a toplevel for use as a dialog.
 | |
| # If available, sets the EWMH dialog hint and if ttk is enabled
 | |
| # place a themed frame over the surface.
 | |
| proc Dialog {w args} {
 | |
| 	eval [linsert $args 0 toplevel $w -class Dialog]
 | |
| 	catch {wm attributes $w -type dialog}
 | |
| 	pave_toplevel $w
 | |
| 	return $w
 | |
| }
 | |
| 
 | |
| # Tk toplevels are not themed - so pave it over with a themed frame to get
 | |
| # the base color correct per theme.
 | |
| proc pave_toplevel {w} {
 | |
| 	global use_ttk
 | |
| 	if {$use_ttk && ![winfo exists $w.!paving]} {
 | |
| 		set paving [ttk::frame $w.!paving]
 | |
| 		place $paving -x 0 -y 0 -relwidth 1 -relheight 1
 | |
| 		lower $paving
 | |
| 	}
 | |
| }
 | |
| 
 | |
| # Create a scrolled listbox with appropriate border for the current theme.
 | |
| # On many themes the border for a scrolled listbox needs to go around the
 | |
| # listbox and the scrollbar.
 | |
| proc slistbox {w args} {
 | |
| 	global use_ttk NS
 | |
| 	if {$use_ttk} {
 | |
| 		set f [ttk::frame $w -style SListbox.TFrame -padding 2]
 | |
| 	} else {
 | |
| 		set f [frame $w -relief flat]
 | |
| 	}
 | |
|     if {[catch {
 | |
| 		if {$use_ttk} {
 | |
| 			eval [linsert $args 0 listbox $f.list -relief flat \
 | |
| 					  -highlightthickness 0 -borderwidth 0]
 | |
| 		} else {
 | |
| 			eval [linsert $args 0 listbox $f.list]
 | |
| 		}
 | |
|         ${NS}::scrollbar $f.vs -command [list $f.list yview]
 | |
|         $f.list configure -yscrollcommand [list $f.vs set]
 | |
|         grid $f.list $f.vs -sticky news
 | |
|         grid rowconfigure $f 0 -weight 1
 | |
|         grid columnconfigure $f 0 -weight 1
 | |
| 		bind $f.list <<ListboxSelect>> \
 | |
| 			[list event generate $w <<ListboxSelect>>]
 | |
|         interp hide {} $w
 | |
|         interp alias {} $w {} $f.list
 | |
|     } err]} {
 | |
|         destroy $f
 | |
|         return -code error $err
 | |
|     }
 | |
|     return $w
 | |
| }
 | |
| 
 | |
| # fetch the background color from a widget.
 | |
| proc get_bg_color {w} {
 | |
| 	global use_ttk
 | |
| 	if {$use_ttk} {
 | |
| 		set bg [ttk::style lookup [winfo class $w] -background]
 | |
| 	} else {
 | |
| 		set bg [$w cget -background]
 | |
| 	}
 | |
| 	return $bg
 | |
| }
 | |
| 
 | |
| # ttk::spinbox didn't get added until 8.6
 | |
| proc tspinbox {w args} {
 | |
| 	global use_ttk
 | |
| 	if {$use_ttk && [llength [info commands ttk::spinbox]] > 0} {
 | |
| 		eval [linsert $args 0 ttk::spinbox $w]
 | |
| 	} else {
 | |
| 		eval [linsert $args 0 spinbox $w]
 | |
| 	}
 | |
| }
 | |
| 
 | |
| # Create a text widget with any theme specific properties.
 | |
| proc ttext {w args} {
 | |
| 	global use_ttk
 | |
| 	if {$use_ttk} {
 | |
| 		switch -- [ttk_get_current_theme] {
 | |
| 			"vista" - "xpnative" {
 | |
| 				lappend args -highlightthickness 0 -borderwidth 0
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	set w [eval [linsert $args 0 text $w]]
 | |
| 	if {$use_ttk} {
 | |
| 		if {[winfo class [winfo parent $w]] eq "EntryFrame"} {
 | |
| 			bind $w <FocusIn> {[winfo parent %W] state focus}
 | |
| 			bind $w <FocusOut> {[winfo parent %W] state !focus}
 | |
| 		}
 | |
| 	}
 | |
| 	return $w
 | |
| }
 | |
| 
 | |
| # themed frame suitable for surrounding a text field.
 | |
| proc textframe {w args} {
 | |
| 	global use_ttk
 | |
| 	if {$use_ttk} {
 | |
| 		if {[catch {ttk::style layout EntryFrame}]} {
 | |
| 			InitEntryFrame
 | |
| 		}
 | |
| 		eval [linsert $args 0 ttk::frame $w -class EntryFrame -style EntryFrame]
 | |
| 	} else {
 | |
| 		eval [linsert $args 0 frame $w]
 | |
| 	}
 | |
| 	return $w
 | |
| }
 | |
| 
 | |
| proc tentry {w args} {
 | |
| 	global use_ttk
 | |
| 	if {$use_ttk} {
 | |
| 		InitTheme
 | |
| 		ttk::entry $w -style Edged.Entry
 | |
| 	} else {
 | |
| 		entry $w
 | |
| 	}
 | |
| 
 | |
| 	rename $w _$w
 | |
| 	interp alias {} $w {} tentry_widgetproc $w
 | |
| 	eval [linsert $args 0 tentry_widgetproc $w configure]
 | |
| 	return $w
 | |
| }
 | |
| proc tentry_widgetproc {w cmd args} {
 | |
| 	global use_ttk
 | |
| 	switch -- $cmd {
 | |
| 		state {
 | |
| 			if {$use_ttk} {
 | |
| 				return [uplevel 1 [list _$w $cmd] $args]
 | |
| 			} else {
 | |
| 				if {[lsearch -exact $args pressed] != -1} {
 | |
| 					_$w configure -background lightpink
 | |
| 				} else {
 | |
| 					_$w configure -background lightgreen
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		configure {
 | |
| 			if {$use_ttk} {
 | |
| 				if {[set n [lsearch -exact $args -background]] != -1} {
 | |
| 					set args [lreplace $args $n [incr n]]
 | |
| 					if {[llength $args] == 0} {return}
 | |
| 				}
 | |
| 			}
 | |
| 			return [uplevel 1 [list _$w $cmd] $args]
 | |
| 		}
 | |
| 		default { return [uplevel 1 [list _$w $cmd] $args] }
 | |
| 	}
 | |
| }
 | |
| 
 | |
| # Tk 8.6 provides a standard font selection dialog. This uses the native
 | |
| # dialogs on Windows and MacOSX or a standard Tk dialog on X11.
 | |
| proc tchoosefont {w title familyvar sizevar} {
 | |
| 	if {[package vsatisfies [package provide Tk] 8.6]} {
 | |
| 		upvar #0 $familyvar family
 | |
| 		upvar #0 $sizevar size
 | |
| 		tk fontchooser configure -parent $w -title $title \
 | |
| 			-font [list $family $size] \
 | |
| 			-command [list on_choosefont $familyvar $sizevar]
 | |
| 		tk fontchooser show
 | |
| 	} else {
 | |
| 		choose_font::pick $w $title $familyvar $sizevar
 | |
| 	}
 | |
| }
 | |
| 
 | |
| # Called when the Tk 8.6 fontchooser selects a font.
 | |
| proc on_choosefont {familyvar sizevar font} {
 | |
| 	upvar #0 $familyvar family
 | |
| 	upvar #0 $sizevar size
 | |
| 	set font [font actual $font]
 | |
| 	set family [dict get $font -family]
 | |
| 	set size [dict get $font -size]
 | |
| }
 | |
| 
 | |
| # Local variables:
 | |
| # mode: tcl
 | |
| # indent-tabs-mode: t
 | |
| # tab-width: 4
 | |
| # End:
 |