249 lines
5.1 KiB
Lua
249 lines
5.1 KiB
Lua
local awful = require('awful')
|
|
local wibox = require('wibox')
|
|
local dpi = require('beautiful').xresources.apply_dpi
|
|
local capi = {button = _G.button}
|
|
local gears = require('gears')
|
|
|
|
local clickable_container = require('widget.clickable-container')
|
|
local icons = require('theme.icons')
|
|
|
|
--- sub string for utf8 format
|
|
-- @s the string need to be sub
|
|
-- @i index of start position
|
|
-- @j index of end position
|
|
local function utf8_sub(s, i, j)
|
|
i = utf8.offset(s, i)
|
|
j = (utf8.offset(s, j + 1) or j + 1) - 1
|
|
return gears.string.xml_escape(s:sub(i, j))
|
|
end
|
|
|
|
--- Common method to create buttons.
|
|
-- @tab buttons
|
|
-- @param object
|
|
-- @treturn table
|
|
local function create_buttons(buttons, object)
|
|
if buttons then
|
|
local btns = {}
|
|
for _, b in ipairs(buttons) do
|
|
-- Create a proxy button object: it will receive the real
|
|
-- press and release events, and will propagate them to the
|
|
-- button object the user provided, but with the object as
|
|
-- argument.
|
|
local btn = capi.button {modifiers = b.modifiers, button = b.button}
|
|
btn:connect_signal(
|
|
'press',
|
|
function()
|
|
b:emit_signal('press', object)
|
|
end
|
|
)
|
|
btn:connect_signal(
|
|
'release',
|
|
function()
|
|
b:emit_signal('release', object)
|
|
end
|
|
)
|
|
btns[#btns + 1] = btn
|
|
end
|
|
|
|
return btns
|
|
end
|
|
end
|
|
|
|
local function list_update(w, buttons, label, data, objects)
|
|
-- update the widgets, creating them if needed
|
|
w:reset()
|
|
for i, o in ipairs(objects) do
|
|
local cache = data[o]
|
|
local ib, cb, tb, cbm, bgb, tbm, ibm, tt, l, ll, bg_clickable
|
|
if cache then
|
|
ib = cache.ib
|
|
tb = cache.tb
|
|
bgb = cache.bgb
|
|
tbm = cache.tbm
|
|
ibm = cache.ibm
|
|
tt = cache.tt
|
|
else
|
|
ib = wibox.widget.imagebox()
|
|
tb = wibox.widget.textbox()
|
|
cb = wibox.widget {
|
|
{
|
|
{
|
|
image = icons.close,
|
|
resize = true,
|
|
widget = wibox.widget.imagebox
|
|
},
|
|
margins = dpi(4),
|
|
widget = wibox.container.margin
|
|
},
|
|
widget = clickable_container
|
|
}
|
|
cb.shape = gears.shape.circle
|
|
cbm = wibox.widget {
|
|
-- 4, 8 ,12 ,12 -- close button
|
|
cb,
|
|
left = dpi(4),
|
|
right = dpi(8),
|
|
top = dpi(4),
|
|
bottom = dpi(4),
|
|
widget = wibox.container.margin
|
|
}
|
|
cbm:buttons(
|
|
gears.table.join(
|
|
awful.button(
|
|
{},
|
|
1,
|
|
nil,
|
|
function()
|
|
o:kill()
|
|
end
|
|
)
|
|
)
|
|
)
|
|
bg_clickable = clickable_container()
|
|
bgb = wibox.container.background()
|
|
tbm = wibox.widget {
|
|
tb,
|
|
left = dpi(4),
|
|
right = dpi(4),
|
|
widget = wibox.container.margin
|
|
}
|
|
ibm = wibox.widget {
|
|
-- 12 top bottom
|
|
ib,
|
|
left = dpi(6),
|
|
right = dpi(6),
|
|
top = dpi(6),
|
|
bottom = dpi(6),
|
|
widget = wibox.container.margin
|
|
}
|
|
l = wibox.layout.fixed.horizontal()
|
|
ll = wibox.layout.fixed.horizontal()
|
|
|
|
-- All of this is added in a fixed widget
|
|
l:fill_space(true)
|
|
l:add(ibm)
|
|
l:add(tbm)
|
|
ll:add(l)
|
|
ll:add(cbm)
|
|
|
|
bg_clickable:set_widget(ll)
|
|
-- And all of this gets a background
|
|
bgb:set_widget(bg_clickable)
|
|
|
|
l:buttons(create_buttons(buttons, o))
|
|
|
|
-- Tooltip to display whole title, if it was truncated
|
|
tt = awful.tooltip({
|
|
objects = {tb},
|
|
mode = 'outside',
|
|
align = 'bottom',
|
|
delay_show = 1,
|
|
})
|
|
|
|
data[o] = {
|
|
ib = ib,
|
|
tb = tb,
|
|
bgb = bgb,
|
|
tbm = tbm,
|
|
ibm = ibm,
|
|
tt = tt
|
|
}
|
|
end
|
|
|
|
local text, bg, bg_image, icon, args = label(o, tb)
|
|
args = args or {}
|
|
|
|
-- The text might be invalid, so use pcall.
|
|
if text == nil or text == '' then
|
|
tbm:set_margins(0)
|
|
else
|
|
-- truncate when title is too long
|
|
local text_only = text:match('>(.-)<')
|
|
if (text_only:len() > 24) then
|
|
text = text:gsub('>(.-)<', '>' .. utf8_sub(text_only,1,21) .. '...<')
|
|
tt:set_text(text_only)
|
|
tt:add_to_object(tb)
|
|
else
|
|
tt:remove_from_object(tb)
|
|
end
|
|
if not tb:set_markup_silently(text) then
|
|
tb:set_markup('<i><Invalid text></i>')
|
|
end
|
|
end
|
|
bgb:set_bg(bg)
|
|
if type(bg_image) == 'function' then
|
|
-- TODO: Why does this pass nil as an argument?
|
|
bg_image = bg_image(tb, o, nil, objects, i)
|
|
end
|
|
bgb:set_bgimage(bg_image)
|
|
if icon then
|
|
ib.image = gears.surface(icon)
|
|
else
|
|
ibm:set_margins(0)
|
|
end
|
|
|
|
bgb.shape = args.shape
|
|
bgb.shape_border_width = args.shape_border_width
|
|
bgb.shape_border_color = args.shape_border_color
|
|
|
|
w:add(bgb)
|
|
end
|
|
end
|
|
local tasklist_buttons =
|
|
awful.util.table.join(
|
|
awful.button(
|
|
{},
|
|
1,
|
|
function(c)
|
|
if c == _G.client.focus then
|
|
c.minimized = true
|
|
else
|
|
-- Without this, the following
|
|
-- :isvisible() makes no sense
|
|
c.minimized = false
|
|
if not c:isvisible() and c.first_tag then
|
|
c.first_tag:view_only()
|
|
end
|
|
-- This will also un-minimize
|
|
-- the client, if needed
|
|
_G.client.focus = c
|
|
c:raise()
|
|
end
|
|
end
|
|
),
|
|
awful.button(
|
|
{},
|
|
2,
|
|
function(c)
|
|
c:kill()
|
|
end
|
|
),
|
|
awful.button(
|
|
{},
|
|
4,
|
|
function()
|
|
awful.client.focus.byidx(1)
|
|
end
|
|
),
|
|
awful.button(
|
|
{},
|
|
5,
|
|
function()
|
|
awful.client.focus.byidx(-1)
|
|
end
|
|
)
|
|
)
|
|
|
|
local TaskList = function(s)
|
|
return awful.widget.tasklist(
|
|
s,
|
|
awful.widget.tasklist.filter.currenttags,
|
|
tasklist_buttons,
|
|
{},
|
|
list_update,
|
|
wibox.layout.fixed.horizontal()
|
|
)
|
|
end
|
|
|
|
return TaskList
|