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