| # 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: |