From 20030e6f932d222e074d6d6d11d71f4be389e379 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sat, 16 Jan 2016 01:13:07 -0500 Subject: [PATCH] wibox.layout: Add new 'stack' layout This layout display the widgets on top of each other. It can also optionally display only the first one. The most common use case is to create a composited widget. Other use case include the creation of a "paged" stack to only display the most relevant widget without adding extra complexity to the parent layout. --- lib/wibox/layout/init.lua | 1 + lib/wibox/layout/stack.lua | 188 +++++++++++++++++++++++++++++++++++++ 2 files changed, 189 insertions(+) create mode 100644 lib/wibox/layout/stack.lua diff --git a/lib/wibox/layout/init.lua b/lib/wibox/layout/init.lua index f9e1c355..66fbafec 100644 --- a/lib/wibox/layout/init.lua +++ b/lib/wibox/layout/init.lua @@ -17,6 +17,7 @@ return mirror = require("wibox.layout.mirror"); constraint = require("wibox.layout.constraint"); scroll = require("wibox.layout.scroll"); + stack = require("wibox.layout.stack"); } -- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80 diff --git a/lib/wibox/layout/stack.lua b/lib/wibox/layout/stack.lua new file mode 100644 index 00000000..ec11febc --- /dev/null +++ b/lib/wibox/layout/stack.lua @@ -0,0 +1,188 @@ +--------------------------------------------------------------------------- +-- A stacked layout. +-- +-- This layout display widgets on top of each other. It can be used to overlay +-- a `wibox.widget.textbox` on top of a `awful.widget.progressbar` or manage +-- "pages" where only one is visible at any given moment. +-- +-- The indices are going from 1 (the bottom of the stack) up to the top of +-- the stack. The order can be changed either using `:swap` or `:raise`. +-- +-- @author Emmanuel Lepage Vallee +-- @copyright 2016 Emmanuel Lepage Vallee +-- @release @AWESOME_VERSION@ +-- @classmod wibox.layout.stack +--------------------------------------------------------------------------- + +local base = require("wibox.widget.base" ) +local fixed = require("wibox.layout.fixed") +local table = table +local pairs = pairs +local floor = math.floor +local util = require("awful.util") + +local stack = {mt={}} + +--- Get all direct children widgets +-- @param layout The layout you are modifying. +-- @return a list of all widgets +-- @name get_children +-- @class function + +--- Add some widgets to the given stack layout +-- @param layout The layout you are modifying. +-- @tparam widget ... Widgets that should be added (must at least be one) +-- @name add +-- @class function + +--- Set a widget at a specific index, replace the current one +-- @tparam number index A widget or a widget index +-- @param widget2 The widget to take the place of the first one +-- @treturn boolean If the operation is successful +-- @name set +-- @class function + +--- Remove a widget from the layout +-- @tparam index The widget index to remove +-- @treturn boolean index If the operation is successful +-- @name remove +-- @class function + +--- Reset a stack layout. This removes all widgets from the layout. +-- @param layout The layout you are modifying. +-- @name reset +-- @class function + +--- Replace the first instance of `widget` in the layout with `widget2` +-- @param widget The widget to replace +-- @param widget2 The widget to replace `widget` with +-- @tparam[opt=false] boolean recursive Digg in all compatible layouts to find the widget. +-- @treturn boolean If the operation is successful +-- @name replace_widget +-- @class function + +--- Swap 2 widgets in a layout +-- @tparam number index1 The first widget index +-- @tparam number index2 The second widget index +-- @treturn boolean If the operation is successful +-- @name swap +-- @class function + +--- Swap 2 widgets in a layout +-- If widget1 is present multiple time, only the first instance is swapped +-- @param widget1 The first widget +-- @param widget2 The second widget +-- @tparam[opt=false] boolean recursive Digg in all compatible layouts to find the widget. +-- @treturn boolean If the operation is successful +-- @name swap_widgets +-- @class function + +--- Insert a new widget in the layout at position `index` +-- @tparam number index The position +-- @param widget The widget +-- @treturn boolean If the operation is successful +-- @name insert +-- @class function + +--- Remove one or more widgets from the layout +-- The last parameter can be a boolean, forcing a recursive seach of the +-- widget(s) to remove. +-- @param widget ... Widgets that should be removed (must at least be one) +-- @treturn boolean If the operation is successful +-- @name remove_widgets +-- @class function + +--- Add spacing between each layout widgets +-- @param spacing Spacing between widgets. +-- @name set_spacing +-- @class function + +--- Layout a stack layout. Each widget get drawn on top of each other +-- @param context The context in which we are drawn. +-- @param width The available width. +-- @param height The available height. +function stack:layout(context, width, height) + local result = {} + local spacing = self._spacing + + for k, v in pairs(self.widgets) do + table.insert(result, base.place_widget_at(v, spacing, spacing, width - 2*spacing, height - 2*spacing)) + if self._top_only then break end + end + + return result +end + +--- Fit the stack layout into the given space +-- @param context The context in which we are fit. +-- @param orig_width The available width. +-- @param orig_height The available height. +function stack:fit(context, orig_width, orig_height) + local max_w, max_h = 0,0 + local spacing = self._spacing + + for k, v in pairs(self.widgets) do + local w, h = base.fit_widget(self, context, v, orig_width, orig_height) + max_w, max_h = math.max(max_w, w+2*spacing), math.max(max_h, h+2*spacing) + end + + return math.min(max_w, orig_width), math.min(max_h, orig_height) +end + +--- Get if only the first stack widget is drawn +-- @return If the only the first stack widget is drawn +function stack:get_display_top_only() + return self._top_only +end + +--- Only draw the first widget of the stack, ignore others +-- @tparam boolean top_only Only draw the top stack widget +function stack:set_display_top_only(top_only) + self._top_only = top_only +end + +--- Raise a widget at `index` to the top of the stack +-- @tparam number index the widget index to raise +function stack:raise(index) + if (not index) or self.widgets[index] then return end + + local w = self.widgets[index] + table.remove(self.widgets, index) + table.insert(self.widgets, w) + + self:emit_signal("widget::layout_changed") +end + +--- Raise the first instance of `widget` +-- @param widget The widget to raise +-- @tparam[opt=false] boolean recursive Also look deeper in the hierarchy to +-- find the widget +function stack:raise_widget(widget, recursive) + local idx, layout = self:index(widget, recursive) + + if not idx or not layout then return end + + -- Bubble up in the stack until the right index is found + while layout and layout ~= self do + idx, layout = self:index(layout, recursive) + end + + if layout == self and idx ~= 1 then + self:raise(idx) + end +end + +local function new(dir, widget1, ...) + local ret = fixed.horizontal(...) + + util.table.crush(ret, stack) + + return ret +end + +function stack.mt:__call(...) + return new(...) +end + +return setmetatable(stack, stack.mt) +-- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80