awful.titlebar: more buttons added
This is a new version of 'titlebar.lua'. There are now additional buttons for: - setting the 'ontop' attribute - setting the 'sticky' attribute - maximizing the client In addition to changing the state of the client the buttons also show if the corresponding state is active for a client. All new buttons now require 4 images for: - unfocused window, state inactive (i.e. ontop == false) - unfocused window, state active (i.e. ontop == true) - focused window, state inactive (i.e. ontop == false) - focused window, state active (i.e. ontop == true) It's not difficult to add more buttons with any kind of functionality. User configuration is currently restricted to theming of the buttons. Buttons are disabled simply by not configuring them in the theme. The icons for the new buttons are taken from kde4. They're not really pretty but they're OK to demonstrate the new features. Signed-off-by: Julien Danjou <julien@danjou.info>
|
@ -28,6 +28,9 @@ module("awful.titlebar")
|
||||||
-- Privata data
|
-- Privata data
|
||||||
local data = otable()
|
local data = otable()
|
||||||
|
|
||||||
|
-- Predeclaration for buttons
|
||||||
|
local button_groups
|
||||||
|
|
||||||
--- Create a standard titlebar.
|
--- Create a standard titlebar.
|
||||||
-- @param c The client.
|
-- @param c The client.
|
||||||
-- @param args Arguments.
|
-- @param args Arguments.
|
||||||
|
@ -71,52 +74,28 @@ function add(c, args)
|
||||||
local appicon = capi.widget({ type = "imagebox", align = "left" })
|
local appicon = capi.widget({ type = "imagebox", align = "left" })
|
||||||
appicon.image = c.icon
|
appicon.image = c.icon
|
||||||
|
|
||||||
-- Also redirect events for appicon (So the entire titlebar behaves consistently)
|
-- for each button group, call create for the client.
|
||||||
appicon:buttons(bts)
|
-- if a button set is created add the set to the
|
||||||
function appicon.mouse_enter(s) hooks.user.call('mouse_enter', c) end
|
-- data[c].button_sets for late updates and add the
|
||||||
|
-- individual buttons to the array part of the widget
|
||||||
local closef
|
-- list
|
||||||
if theme.titlebar_close_button_focus then
|
local widget_list = { appicon = appicon, title = title }
|
||||||
closef = widget.button({ align = "right",
|
local iw = #widget_list
|
||||||
image = image(theme.titlebar_close_button_focus) })
|
local is = 1
|
||||||
end
|
data[c].button_sets = {}
|
||||||
|
for i = 1, #button_groups do
|
||||||
local close
|
local set = button_groups[i].create(c, modkey, theme)
|
||||||
if theme.titlebar_close_button_normal then
|
if (set) then
|
||||||
close = widget.button({ align = "right",
|
data[c].button_sets[is] = set
|
||||||
image = image(theme.titlebar_close_button_normal) })
|
is = is + 1
|
||||||
end
|
for n,b in pairs(set) do
|
||||||
|
widget_list[iw] = b
|
||||||
if close or closef then
|
iw = iw + 1
|
||||||
-- Bind kill button, also allow moving and resizing on this widget
|
|
||||||
local bts =
|
|
||||||
{
|
|
||||||
capi.button({ }, 1, nil, function (t) t.client:kill() end),
|
|
||||||
capi.button({ args.modkey }, 1, function (t) mouse.client.move(t.client) end),
|
|
||||||
capi.button({ args.modkey }, 3, function (t) mouse.client.resize(t.client) end)
|
|
||||||
}
|
|
||||||
if close then
|
|
||||||
local rbts = close:buttons()
|
|
||||||
for k, v in pairs(rbts) do
|
|
||||||
bts[#bts + 1] = v
|
|
||||||
end
|
end
|
||||||
close:buttons(bts)
|
|
||||||
function close.mouse_enter(s) hooks.user.call('mouse_enter', c) end
|
|
||||||
end
|
|
||||||
if closef then
|
|
||||||
local rbts = closef:buttons()
|
|
||||||
for k, v in pairs(rbts) do
|
|
||||||
bts[#bts + 1] = v
|
|
||||||
end
|
|
||||||
closef:buttons(bts)
|
|
||||||
-- Needed for sloppy focus beheaviour
|
|
||||||
function closef.mouse_enter(s) hooks.user.call('mouse_enter', c) end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
tb.widgets = { appicon = appicon, title = title,
|
tb.widgets = widget_list
|
||||||
closef = closef, close = close }
|
|
||||||
|
|
||||||
c.titlebar = tb
|
c.titlebar = tb
|
||||||
|
|
||||||
update(c)
|
update(c)
|
||||||
|
@ -126,7 +105,7 @@ end
|
||||||
-- @param c The client to update.
|
-- @param c The client to update.
|
||||||
-- @param prop The property name which has changed.
|
-- @param prop The property name which has changed.
|
||||||
function update(c, prop)
|
function update(c, prop)
|
||||||
if c.titlebar and data[c] then
|
if c.titlebar and data[c] then
|
||||||
local widgets = c.titlebar.widgets
|
local widgets = c.titlebar.widgets
|
||||||
if prop == "name" then
|
if prop == "name" then
|
||||||
if widgets.title then
|
if widgets.title then
|
||||||
|
@ -140,13 +119,15 @@ function update(c, prop)
|
||||||
if capi.client.focus == c then
|
if capi.client.focus == c then
|
||||||
c.titlebar.fg = data[c].fg_focus
|
c.titlebar.fg = data[c].fg_focus
|
||||||
c.titlebar.bg = data[c].bg_focus
|
c.titlebar.bg = data[c].bg_focus
|
||||||
if widgets.closef then widgets.closef.visible = true end
|
|
||||||
if widgets.close then widgets.close.visible = false end
|
|
||||||
else
|
else
|
||||||
c.titlebar.fg = data[c].fg
|
c.titlebar.fg = data[c].fg
|
||||||
c.titlebar.bg = data[c].bg
|
c.titlebar.bg = data[c].bg
|
||||||
if widgets.closef then widgets.closef.visible = false end
|
end
|
||||||
if widgets.close then widgets.close.visible = true end
|
|
||||||
|
-- iterated of all registered button_sets and update
|
||||||
|
local sets = data[c].button_sets
|
||||||
|
for i = 1, #sets do
|
||||||
|
sets[i].update(c,prop)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -158,6 +139,205 @@ function remove(c)
|
||||||
data[c] = nil
|
data[c] = nil
|
||||||
end
|
end
|
||||||
|
|
||||||
|
-- Create a new button for the toolbar
|
||||||
|
-- @param c The client of the titlebar
|
||||||
|
-- @param name The base name of the button (i.e. close)
|
||||||
|
-- @param modkey ... you know that one, don't you?
|
||||||
|
-- @param theme The theme from beautifull. Used to get the image paths
|
||||||
|
-- @param state The state the button is associated to. Containse path the action and info about the image
|
||||||
|
local function button_new(c, name, modkey, theme, state)
|
||||||
|
local bts =
|
||||||
|
{
|
||||||
|
capi.button({ }, 1, nil, state.action),
|
||||||
|
capi.button({ modkey }, 1, function (t) mouse.client.move(t.client) end),
|
||||||
|
capi.button({ modkey }, 3, function (t) mouse.client.resize(t.client) end)
|
||||||
|
}
|
||||||
|
|
||||||
|
-- get the image path from the theme. Only return a button if we find an image
|
||||||
|
local img
|
||||||
|
img = "titlebar_" .. name .. "_button_" .. state.img
|
||||||
|
img = theme[img]
|
||||||
|
if not img then return end
|
||||||
|
img = image(img)
|
||||||
|
if not img then return end
|
||||||
|
|
||||||
|
-- now create the button
|
||||||
|
local bname = name .. "_" .. state.idx
|
||||||
|
local button = widget.button({ image = img })
|
||||||
|
if not button then return end
|
||||||
|
local rbts = button:buttons()
|
||||||
|
|
||||||
|
for k, v in pairs(rbts) do
|
||||||
|
bts[#bts + 1] = v
|
||||||
|
end
|
||||||
|
|
||||||
|
button:buttons(bts)
|
||||||
|
function button.mouse_enter(s) hooks.user.call('mouse_enter', c) end
|
||||||
|
button.visible = false
|
||||||
|
return button
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Update the buttons in a button group
|
||||||
|
-- @param s The button group to update
|
||||||
|
-- @param c The client of the titlebar
|
||||||
|
-- @param p The property that has changed
|
||||||
|
local function button_group_update(s,c,p)
|
||||||
|
-- hide the currently active button, get the new state and show the new button
|
||||||
|
local n = s.select_state(c,p)
|
||||||
|
if n == nil then return end
|
||||||
|
if (s.active ~= nil) then s.active.visible = false end
|
||||||
|
s.active = s.buttons[n]
|
||||||
|
s.active.visible = true
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Create all buttons in a group
|
||||||
|
-- @param c The client of the titlebar
|
||||||
|
-- @param group The button group to create the buttons for
|
||||||
|
-- @param modkey ...
|
||||||
|
-- @param theme Theme for the image paths
|
||||||
|
local function button_group_create(c, group, modkey, theme )
|
||||||
|
local s = {}
|
||||||
|
s.name = group.name
|
||||||
|
s.select_state = group.select_state
|
||||||
|
s.buttons = {}
|
||||||
|
for n,state in pairs(group.states) do
|
||||||
|
s.buttons[n] = button_new(c, s.name, modkey, theme, state)
|
||||||
|
if (s.buttons[n] == nil) then return end
|
||||||
|
for a,v in pairs(group.attributes) do
|
||||||
|
s.buttons[n][a] = v
|
||||||
|
end
|
||||||
|
end
|
||||||
|
function s.update(c,p) button_group_update(s,c,p) end
|
||||||
|
return s
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Builds a new button group
|
||||||
|
-- @param name The base name for the buttons in the group (i.e. "close")
|
||||||
|
-- @param attrs Common attributes for the buttons (i.e. {align = "right")
|
||||||
|
-- @param sfn State select function.
|
||||||
|
-- @param args The states of the button
|
||||||
|
local function button_group(name, attrs, sfn, ...)
|
||||||
|
local s = {}
|
||||||
|
s.name = name
|
||||||
|
s.select_state = sfn
|
||||||
|
s.attributes = attrs
|
||||||
|
s.states = {}
|
||||||
|
|
||||||
|
for i = 1, #arg do
|
||||||
|
local state = arg[i]
|
||||||
|
s.states[state.idx] = state
|
||||||
|
end
|
||||||
|
|
||||||
|
function s.create(c,modkey, theme) return button_group_create(c,s,modkey, theme) end
|
||||||
|
return s
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Select a state for a client based on an attribute of the client and whether it has focus
|
||||||
|
-- @param c The client of the titlebar
|
||||||
|
-- @param p The property that has changed
|
||||||
|
-- @param a The property to check
|
||||||
|
local function select_state(c,p,a)
|
||||||
|
if (c == nil) then return "n/i" end
|
||||||
|
if capi.client.focus == c then
|
||||||
|
if c[a] then
|
||||||
|
return "f/a"
|
||||||
|
else
|
||||||
|
return "f/i"
|
||||||
|
end
|
||||||
|
else
|
||||||
|
if c[a] then
|
||||||
|
return "n/a"
|
||||||
|
else
|
||||||
|
return "n/i"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Select a state for a client based on whether it's maximized or not
|
||||||
|
-- @param c The client of the titlebar
|
||||||
|
-- @param p The property that has changed
|
||||||
|
local function select_state_maximized(c,p)
|
||||||
|
if (c == nil) then return "n/i" end
|
||||||
|
if capi.client.focus == c then
|
||||||
|
if c.maximized_horizontal or c.maximized_vertical then
|
||||||
|
return "f/a"
|
||||||
|
else
|
||||||
|
return "f/i"
|
||||||
|
end
|
||||||
|
else
|
||||||
|
if c.maximized_horizontal or c.maximized_vertical then
|
||||||
|
return "n/a"
|
||||||
|
else
|
||||||
|
return "n/i"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
-- Select a state for a client based on whether it has focus or not
|
||||||
|
-- @param c The client of the titlebar
|
||||||
|
-- @param p The property that has changed
|
||||||
|
local function select_state_focus(c,p)
|
||||||
|
if c and capi.client.focus == c then
|
||||||
|
return "f"
|
||||||
|
end
|
||||||
|
return "n"
|
||||||
|
end
|
||||||
|
|
||||||
|
-- These are the predefined button groups
|
||||||
|
-- A short explanation using 'close_buttons' as an example:
|
||||||
|
-- "close" : name of the button, the images for this button are taken from the
|
||||||
|
-- theme variables titlebar_close_button_...
|
||||||
|
-- { align ... : attributes of all the buttons
|
||||||
|
-- select_state_focus : This function returns a short string used to describe
|
||||||
|
-- the state. In this case either "n" or "f" depending on
|
||||||
|
-- the focus state of the client. These strings can be
|
||||||
|
-- choosen freely but the< must match one of the idx fuekds
|
||||||
|
-- of the states below
|
||||||
|
-- { idx = "n" ... : This is the state of the button for the 'unfocussed'
|
||||||
|
-- (normal) state. The idx = "n" parameter connects this
|
||||||
|
-- button to the return value of the 'select_state_focus'
|
||||||
|
-- function. The img = "normal" parameter is used to
|
||||||
|
-- determine its image. In this case the iamge is taken from
|
||||||
|
-- the theme variable "titlebar_close_button_normal".
|
||||||
|
-- Finally the last parameter is the action for mouse
|
||||||
|
-- button 1
|
||||||
|
|
||||||
|
local close_buttons = button_group("close",
|
||||||
|
{ align = "left"},
|
||||||
|
select_state_focus,
|
||||||
|
{ idx = "n", img = "normal", action = function (t) t.client:kill() end},
|
||||||
|
{ idx = "f", img = "focus", action = function (t) t.client:kill() end}
|
||||||
|
)
|
||||||
|
|
||||||
|
local ontop_buttons = button_group("ontop",
|
||||||
|
{ align = "right"},
|
||||||
|
function(c,p) return select_state(c, p, "ontop") end,
|
||||||
|
{ idx = "n/i", img = "normal_inactive", action = function(t) t.client.ontop = true ; update(t.client) end},
|
||||||
|
{ idx = "f/i", img = "focus_inactive", action = function(t) t.client.ontop = true ; update(t.client) end},
|
||||||
|
{ idx = "n/a", img = "normal_active", action = function(t) t.client.ontop = false ; update(t.client) end},
|
||||||
|
{ idx = "f/a", img = "focus_active", action = function(t) t.client.ontop = false ; update(t.client) end}
|
||||||
|
)
|
||||||
|
|
||||||
|
local sticky_buttons = button_group("sticky",
|
||||||
|
{ align = "right"},
|
||||||
|
function(c,p) return select_state(c,p,"sticky") end,
|
||||||
|
{ idx = "n/i", img = "normal_inactive", action = function(t) t.client.sticky = true ; update(t.client) end},
|
||||||
|
{ idx = "f/i", img = "focus_inactive", action = function(t) t.client.sticky = true ; update(t.client) end},
|
||||||
|
{ idx = "n/a", img = "normal_active", action = function(t) t.client.sticky = false ; update(t.client) end},
|
||||||
|
{ idx = "f/a", img = "focus_active", action = function(t) t.client.sticky = false ; update(t.client) end}
|
||||||
|
)
|
||||||
|
|
||||||
|
local maximized_buttons = button_group("maximized",
|
||||||
|
{ align = "right"},
|
||||||
|
select_state_maximized,
|
||||||
|
{ idx = "n/i", img = "normal_inactive", action = function(t) t.client.maximized_horizontal = true ; t.client.maximized_vertical = true ; update(t.client) end},
|
||||||
|
{ idx = "f/i", img = "focus_inactive", action = function(t) t.client.maximized_horizontal = true ; t.client.maximized_vertical = true ; update(t.client) end},
|
||||||
|
{ idx = "n/a", img = "normal_active", action = function(t) t.client.maximized_horizontal = false ; t.client.maximized_vertical = false ; update(t.client) end},
|
||||||
|
{ idx = "f/a", img = "focus_active", action = function(t) t.client.maximized_horizontal = false ; t.client.maximized_vertical = false ; update(t.client) end}
|
||||||
|
)
|
||||||
|
|
||||||
|
button_groups = { ontop_buttons, sticky_buttons, maximized_buttons, close_buttons}
|
||||||
|
|
||||||
-- Register standards hooks
|
-- Register standards hooks
|
||||||
hooks.focus.register(update)
|
hooks.focus.register(update)
|
||||||
hooks.unfocus.register(update)
|
hooks.unfocus.register(update)
|
||||||
|
|
|
@ -46,8 +46,28 @@ menu_width = 100
|
||||||
#bg_widget = #cc0000
|
#bg_widget = #cc0000
|
||||||
|
|
||||||
# Define the image to load
|
# Define the image to load
|
||||||
titlebar_close_button_normal = @AWESOME_THEMES_PATH@/default/titlebar/close.png
|
titlebar_close_button_normal = @AWESOME_THEMES_PATH@/default/titlebar/close_normal.png
|
||||||
titlebar_close_button_focus = @AWESOME_THEMES_PATH@/default/titlebar/closer.png
|
titlebar_close_button_focus = @AWESOME_THEMES_PATH@/default/titlebar/close_focus.png
|
||||||
|
|
||||||
|
titlebar_ontop_button_normal_inactive = @AWESOME_THEMES_PATH@/default/titlebar/ontop_normal_inactive.png
|
||||||
|
titlebar_ontop_button_focus_inactive = @AWESOME_THEMES_PATH@/default/titlebar/ontop_focus_inactive.png
|
||||||
|
titlebar_ontop_button_normal_active = @AWESOME_THEMES_PATH@/default/titlebar/ontop_normal_active.png
|
||||||
|
titlebar_ontop_button_focus_active = @AWESOME_THEMES_PATH@/default/titlebar/ontop_focus_active.png
|
||||||
|
|
||||||
|
titlebar_sticky_button_normal_inactive = @AWESOME_THEMES_PATH@/default/titlebar/sticky_normal_inactive.png
|
||||||
|
titlebar_sticky_button_focus_inactive = @AWESOME_THEMES_PATH@/default/titlebar/sticky_focus_inactive.png
|
||||||
|
titlebar_sticky_button_normal_active = @AWESOME_THEMES_PATH@/default/titlebar/sticky_normal_active.png
|
||||||
|
titlebar_sticky_button_focus_active = @AWESOME_THEMES_PATH@/default/titlebar/sticky_focus_active.png
|
||||||
|
|
||||||
|
titlebar_floating_button_normal_inactive = @AWESOME_THEMES_PATH@/default/titlebar/floating_normal_inactive.png
|
||||||
|
titlebar_floating_button_focus_inactive = @AWESOME_THEMES_PATH@/default/titlebar/floating_focus_inactive.png
|
||||||
|
titlebar_floating_button_normal_active = @AWESOME_THEMES_PATH@/default/titlebar/floating_normal_active.png
|
||||||
|
titlebar_floating_button_focus_active = @AWESOME_THEMES_PATH@/default/titlebar/floating_focus_active.png
|
||||||
|
|
||||||
|
titlebar_maximized_button_normal_inactive = @AWESOME_THEMES_PATH@/default/titlebar/maximized_normal_inactive.png
|
||||||
|
titlebar_maximized_button_focus_inactive = @AWESOME_THEMES_PATH@/default/titlebar/maximized_focus_inactive.png
|
||||||
|
titlebar_maximized_button_normal_active = @AWESOME_THEMES_PATH@/default/titlebar/maximized_normal_active.png
|
||||||
|
titlebar_maximized_button_focus_active = @AWESOME_THEMES_PATH@/default/titlebar/maximized_focus_active.png
|
||||||
|
|
||||||
# You can use your own command to set your wallpaper
|
# You can use your own command to set your wallpaper
|
||||||
wallpaper_cmd = awsetbg @AWESOME_THEMES_PATH@/default/background.png
|
wallpaper_cmd = awsetbg @AWESOME_THEMES_PATH@/default/background.png
|
||||||
|
|
Before Width: | Height: | Size: 358 B After Width: | Height: | Size: 358 B |
After Width: | Height: | Size: 666 B |
After Width: | Height: | Size: 431 B |
After Width: | Height: | Size: 598 B |
After Width: | Height: | Size: 414 B |
After Width: | Height: | Size: 386 B |
After Width: | Height: | Size: 349 B |
After Width: | Height: | Size: 1013 B |
After Width: | Height: | Size: 816 B |
After Width: | Height: | Size: 738 B |
After Width: | Height: | Size: 682 B |
After Width: | Height: | Size: 774 B |
After Width: | Height: | Size: 669 B |
After Width: | Height: | Size: 565 B |
After Width: | Height: | Size: 590 B |
After Width: | Height: | Size: 833 B |
After Width: | Height: | Size: 659 B |
After Width: | Height: | Size: 611 B |
After Width: | Height: | Size: 597 B |