From e56f8f30d0d66f90cae4843b0366f2c594f84fad Mon Sep 17 00:00:00 2001 From: Julien Danjou Date: Fri, 27 Feb 2009 17:22:03 +0100 Subject: [PATCH] awful.widget: split Signed-off-by: Julien Danjou --- lib/awful/widget.lua.in | 453 ------------------------------- lib/awful/widget/button.lua.in | 43 +++ lib/awful/widget/common.lua.in | 85 ++++++ lib/awful/widget/init.lua.in | 15 + lib/awful/widget/launcher.lua.in | 36 +++ lib/awful/widget/taglist.lua.in | 177 ++++++++++++ lib/awful/widget/tasklist.lua.in | 176 ++++++++++++ 7 files changed, 532 insertions(+), 453 deletions(-) delete mode 100644 lib/awful/widget.lua.in create mode 100644 lib/awful/widget/button.lua.in create mode 100644 lib/awful/widget/common.lua.in create mode 100644 lib/awful/widget/init.lua.in create mode 100644 lib/awful/widget/launcher.lua.in create mode 100644 lib/awful/widget/taglist.lua.in create mode 100644 lib/awful/widget/tasklist.lua.in diff --git a/lib/awful/widget.lua.in b/lib/awful/widget.lua.in deleted file mode 100644 index 94dd8140d..000000000 --- a/lib/awful/widget.lua.in +++ /dev/null @@ -1,453 +0,0 @@ ---------------------------------------------------------------------------- --- @author Julien Danjou <julien@danjou.info> --- @copyright 2008 Julien Danjou --- @release @AWESOME_VERSION@ ---------------------------------------------------------------------------- - --- Grab environment we need -local ipairs = ipairs -local pairs = pairs -local table = table -local otable = otable -local type = type -local math = math -local capi = -{ - screen = screen, - client = client, - button = button, - widget = widget, - image = image, - mouse = mouse -} -local util = require("awful.util") -local hooks = require("awful.hooks") -local beautiful = require("beautiful") -local menu = require("awful.menu") -local client = require("awful.client") -local tag = require("awful.tag") - ---- Widget module for awful -module("awful.widget") - --- Various public structures -taglist = {} -taglist.label = {} -tasklist = {} -tasklist.label = {} - --- Private structures -local tagwidgets = otable() - -local function list_update(w, buttons, label, data, widgets, objects) - -- Hack: if it has been registered as a widget in a wibox, - -- it's w.len since __len meta does not work on table until Lua 5.2. - -- Otherwise it's standard #w. - local len = (w.len or #w) / 2 - -- Add more widgets - if len < #objects then - for i = len * 2 + 1, #objects * 2, 2 do - local ib = capi.widget({ type = "imagebox", align = widgets.imagebox.align }) - local tb = capi.widget({ type = "textbox", align = widgets.textbox.align }) - - w[i] = ib - w[i + 1] = tb - w[i + 1]:margin({ left = widgets.textbox.margin.left, right = widgets.textbox.margin.right }) - w[i + 1].bg_resize = widgets.textbox.bg_resize or false - w[i + 1].bg_align = widgets.textbox.bg_align or "" - - if type(objects[math.floor(i / 2) + 1]) == "tag" then - tagwidgets[ib] = objects[math.floor(i / 2) + 1] - tagwidgets[tb] = objects[math.floor(i / 2) + 1] - end - end - -- Remove widgets - elseif len > #objects then - for i = #objects * 2 + 1, len * 2, 2 do - w[i] = nil - w[i + 1] = nil - end - end - - -- update widgets text - for k = 1, #objects * 2, 2 do - local o = objects[(k + 1) / 2] - if buttons then - if not data[o] then - data[o] = { } - -- Replace press function by a new one calling with object as - -- argument - for kb, b in ipairs(buttons) do - -- Copy object - data[o][kb] = capi.button(b) - data[o][kb].press = function () b.press(o) end - end - end - w[k]:buttons(data[o]) - w[k + 1]:buttons(data[o]) - end - - local text, bg, bg_image, icon = label(o) - w[k + 1].text, w[k + 1].bg, w[k + 1].bg_image = text, bg, bg_image - w[k].bg, w[k].image = bg, icon - if not w[k + 1].text then - w[k+1].visible = false - else - w[k+1].visible = true - end - if not w[k].image then - w[k].visible = false - else - w[k].visible = true - end - end -end - -local function taglist_update (screen, w, label, buttons, data, widgets) - local tags = capi.screen[screen]:tags() - - list_update(w, buttons, label, data, widgets, tags) -end - ---- Get the tag object the given widget appears on. --- @param widget The widget the look for. --- @return The tag object. -function taglist.gettag(widget) - return tagwidgets[widget] -end - ---- Create a new taglist widget. --- @param screen The screen to draw tag list for. --- @param label Label function to use. --- @param buttons A table with buttons binding to set. -function taglist.new(screen, label, buttons) - local w = {} - local widgets = { } - widgets.imagebox = { } - widgets.textbox = { ["margin"] = { ["left"] = 0, - ["right"] = 0}, - ["bg_resize"] = true - } - local data = otable() - local u = function (s) - if s == screen then - taglist_update(s, w, label, buttons, data, widgets) - end - end - local uc = function (c) return u(c.screen) end - hooks.focus.register(uc) - hooks.unfocus.register(uc) - hooks.arrange.register(u) - hooks.tags.register(u) - hooks.tagged.register(uc) - hooks.property.register(function (c, prop) - if prop == "urgent" then - u(c.screen) - end - end) - -- Free data on tag removal - hooks.tags.register(function (s, tag, action) - if action == "remove" then data[tag] = nil end - end) - u(screen) - return w -end - ---- Return labels for a taglist widget with all tag from screen. --- It returns the tag name and set a special --- foreground and background color for selected tags. --- @param t The tag. --- @param args The arguments table. --- bg_focus The background color for selected tag. --- fg_focus The foreground color for selected tag. --- bg_urgent The background color for urgent tags. --- fg_urgent The foreground color for urgent tags. --- squares_sel Optional: a user provided image for selected squares. --- squares_unsel Optional: a user provided image for unselected squares. --- squares_resize Optional: true or false to resize squares. --- @return A string to print, a background color, a background image and a --- background resize value. -function taglist.label.all(t, args) - if not args then args = {} end - local theme = beautiful.get() - local fg_focus = args.fg_focus or theme.taglist_fg_focus or theme.fg_focus - local bg_focus = args.bg_focus or theme.taglist_bg_focus or theme.bg_focus - local fg_urgent = args.fg_urgent or theme.taglist_fg_urgent or theme.fg_urgent - local bg_urgent = args.bg_urgent or theme.taglist_bg_urgent or theme.bg_urgent - local taglist_squares_sel = args.squares_sel or theme.taglist_squares_sel - local taglist_squares_unsel = args.squares_unsel or theme.taglist_squares_unsel - local taglist_squares_resize = theme.taglist_squares_resize or args.squares_resize or "true" - local font = args.font or theme.taglist_font or theme.font or "" - local text = "" - local sel = capi.client.focus - local bg_color = nil - local fg_color = nil - local bg_image - local icon - local bg_resize = false - if t.selected then - bg_color = bg_focus - fg_color = fg_focus - end - if sel and sel:tags()[t] then - if taglist_squares_sel then - bg_image = capi.image(taglist_squares_sel) - bg_resize = taglist_squares_resize == "true" - end - else - local cls = t:clients() - if #cls > 0 and taglist_squares_unsel then - bg_image = capi.image(taglist_squares_unsel) - bg_resize = taglist_squares_resize == "true" - end - for k, c in pairs(cls) do - if c.urgent then - if bg_urgent then bg_color = bg_urgent end - if fg_urgent then fg_color = fg_urgent end - break - end - end - end - local taglist_squares = false - if taglist_squares_sel or taglist_squares_unsel then - taglist_squares = true - end - if t.name then - if fg_color then - text = text .. "" - if taglist_squares then - text = text .. " " - end - text = text..util.escape(t.name).." " - else - if taglist_squares then - text = text .. " " - end - text = text .. util.escape(t.name) .. " " - end - elseif taglist_squares then - text = text .. " " - end - text = text .. "" - if tag.geticon(t) and type(tag.geticon(t)) == "image" then - icon = tag.geticon(t) - elseif tag.geticon(t) then - icon = capi.image(tag.geticon(t)) - end - - return text, bg_color, bg_image, icon -end - ---- Return labels for a taglist widget with all *non empty* tags from screen. --- It returns the tag name and set a special --- foreground and background color for selected tags. --- @param t The tag. --- @param args The arguments table. --- bg_focus The background color for selected tag. --- fg_focus The foreground color for selected tag. --- bg_urgent The background color for urgent tags. --- fg_urgent The foreground color for urgent tags. --- @return A string to print, a background color, a background image and a --- background resize value. -function taglist.label.noempty(t, args) - if #t:clients() > 0 or t.selected then - return taglist.label.all(t, args) - end -end - -local function tasklist_update(w, buttons, label, data, widgets) - local clients = capi.client.get() - local shownclients = {} - for k, c in ipairs(clients) do - if not (c.skip_taskbar or c.hide - or c.type == "splash" or c.type == "dock" or c.type == "desktop") then - table.insert(shownclients, c) - end - end - clients = shownclients - - list_update(w, buttons, label, data, widgets, clients) -end - ---- Create a new tasklist widget. --- @param label Label function to use. --- @param buttons A table with buttons binding to set. -function tasklist.new(label, buttons) - local w = {} - local widgets = { } - widgets.imagebox = { ["align"] = "flex" } - widgets.textbox = { ["align"] = "flex", - ["margin"] = { ["left"] = 2, - ["right"] = 2 }, - ["bg_resize"] = true, - ["bg_align"] = "right" - } - local data = otable() - local u = function () tasklist_update(w, buttons, label, data, widgets) end - hooks.arrange.register(u) - hooks.clients.register(u) - hooks.tagged.register(u) - hooks.focus.register(u) - hooks.unfocus.register(u) - hooks.property.register(function (c, prop) - if prop == "urgent" - or prop == "floating" - or prop == "maximized_horizontal" - or prop == "maximized_vertical" - or prop == "icon" - or prop == "name" - or prop == "icon_name" then - u() - end - end) - u() - -- Free data on unmanage - hooks.unmanage.register(function (c) data[c] = nil end) - return w -end - -local function widget_tasklist_label_common(c, args) - if not args then args = {} end - local theme = beautiful.get() - local fg_focus = args.fg_focus or theme.tasklist_fg_focus or theme.fg_focus - local bg_focus = args.bg_focus or theme.tasklist_bg_focus or theme.bg_focus - local fg_urgent = args.fg_urgent or theme.tasklist_fg_urgent or theme.fg_urgent - local bg_urgent = args.bg_urgent or theme.tasklist_bg_urgent or theme.bg_urgent - local fg_minimize = args.fg_minimize or theme.tasklist_fg_minimize or theme.fg_minimize - local bg_minimize = args.bg_minimize or theme.tasklist_bg_minimize or theme.bg_minimize - local floating_icon = args.floating_icon or theme.tasklist_floating_icon - local font = args.font or theme.tasklist_font or theme.font or "" - local bg = nil - local text = "" - local name - local status_image - if client.floating.get(c) and floating_icon then - status_image = capi.image(floating_icon) - end - if c.minimized then - name = util.escape(c.icon_name) or util.escape("") - else - name = util.escape(c.name) or util.escape("") - end - if capi.client.focus == c then - bg = bg_focus - if fg_focus then - text = text .. ""..name.."" - else - text = text .. name - end - elseif c.urgent and fg_urgent then - bg = bg_urgent - text = text .. ""..name.."" - elseif c.minimized and fg_minimize and bg_minimize then - bg = bg_minimize - text = text .. ""..name.."" - else - text = text .. name - end - text = text .. "" - return text, bg, status_image, c.icon -end - ---- Return labels for a tasklist widget with clients from all tags and screen. --- It returns the client name and set a special --- foreground and background color for focused client. --- It also puts a special icon for floating windows. --- @param c The client. --- @param screen The screen we are drawing on. --- @param args The arguments table. --- bg_focus The background color for focused client. --- fg_focus The foreground color for focused client. --- bg_urgent The background color for urgent clients. --- fg_urgent The foreground color for urgent clients. --- @return A string to print, a background color and a status image. -function tasklist.label.allscreen(c, screen, args) - return widget_tasklist_label_common(c, args) -end - ---- Return labels for a tasklist widget with clients from all tags. --- It returns the client name and set a special --- foreground and background color for focused client. --- It also puts a special icon for floating windows. --- @param c The client. --- @param screen The screen we are drawing on. --- @param args The arguments table. --- bg_focus The background color for focused client. --- fg_focus The foreground color for focused client. --- bg_urgent The background color for urgent clients. --- fg_urgent The foreground color for urgent clients. --- @return A string to print, a background color and a status image. -function tasklist.label.alltags(c, screen, args) - -- Only print client on the same screen as this widget - if c.screen ~= screen then return end - return widget_tasklist_label_common(c, args) -end - ---- Return labels for a tasklist widget with clients from currently selected tags. --- It returns the client name and set a special --- foreground and background color for focused client. --- It also puts a special icon for floating windows. --- @param c The client. --- @param screen The screen we are drawing on. --- @param args The arguments table. --- bg_focus The background color for focused client. --- fg_focus The foreground color for focused client. --- bg_urgent The background color for urgent clients. --- fg_urgent The foreground color for urgent clients. --- @return A string to print, a background color and a status image. -function tasklist.label.currenttags(c, screen, args) - -- Only print client on the same screen as this widget - if c.screen ~= screen then return end - for k, t in ipairs(capi.screen[screen]:tags()) do - if t.selected and c:tags()[t] then - return widget_tasklist_label_common(c, args) - end - end -end - ---- Create a button widget. When clicked, the image is deplaced to make it like --- a real button. --- @param args Standard widget table arguments, plus image for the image path or --- the image object. --- @return A textbox widget configured as a button. -function button(args) - if not args or not args.image then return end - local img_release - if type(args.image) == "string" then - img_release = capi.image(args.image) - elseif type(args.image) == "image" then - img_release = args.image - else - return - end - local img_press = img_release:crop(-2, -2, img_release.width, img_release.height) - args.type = "imagebox" - local w = capi.widget(args) - w.image = img_release - w:buttons({ capi.button({}, 1, function () w.image = img_press end, function () w.image = img_release end) }) - function w.mouse_leave(s) w.image = img_release end - function w.mouse_enter(s) if capi.mouse.coords().buttons[1] then w.image = img_press end end - return w -end - ---- Create a button widget which will launch a command. --- @param args Standard widget table arguments, plus image for the image path --- and command for the command to run on click, or either menu to create menu. --- @return A launcher widget. -function launcher(args) - if not args.command and not args.menu then return end - local w = button(args) - if not w then return end - local b = w:buttons() - - if args.command then - b[#b + 1] = capi.button({}, 1, nil, function () util.spawn(args.command) end) - elseif args.menu then - b[#b + 1] = capi.button({}, 1, nil, function () args.menu:toggle() end) - end - - w:buttons(b) - return w -end - --- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80 diff --git a/lib/awful/widget/button.lua.in b/lib/awful/widget/button.lua.in new file mode 100644 index 000000000..2e5bd3457 --- /dev/null +++ b/lib/awful/widget/button.lua.in @@ -0,0 +1,43 @@ +--------------------------------------------------------------------------- +-- @author Julien Danjou <julien@danjou.info> +-- @copyright 2008-2009 Julien Danjou +-- @release @AWESOME_VERSION@ +--------------------------------------------------------------------------- + +local setmetatable = setmetatable +local type = type +local capi = { image = image, + widget = widget, + mouse = mouse, + button = button } + +module("awful.widget.button") + +--- Create a button widget. When clicked, the image is deplaced to make it like +-- a real button. +-- @param args Standard widget table arguments, plus image for the image path or +-- the image object. +-- @return A textbox widget configured as a button. +function new(_, args) + if not args or not args.image then return end + local img_release + if type(args.image) == "string" then + img_release = capi.image(args.image) + elseif type(args.image) == "image" then + img_release = args.image + else + return + end + local img_press = img_release:crop(-2, -2, img_release.width, img_release.height) + args.type = "imagebox" + local w = capi.widget(args) + w.image = img_release + w:buttons({ capi.button({}, 1, function () w.image = img_press end, function () w.image = img_release end) }) + function w.mouse_leave(s) w.image = img_release end + function w.mouse_enter(s) if capi.mouse.coords().buttons[1] then w.image = img_press end end + return w +end + +setmetatable(_M, { __call = new }) + +-- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80 diff --git a/lib/awful/widget/common.lua.in b/lib/awful/widget/common.lua.in new file mode 100644 index 000000000..a94fb866a --- /dev/null +++ b/lib/awful/widget/common.lua.in @@ -0,0 +1,85 @@ +--------------------------------------------------------------------------- +-- @author Julien Danjou <julien@danjou.info> +-- @copyright 2008-2009 Julien Danjou +-- @release @AWESOME_VERSION@ +--------------------------------------------------------------------------- + +-- Grab environment we need +local math = math +local type = type +local otable = otable +local ipairs = ipairs +local capi = { widget = widget, + button = button } + +--- Common widget code +module("awful.widget.common") + +-- Private structures +tagwidgets = otable() + +function list_update(w, buttons, label, data, widgets, objects) + -- Hack: if it has been registered as a widget in a wibox, + -- it's w.len since __len meta does not work on table until Lua 5.2. + -- Otherwise it's standard #w. + local len = (w.len or #w) / 2 + -- Add more widgets + if len < #objects then + for i = len * 2 + 1, #objects * 2, 2 do + local ib = capi.widget({ type = "imagebox", align = widgets.imagebox.align }) + local tb = capi.widget({ type = "textbox", align = widgets.textbox.align }) + + w[i] = ib + w[i + 1] = tb + w[i + 1]:margin({ left = widgets.textbox.margin.left, right = widgets.textbox.margin.right }) + w[i + 1].bg_resize = widgets.textbox.bg_resize or false + w[i + 1].bg_align = widgets.textbox.bg_align or "" + + if type(objects[math.floor(i / 2) + 1]) == "tag" then + tagwidgets[ib] = objects[math.floor(i / 2) + 1] + tagwidgets[tb] = objects[math.floor(i / 2) + 1] + end + end + -- Remove widgets + elseif len > #objects then + for i = #objects * 2 + 1, len * 2, 2 do + w[i] = nil + w[i + 1] = nil + end + end + + -- update widgets text + for k = 1, #objects * 2, 2 do + local o = objects[(k + 1) / 2] + if buttons then + if not data[o] then + data[o] = { } + -- Replace press function by a new one calling with object as + -- argument + for kb, b in ipairs(buttons) do + -- Copy object + data[o][kb] = capi.button(b) + data[o][kb].press = function () b.press(o) end + end + end + w[k]:buttons(data[o]) + w[k + 1]:buttons(data[o]) + end + + local text, bg, bg_image, icon = label(o) + w[k + 1].text, w[k + 1].bg, w[k + 1].bg_image = text, bg, bg_image + w[k].bg, w[k].image = bg, icon + if not w[k + 1].text then + w[k+1].visible = false + else + w[k+1].visible = true + end + if not w[k].image then + w[k].visible = false + else + w[k].visible = true + end + end +end + +-- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80 diff --git a/lib/awful/widget/init.lua.in b/lib/awful/widget/init.lua.in new file mode 100644 index 000000000..0451385a0 --- /dev/null +++ b/lib/awful/widget/init.lua.in @@ -0,0 +1,15 @@ +--------------------------------------------------------------------------- +-- @author Julien Danjou <julien@danjou.info> +-- @copyright 2008-2009 Julien Danjou +-- @release @AWESOME_VERSION@ +--------------------------------------------------------------------------- + +require("awful.widget.taglist") +require("awful.widget.tasklist") +require("awful.widget.button") +require("awful.widget.launcher") + +--- Widget module for awful +module("awful.widget") + +-- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80 diff --git a/lib/awful/widget/launcher.lua.in b/lib/awful/widget/launcher.lua.in new file mode 100644 index 000000000..1ad63c931 --- /dev/null +++ b/lib/awful/widget/launcher.lua.in @@ -0,0 +1,36 @@ +--------------------------------------------------------------------------- +-- @author Julien Danjou <julien@danjou.info> +-- @copyright 2008-2009 Julien Danjou +-- @release @AWESOME_VERSION@ +--------------------------------------------------------------------------- + +local setmetatable = setmetatable +local capi = { button = button } +local util = require("awful.util") +local button = require("awful.widget.button") + +module("awful.widget.launcher") + +--- Create a button widget which will launch a command. +-- @param args Standard widget table arguments, plus image for the image path +-- and command for the command to run on click, or either menu to create menu. +-- @return A launcher widget. +local function new(_, args) + if not args.command and not args.menu then return end + local w = button(args) + if not w then return end + local b = w:buttons() + + if args.command then + b[#b + 1] = capi.button({}, 1, nil, function () util.spawn(args.command) end) + elseif args.menu then + b[#b + 1] = capi.button({}, 1, nil, function () args.menu:toggle() end) + end + + w:buttons(b) + return w +end + +setmetatable(_M, { __call = new }) + +-- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80 diff --git a/lib/awful/widget/taglist.lua.in b/lib/awful/widget/taglist.lua.in new file mode 100644 index 000000000..4d339b9da --- /dev/null +++ b/lib/awful/widget/taglist.lua.in @@ -0,0 +1,177 @@ +--------------------------------------------------------------------------- +-- @author Julien Danjou <julien@danjou.info> +-- @copyright 2008-2009 Julien Danjou +-- @release @AWESOME_VERSION@ +--------------------------------------------------------------------------- + +-- Grab environment we need +local capi = { widget = widget, + screen = screen, + image = image, + client = client } +local otable = otable +local type = type +local pairs = pairs +local hooks = require("awful.hooks") +local common = require("awful.widget.common") +local util = require("awful.util") +local tag = require("awful.tag") +local beautiful = require("beautiful") + +--- Taglist widget module for awful +module("awful.widget.taglist") + +label = {} + +local function taglist_update (screen, w, label, buttons, data, widgets) + local tags = capi.screen[screen]:tags() + common.list_update(w, buttons, label, data, widgets, tags) +end + +--- Get the tag object the given widget appears on. +-- @param widget The widget the look for. +-- @return The tag object. +function gettag(widget) + return common.tagwidgets[widget] +end + +--- Create a new taglist widget. +-- @param screen The screen to draw tag list for. +-- @param label Label function to use. +-- @param buttons A table with buttons binding to set. +function new(screen, label, buttons) + local w = {} + local widgets = { } + widgets.imagebox = { } + widgets.textbox = { ["margin"] = { ["left"] = 0, + ["right"] = 0}, + ["bg_resize"] = true + } + local data = otable() + local u = function (s) + if s == screen then + taglist_update(s, w, label, buttons, data, widgets) + end + end + local uc = function (c) return u(c.screen) end + hooks.focus.register(uc) + hooks.unfocus.register(uc) + hooks.arrange.register(u) + hooks.tags.register(u) + hooks.tagged.register(uc) + hooks.property.register(function (c, prop) + if prop == "urgent" then + u(c.screen) + end + end) + -- Free data on tag removal + hooks.tags.register(function (s, tag, action) + if action == "remove" then data[tag] = nil end + end) + u(screen) + return w +end + +--- Return labels for a taglist widget with all tag from screen. +-- It returns the tag name and set a special +-- foreground and background color for selected tags. +-- @param t The tag. +-- @param args The arguments table. +-- bg_focus The background color for selected tag. +-- fg_focus The foreground color for selected tag. +-- bg_urgent The background color for urgent tags. +-- fg_urgent The foreground color for urgent tags. +-- squares_sel Optional: a user provided image for selected squares. +-- squares_unsel Optional: a user provided image for unselected squares. +-- squares_resize Optional: true or false to resize squares. +-- @return A string to print, a background color, a background image and a +-- background resize value. +function label.all(t, args) + if not args then args = {} end + local theme = beautiful.get() + local fg_focus = args.fg_focus or theme.taglist_fg_focus or theme.fg_focus + local bg_focus = args.bg_focus or theme.taglist_bg_focus or theme.bg_focus + local fg_urgent = args.fg_urgent or theme.taglist_fg_urgent or theme.fg_urgent + local bg_urgent = args.bg_urgent or theme.taglist_bg_urgent or theme.bg_urgent + local taglist_squares_sel = args.squares_sel or theme.taglist_squares_sel + local taglist_squares_unsel = args.squares_unsel or theme.taglist_squares_unsel + local taglist_squares_resize = theme.taglist_squares_resize or args.squares_resize or "true" + local font = args.font or theme.taglist_font or theme.font or "" + local text = "" + local sel = capi.client.focus + local bg_color = nil + local fg_color = nil + local bg_image + local icon + local bg_resize = false + if t.selected then + bg_color = bg_focus + fg_color = fg_focus + end + if sel and sel:tags()[t] then + if taglist_squares_sel then + bg_image = capi.image(taglist_squares_sel) + bg_resize = taglist_squares_resize == "true" + end + else + local cls = t:clients() + if #cls > 0 and taglist_squares_unsel then + bg_image = capi.image(taglist_squares_unsel) + bg_resize = taglist_squares_resize == "true" + end + for k, c in pairs(cls) do + if c.urgent then + if bg_urgent then bg_color = bg_urgent end + if fg_urgent then fg_color = fg_urgent end + break + end + end + end + local taglist_squares = false + if taglist_squares_sel or taglist_squares_unsel then + taglist_squares = true + end + if t.name then + if fg_color then + text = text .. "" + if taglist_squares then + text = text .. " " + end + text = text..util.escape(t.name).." " + else + if taglist_squares then + text = text .. " " + end + text = text .. util.escape(t.name) .. " " + end + elseif taglist_squares then + text = text .. " " + end + text = text .. "" + if tag.geticon(t) and type(tag.geticon(t)) == "image" then + icon = tag.geticon(t) + elseif tag.geticon(t) then + icon = capi.image(tag.geticon(t)) + end + + return text, bg_color, bg_image, icon +end + +--- Return labels for a taglist widget with all *non empty* tags from screen. +-- It returns the tag name and set a special +-- foreground and background color for selected tags. +-- @param t The tag. +-- @param args The arguments table. +-- bg_focus The background color for selected tag. +-- fg_focus The foreground color for selected tag. +-- bg_urgent The background color for urgent tags. +-- fg_urgent The foreground color for urgent tags. +-- @return A string to print, a background color, a background image and a +-- background resize value. +function label.noempty(t, args) + if #t:clients() > 0 or t.selected then + return label.all(t, args) + end +end + +-- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80 diff --git a/lib/awful/widget/tasklist.lua.in b/lib/awful/widget/tasklist.lua.in new file mode 100644 index 000000000..3f94ec21f --- /dev/null +++ b/lib/awful/widget/tasklist.lua.in @@ -0,0 +1,176 @@ +--------------------------------------------------------------------------- +-- @author Julien Danjou <julien@danjou.info> +-- @copyright 2008-2009 Julien Danjou +-- @release @AWESOME_VERSION@ +--------------------------------------------------------------------------- + +-- Grab environment we need +local capi = { screen = screen, + image = image, + client = client } +local otable = otable +local ipairs = ipairs +local table = table +local hooks = require("awful.hooks") +local common = require("awful.widget.common") +local beautiful = require("beautiful") +local client = require("awful.client") +local util = require("awful.util") + +--- Tasklist widget module for awful +module("awful.widget.tasklist") + +-- Public structures +label = {} + +local function tasklist_update(w, buttons, label, data, widgets) + local clients = capi.client.get() + local shownclients = {} + for k, c in ipairs(clients) do + if not (c.skip_taskbar or c.hide + or c.type == "splash" or c.type == "dock" or c.type == "desktop") then + table.insert(shownclients, c) + end + end + clients = shownclients + + common.list_update(w, buttons, label, data, widgets, clients) +end + +--- Create a new tasklist widget. +-- @param label Label function to use. +-- @param buttons A table with buttons binding to set. +function new(label, buttons) + local w = {} + local widgets = { } + widgets.imagebox = { align = "flex" } + widgets.textbox = { align = "flex", + margin = { left = 2, + right = 2 }, + bg_resize = true, + bg_align = "right" + } + local data = otable() + local u = function () tasklist_update(w, buttons, label, data, widgets) end + hooks.arrange.register(u) + hooks.clients.register(u) + hooks.tagged.register(u) + hooks.focus.register(u) + hooks.unfocus.register(u) + hooks.property.register(function (c, prop) + if prop == "urgent" + or prop == "floating" + or prop == "maximized_horizontal" + or prop == "maximized_vertical" + or prop == "icon" + or prop == "name" + or prop == "icon_name" then + u() + end + end) + u() + -- Free data on unmanage + hooks.unmanage.register(function (c) data[c] = nil end) + return w +end + +local function widget_tasklist_label_common(c, args) + if not args then args = {} end + local theme = beautiful.get() + local fg_focus = args.fg_focus or theme.tasklist_fg_focus or theme.fg_focus + local bg_focus = args.bg_focus or theme.tasklist_bg_focus or theme.bg_focus + local fg_urgent = args.fg_urgent or theme.tasklist_fg_urgent or theme.fg_urgent + local bg_urgent = args.bg_urgent or theme.tasklist_bg_urgent or theme.bg_urgent + local fg_minimize = args.fg_minimize or theme.tasklist_fg_minimize or theme.fg_minimize + local bg_minimize = args.bg_minimize or theme.tasklist_bg_minimize or theme.bg_minimize + local floating_icon = args.floating_icon or theme.tasklist_floating_icon + local font = args.font or theme.tasklist_font or theme.font or "" + local bg = nil + local text = "" + local name + local status_image + if client.floating.get(c) and floating_icon then + status_image = capi.image(floating_icon) + end + if c.minimized then + name = util.escape(c.icon_name) or util.escape("") + else + name = util.escape(c.name) or util.escape("") + end + if capi.client.focus == c then + bg = bg_focus + if fg_focus then + text = text .. ""..name.."" + else + text = text .. name + end + elseif c.urgent and fg_urgent then + bg = bg_urgent + text = text .. ""..name.."" + elseif c.minimized and fg_minimize and bg_minimize then + bg = bg_minimize + text = text .. ""..name.."" + else + text = text .. name + end + text = text .. "" + return text, bg, status_image, c.icon +end + +--- Return labels for a tasklist widget with clients from all tags and screen. +-- It returns the client name and set a special +-- foreground and background color for focused client. +-- It also puts a special icon for floating windows. +-- @param c The client. +-- @param screen The screen we are drawing on. +-- @param args The arguments table. +-- bg_focus The background color for focused client. +-- fg_focus The foreground color for focused client. +-- bg_urgent The background color for urgent clients. +-- fg_urgent The foreground color for urgent clients. +-- @return A string to print, a background color and a status image. +function label.allscreen(c, screen, args) + return widget_tasklist_label_common(c, args) +end + +--- Return labels for a tasklist widget with clients from all tags. +-- It returns the client name and set a special +-- foreground and background color for focused client. +-- It also puts a special icon for floating windows. +-- @param c The client. +-- @param screen The screen we are drawing on. +-- @param args The arguments table. +-- bg_focus The background color for focused client. +-- fg_focus The foreground color for focused client. +-- bg_urgent The background color for urgent clients. +-- fg_urgent The foreground color for urgent clients. +-- @return A string to print, a background color and a status image. +function label.alltags(c, screen, args) + -- Only print client on the same screen as this widget + if c.screen ~= screen then return end + return widget_tasklist_label_common(c, args) +end + +--- Return labels for a tasklist widget with clients from currently selected tags. +-- It returns the client name and set a special +-- foreground and background color for focused client. +-- It also puts a special icon for floating windows. +-- @param c The client. +-- @param screen The screen we are drawing on. +-- @param args The arguments table. +-- bg_focus The background color for focused client. +-- fg_focus The foreground color for focused client. +-- bg_urgent The background color for urgent clients. +-- fg_urgent The foreground color for urgent clients. +-- @return A string to print, a background color and a status image. +function label.currenttags(c, screen, args) + -- Only print client on the same screen as this widget + if c.screen ~= screen then return end + for k, t in ipairs(capi.screen[screen]:tags()) do + if t.selected and c:tags()[t] then + return widget_tasklist_label_common(c, args) + end + end +end + +-- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=80