---------------------------------------------------------------------------- --- Theme library. -- -- @author Damien Leone <damien.leone@gmail.com> -- @author Julien Danjou <julien@danjou.info> -- @copyright 2008-2009 Damien Leone, Julien Danjou -- @themelib beautiful ---------------------------------------------------------------------------- -- Grab environment local os = os local pairs = pairs local type = type local dofile = dofile local setmetatable = setmetatable local lgi = require("lgi") local Pango = lgi.Pango local PangoCairo = lgi.PangoCairo local gears_debug = require("gears.debug") local Gio = require("lgi").Gio local protected_call = require("gears.protected_call") local xresources = require("beautiful.xresources") local theme_assets = require("beautiful.theme_assets") local gtk = require("beautiful.gtk") local beautiful = { xresources = xresources, theme_assets = theme_assets, gtk = gtk, mt = {} } -- Local data local theme = {} local descs = setmetatable({}, { __mode = 'k' }) local fonts = setmetatable({}, { __mode = 'v' }) local active_font -- luacheck: max comment line length 300 --- Get GTK+3 theme variables from GtkStyleContext -- @treturn table Key-value table with the following structure: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- --
Result key StyleContext key StyleContext fallback #1 StyleContext fallback #2 GTK Widget fallback
`font_size` Label font-size
`font_family` Label font-family
`bg_color` `theme_bg_color` Window bg
`fg_color` `theme_fg_color` Window fg
`base_color` `theme_base_color` Entry bg
`text_color` `theme_text_color` Entry fg
`button_bg_color` `theme_button_bg_color` `theme_bg_color` Button bg
`button_fg_color` `theme_button_fg_color` `theme_fg_color` Button fg
`button_border_color` Button border-color
`button_border_radius` Button border-radius
`button_border_width` Button border-top-width
`selected_bg_color` `theme_selected_bg_color` ToggleButton bg
`selected_fg_color` `theme_selected_fg_color` ToggleButton fg
`menubar_bg_color` `menubar_bg_color` `theme_bg_color` HeaderBar bg
`menubar_fg_color` `menubar_fg_color` `theme_fg_color` HeaderBar fg
`header_button_bg_color` `header_button_bg_color` `menubar_bg_color` `theme_bg_color` HeaderBar > Button bg
`header_button_fg_color` `header_button_fg_color` `menubar_fg_color` `theme_fg_color` HeaderBar > Button fg
`header_button_border_color` HeaderBar > Button border-color
`error_color` `error_color` `error_bg_color` destructive Button bg
`error_bg_color` `error_bg_color` `error_color` destructive Button bg
`error_fg_color` `error_fg_color` `theme_selected_fg_color` destructive Button fg
`warning_color` `warning_color` `warning_bg_color`
`warning_bg_color` `warning_bg_color` `warning_color`
`warning_fg_color` `warning_fg_color` `theme_selected_fg_color`
`success_color` `success_color` `success_bg_color`
`success_bg_color` `success_bg_color` `success_color`
`success_fg_color` `success_fg_color` `theme_selected_fg_color`
`tooltip_bg_color` `theme_tooltip_bg_color` `theme_bg_color`
`tooltip_fg_color` `theme_tooltip_fg_color` `theme_fg_color`
`osd_bg_color` `osd_bg` `theme_tooltip_bg_color` `theme_bg_color`
`osd_fg_color` `osd_fg` `theme_tooltip_fg_color` `theme_fg_color`
`osd_border_color` `osd_borders_color` `osd_fg_color`
`wm_bg_color` `wm_bg` `menubar_bg_color` `theme_bg_color` HeaderBar bg
`wm_border_focused_color` `wm_border_focused` `theme_selected_bg_color` ToggleButton bg
`wm_border_unfocused_color` `wm_border_unfocused` `wm_border` `menubar_bg_color` HeaderBar bg
`wm_title_focused_color` `wm_title_focused` `wm_title` `theme_selected_fg_color` ToggleButton fg
`wm_title_unfocused_color` `wm_title_unfocused` `wm_unfocused_title` `menubar_fg_color` HeaderBar fg
`wm_icons_focused_color` `wm_icons_focused` `wm_title_focused` `theme_selected_fg_color` ToggleButton fg
`wm_icons_unfocused_color` `wm_icons_unfocused` `wm_title_unfocused` `menubar_fg_color` HeaderBar fg
-- -- @staticfct beautiful.gtk.get_theme_variables --- The default font. -- @beautiful beautiful.font -- The default background color. -- @beautiful beautiful.bg_normal -- The default focused element background color. -- @beautiful beautiful.bg_focus -- The default urgent element background color. -- @beautiful beautiful.bg_urgent -- The default minimized element background color. -- @beautiful beautiful.bg_minimize -- The system tray background color. -- Please note that only solid colors are currently supported. -- @beautiful beautiful.bg_systray -- The default focused element foreground (text) color. -- @beautiful beautiful.fg_normal -- The default focused element foreground (text) color. -- @beautiful beautiful.fg_focus -- The default urgent element foreground (text) color. -- @beautiful beautiful.fg_urgent -- The default minimized element foreground (text) color. -- @beautiful beautiful.fg_minimize --- The gap between clients. -- @beautiful beautiful.useless_gap -- @param[opt=0] number --- The client border width. -- @beautiful beautiful.border_width --- The default clients border color. -- Note that only solid colors are supported. -- @beautiful beautiful.border_normal --- The focused client border color. -- Note that only solid colors are supported. -- @beautiful beautiful.border_focus --- The marked clients border color. -- Note that only solid colors are supported. -- @beautiful beautiful.border_marked --- The wallpaper path. -- @beautiful beautiful.wallpaper -- The icon theme name. -- It has to be a directory in `/usr/share/icons` or an XDG icon folder. -- @beautiful beautiful.icon_theme --- The Awesome icon path. -- @beautiful beautiful.awesome_icon --- The current theme path (if any) -- @tfield string beautiful.theme_path --- Load a font from a string or a font description. -- -- @see https://developer.gnome.org/pango/stable/pango-Fonts.html#pango-font-description-from-string -- @tparam string|lgi.Pango.FontDescription name Font, which can be a -- string or a lgi.Pango.FontDescription. -- @treturn table A table with `name`, `description` and `height`. local function load_font(name) name = name or active_font if name and type(name) ~= "string" then if descs[name] then name = descs[name] else name = name:to_string() end end if fonts[name] then return fonts[name] end -- Load new font local desc = Pango.FontDescription.from_string(name) local ctx = PangoCairo.font_map_get_default():create_context() ctx:set_resolution(beautiful.xresources.get_dpi()) -- Apply default values from the context (e.g. a default font size) desc:merge(ctx:get_font_description(), false) -- Calculate font height. local metrics = ctx:get_metrics(desc, nil) local height = math.ceil((metrics:get_ascent() + metrics:get_descent()) / Pango.SCALE) if height == 0 then height = desc:get_size() / Pango.SCALE gears_debug.print_warning(string.format( "beautiful.load_font: could not get height for '%s' (likely missing font), using %d.", name, height)) end local font = { name = name, description = desc, height = height } fonts[name] = font descs[desc] = name return font end --- Set an active font -- -- @param name The font local function set_font(name) active_font = load_font(name).name end --- Get a font description. -- -- See https://developer.gnome.org/pango/stable/pango-Fonts.html#PangoFontDescription. -- @tparam string|lgi.Pango.FontDescription name The name of the font. -- @treturn lgi.Pango.FontDescription -- @staticfct beautiful.get_font function beautiful.get_font(name) return load_font(name).description end --- Get a new font with merged attributes, based on another one. -- -- See https://developer.gnome.org/pango/stable/pango-Fonts.html#pango-font-description-from-string. -- @tparam string|Pango.FontDescription name The base font. -- @tparam string merge Attributes that should be merged, e.g. "bold". -- @treturn lgi.Pango.FontDescription -- @staticfct beautiful.get_merged_font function beautiful.get_merged_font(name, merge) local font = beautiful.get_font(name) merge = Pango.FontDescription.from_string(merge) local merged = font:copy_static() merged:merge(merge, true) return beautiful.get_font(merged:to_string()) end --- Get the height of a font. -- -- @param name Name of the font. -- @staticfct beautiful.get_font_height function beautiful.get_font_height(name) return load_font(name).height end --- Function that initializes the theme settings. Should be run at the -- beginning of the awesome configuration file (normally rc.lua). -- -- Example usages: -- -- -- Using a table -- beautiful.init({font = 'Monospace Bold 10'}) -- -- -- From a config file -- beautiful.init("/theme.lua") -- -- Example "/theme.lua" (see `05-awesomerc.md:Variable_definitions`): -- -- theme = {} -- theme.font = 'Monospace Bold 10' -- return theme -- -- Example using the return value: -- -- local beautiful = require("beautiful") -- if not beautiful.init("/theme.lua") then -- beautiful.init("/.last.theme.lua") -- a known good fallback -- end -- -- @tparam string|table config The theme to load. It can be either the path to -- the theme file (which should return a table) or directly a table -- containing all the theme values. -- @treturn true|nil True if successful, nil in case of error. -- @staticfct beautiful.init function beautiful.init(config) if config then local state, t_theme = nil, nil local homedir = os.getenv("HOME") -- If `config` is the path to a theme file, run this file, -- otherwise if it is a theme table, save it. local t_config = type(config) if t_config == 'string' then -- Expand the '~' $HOME shortcut config = config:gsub("^~/", homedir .. "/") local dir = Gio.File.new_for_path(config):get_parent() rawset(beautiful, "theme_path", dir and (dir:get_path().."/") or nil) theme = protected_call(dofile, config) t_theme = type(theme) state = t_theme == 'table' and next(theme) elseif t_config == 'table' then rawset(beautiful, "theme_path", nil) theme = config state = next(theme) end if state then -- expand '~' if homedir then for k, v in pairs(theme) do if type(v) == "string" then theme[k] = v:gsub("^~/", homedir .. "/") end end end if theme.font then set_font(theme.font) end return true else rawset(beautiful, "theme_path", nil) theme = {} local file = t_config == 'string' and (" from: " .. config) local err = (file and t_theme == 'table' and "got an empty table" .. file) or (file and t_theme ~= 'table' and "got a " .. t_theme .. file) or (t_config == 'table' and "got an empty table") or ("got a " .. t_config) return gears_debug.print_error("beautiful: error loading theme: " .. err) end else return gears_debug.print_error("beautiful: error loading theme: no theme specified") end end --- Get the current theme. -- -- @treturn table The current theme table. -- @staticfct beautiful.get function beautiful.get() return theme end function beautiful.mt:__index(k) return theme[k] end function beautiful.mt:__newindex(k, v) theme[k] = v end -- Set the default font set_font("sans 8") return setmetatable(beautiful, beautiful.mt) -- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80