--------------------------------------------------------------------------- --- Place multiple widgets in multiple rows and columns. -- -- Widgets spanning several columns or rows cannot be included using the -- declarative system. -- Instead, create the grid layout and call the `add_widget_at` method. -- --@DOC_wibox_layout_grid_imperative_EXAMPLE@ -- -- The same can be done using the declarative syntax. When using this mode, -- all widgets will again the `row_span` and `col_span` properties: -- --@DOC_wibox_layout_grid_declarative1_EXAMPLE@ -- -- When `col_index` and `row_index` are not provided, the widgets are -- automatically added next to each other spanning only one cell: -- --@DOC_wibox_layout_grid_declarative2_EXAMPLE@ -- --@DOC_wibox_layout_defaults_grid_EXAMPLE@ -- @author getzze -- @copyright 2017 getzze -- @layoutmod wibox.layout.grid -- @supermodule wibox.widget.base --------------------------------------------------------------------------- local setmetatable = setmetatable local unpack = unpack or table.unpack -- luacheck: globals unpack (compatibility with Lua 5.1) local table = table local pairs = pairs local ipairs = ipairs local math = math local gtable = require("gears.table") local base = require("wibox.widget.base") local grid = { mt = {} } local properties = { "orientation", "superpose", "forced_num_rows", "forced_num_cols", "min_cols_size", "min_rows_size", } local dir_properties = { "spacing", "homogeneous", "expand" } --- Set the preferred orientation of the grid layout. -- -- When calling `get_next_empty`, empty cells are browsed differently. -- --@DOC_wibox_layout_grid_orientation_EXAMPLE@ -- @tparam[opt="vertical"] string orientation Preferred orientation. -- @propertyvalue "horizontal" The grid can be extended horizontally. The current -- column is filled first; if no empty cell is found up to `forced_num_rows`, -- the next column is filled, creating it if it does not exist. -- @propertyvalue "vertical" The grid can be extended vertically. The current row is -- filled first; if no empty cell is found up to `forced_num_cols`, the next -- row is filled, creating it if it does not exist. -- @property orientation --- Allow to superpose widgets in the same cell. -- If false, check before adding a new widget if it will superpose with another -- widget and prevent from adding it. -- --@DOC_wibox_layout_grid_superpose_EXAMPLE@ -- @tparam[opt=false] boolean superpose -- @property superpose --- Force the number of rows of the layout. -- @property forced_num_rows -- @tparam[opt=nil] number|nil forced_num_rows -- @propertytype nil Automatically determine the number of rows. -- @propertyunit rows -- @negativeallowed false -- @see forced_num_cols --- Force the number of columns of the layout. -- @property forced_num_cols -- @tparam[opt=nil] number|nil forced_num_cols -- @propertytype nil Automatically determine the number of columns.' -- @propertyunit columns -- @negativeallowed false -- @see forced_num_rows --- Set the minimum size for the columns. -- --@DOC_wibox_layout_grid_min_size_EXAMPLE@ -- @tparam[opt=0] number min_cols_size Minimum size of the columns. -- @property min_cols_size -- @propertyunit pixel -- @negativeallowed false -- @see min_rows_size --- Set the minimum size for the rows. -- @tparam[opt=0] number min_rows_size Minimum size of the rows. -- @property min_rows_size -- @propertyunit pixel -- @negativeallowed false -- @see min_cols_size --- The spacing between columns. -- -- @tparam[opt=0] number horizontal_spacing -- @property horizontal_spacing -- @propertyunit pixel -- @negativeallowed false -- @see spacing -- @see vertical_spacing --- The spacing between rows. -- -- @tparam[opt=0] number vertical_spacing -- @property vertical_spacing -- @propertyunit pixel -- @negativeallowed false -- @see spacing -- @see horizontal_spacing --- The spacing between rows and columns. -- -- Get the value `horizontal_spacing` or `vertical_spacing` defined by the -- preferred `orientation`. -- --@DOC_wibox_layout_grid_spacing_EXAMPLE@ -- @tparam[opt=0] number spacing -- @property spacing -- @negativeallowed false -- @see vertical_spacing -- @see horizontal_spacing --- Controls if the columns are expanded to use all the available width. -- -- @tparam[opt=false] boolean horizontal_expand Expand the grid into the available space -- @property horizontal_expand -- @see expand -- @see vertical_expand --- Controls if the rows are expanded to use all the available height. -- -- @tparam[opt=false] boolean vertical_expand Expand the grid into the available space -- @property vertical_expand -- @see expand -- @see horizontal_expand --- Controls if the columns/rows are expanded to use all the available space. -- -- Get the value `horizontal_expand` or `vertical_expand` defined by the -- preferred `orientation`. -- --@DOC_wibox_layout_grid_expand_EXAMPLE@ -- @tparam[opt=false] boolean expand Expand the grid into the available space -- @property expand -- @see horizontal_expand -- @see vertical_expand --- Controls if the columns all have the same width or if the width of each -- column depends on the content. -- -- see `homogeneous` -- -- @tparam[opt=true] boolean horizontal_homogeneous All the columns have the same width. -- @property horizontal_homogeneous -- @see vertical_homogeneous -- @see homogeneous --- Controls if the rows all have the same height or if the height of each row -- depends on the content. -- -- see `homogeneous` -- -- @tparam[opt=true] boolean vertical_homogeneous All the rows have the same height. -- @property vertical_homogeneous -- @see homogeneous -- @see horizontal_homogeneous --- Controls if the columns/rows all have the same size or if the size depends -- on the content. -- Set both `horizontal_homogeneous` and `vertical_homogeneous` to the same value. -- Get the value `horizontal_homogeneous` or `vertical_homogeneous` defined -- by the preferred `orientation`. -- --@DOC_wibox_layout_grid_expand_EXAMPLE@ -- @tparam[opt=true] boolean homogeneous All the columns/rows have the same size. -- @property homogeneous -- @see vertical_homogeneous -- @see horizontal_homogeneous --- Child widget position. Return of `get_widget_position`. -- @field row Top row index -- @field col Left column index -- @field row_span Number of rows to span -- @field col_span Number of columns to span -- @table position -- Return the maximum value of a table. local function max_value(t) local m = 0 for _,v in ipairs(t) do if m < v then m = v end end return m end -- Return the sum of the values in the table. local function sum_values(t) local m = 0 for _,v in ipairs(t) do m = m + v end return m end -- Find a widget in a widget_table, by matching the coordinates. -- Using the `row`:`col` coordinates, and the spans `row_span` and `col_span` -- @tparam table widgets_table Table of the widgets present in the grid -- @tparam number row Row number for the top left corner of the widget -- @tparam number col Column number for the top left corner of the widget -- @tparam number row_span The number of rows the widget spans (default to 1) -- @tparam number col_span The number of columns the widget spans (default to 1) -- @treturn table Table of index of widget_table local function find_widgets_at(widgets_table, row, col, row_span, col_span) if not row or row < 1 or not col or col < 1 then return nil end row_span = (row_span and row_span > 0) and row_span or 1 col_span = (col_span and col_span > 0) and col_span or 1 local ret = {} for index, data in ipairs(widgets_table) do -- If one rectangular widget is on left side of other local test_horizontal = not (row > data.row + data.row_span - 1 or data.row > row + row_span - 1) -- If one rectangular widget is above other local test_vertical = not (col > data.col + data.col_span - 1 or data.col > col + col_span - 1) if test_horizontal and test_vertical then table.insert(ret, index) end end -- reverse sort for safe removal of indices table.sort(ret, function(a,b) return a>b end) return #ret > 0 and ret or nil end -- Find a widget in a widget_table, by matching the object. -- @tparam table widgets_table Table of the widgets present in the grid -- @param widget The widget to find -- @treturn number|nil The index of the widget in widget_table, `nil` if not found local function find_widget(widgets_table, widget) for index, data in ipairs(widgets_table) do if data.widget == widget then return index end end return nil end --- Get the number of rows and columns occupied by the widgets in the grid. -- @method get_dimension -- @treturn number,number The number of rows and columns function grid:get_dimension() return self._private.num_rows, self._private.num_cols end -- Update the number of rows and columns occupied by the widgets in the grid. local function update_dimension(self) local num_rows, num_cols = 0, 0 if self._private.forced_num_rows then num_rows = self._private.forced_num_rows end if self._private.forced_num_cols then num_cols = self._private.forced_num_cols end for _, data in ipairs(self._private.widgets) do num_rows = math.max(num_rows, data.row + data.row_span - 1) num_cols = math.max(num_cols, data.col + data.col_span - 1) end self._private.num_rows = num_rows self._private.num_cols = num_cols end --- Find the next available cell to insert a widget. -- The grid is browsed according to the `orientation`. -- @method get_next_empty -- @tparam[opt=1] number hint_row The row coordinate of the last occupied cell. -- @tparam[opt=1] number hint_column The column coordinate of the last occupied cell. -- @return number,number The row,column coordinate of the next empty cell function grid:get_next_empty(hint_row, hint_column) local row = (hint_row and hint_row > 0) and hint_row or 1 local column = (hint_column and hint_column > 0) and hint_column or 1 local next_field if self._private.orientation == "vertical" then next_field = function(x, y) if y < self._private.num_cols then return x, y+1 end return x+1,1 end elseif self._private.orientation == "horizontal" then next_field = function(x, y) if x < self._private.num_rows then return x+1, y end return 1,y+1 end end while true do if find_widgets_at(self._private.widgets, row, column, 1, 1) == nil then return row, column end row, column = next_field(row, column) end end --- Add some widgets to the given grid layout. -- -- The widgets are assumed to span one cell. -- -- If the widgets have a `row_index`, `col_index`, `col_span` -- or `row_span` property, it will be honorred. -- -- @method add -- @tparam wibox.widget ... Widgets that should be added (must at least be one) -- @interface layout -- @noreturn function grid:add(...) local args = { n=select('#', ...), ... } assert(args.n > 0, "need at least one widget to add") local row, column for i=1, args.n do local w = args[i] -- Get the next empty coordinate to insert the widget row, column = self:get_next_empty(row, column) self:add_widget_at( w, w.row_index or row, w.col_index or column, w.row_span or 1, w.col_span or 1 ) end end --- Add a widget to the grid layout at specific coordinate. -- --@DOC_wibox_layout_grid_add_EXAMPLE@ -- -- @method add_widget_at -- @tparam wibox.widget child Widget that should be added -- @tparam number row Row number for the top left corner of the widget -- @tparam number col Column number for the top left corner of the widget -- @tparam[opt=1] number row_span The number of rows the widget spans. -- @tparam[opt=1] number col_span The number of columns the widget spans. -- @treturn boolean index If the operation is successful function grid:add_widget_at(child, row, col, row_span, col_span) if not row or row < 1 or not col or col < 1 then return false end row_span = (row_span and row_span > 0) and row_span or 1 col_span = (col_span and col_span > 0) and col_span or 1 -- check if the object is a widget child = base.make_widget_from_value(child) base.check_widget(child) -- test if the new widget superpose with existing ones local superpose = find_widgets_at( self._private.widgets, row, col, row_span, col_span ) if not self._private.superpose and superpose then return false end -- Add grid information attached to the widget local child_data = { widget = child, row = row, col = col, row_span = row_span, col_span = col_span } table.insert(self._private.widgets, child_data) -- Update the row and column numbers self._private.num_rows = math.max(self._private.num_rows, row + row_span - 1) self._private.num_cols = math.max(self._private.num_cols, col + col_span - 1) self:emit_signal("widget::layout_changed") self:emit_signal("widget::redraw_needed") return true end --- Remove one or more widgets from the layout. -- @method remove -- @param ... Widgets that should be removed (must at least be one) -- @treturn boolean If the operation is successful function grid:remove(...) local args = { ... } local ret = false for _, rem_widget in ipairs(args) do local index = find_widget(self._private.widgets, rem_widget) if index ~= nil then table.remove(self._private.widgets, index) ret = true end end if ret then -- Recalculate num_rows and num_cols update_dimension(self) self:emit_signal("widget::layout_changed") self:emit_signal("widget::redraw_needed") end return ret end --- Remove widgets at the coordinates. -- --@DOC_wibox_layout_grid_remove_EXAMPLE@ -- -- @method remove_widgets_at -- @tparam number row The row coordinate of the widget to remove -- @tparam number col The column coordinate of the widget to remove -- @tparam[opt=1] number row_span The number of rows the area to remove spans. -- @tparam[opt=1] number col_span The number of columns the area to remove spans. -- @treturn boolean If the operation is successful (widgets found) function grid:remove_widgets_at(row, col, row_span, col_span) local widget_indices = find_widgets_at( self._private.widgets, row, col, row_span, col_span ) if widget_indices == nil then return false end for _,index in ipairs(widget_indices) do table.remove(self._private.widgets, index) end -- Recalculate num_rows and num_cols update_dimension(self) self:emit_signal("widget::layout_changed") self:emit_signal("widget::redraw_needed") return true end --- Return the coordinates of the widget. -- @method get_widget_position -- @tparam widget widget The widget -- @treturn table The `position` table of the coordinates in the grid, with -- fields `row`, `col`, `row_span` and `col_span`. function grid:get_widget_position(widget) local index = find_widget(self._private.widgets, widget) if index == nil then return nil end local data = self._private.widgets[index] local ret = {} ret["row"] = data.row ret["col"] = data.col ret["row_span"] = data.row_span ret["col_span"] = data.col_span return ret end --- Return the widgets at the coordinates. -- @method get_widgets_at -- @tparam number row The row coordinate of the widget -- @tparam number col The column coordinate of the widget -- @tparam[opt=1] number row_span The number of rows to span. -- @tparam[opt=1] number col_span The number of columns to span. -- @treturn table The widget(s) found at the specific coordinates, nil if no widgets found function grid:get_widgets_at(row, col, row_span, col_span) local widget_indices = find_widgets_at( self._private.widgets, row, col, row_span, col_span ) if widget_indices == nil then return nil end local ret = {} for _,index in ipairs(widget_indices) do local data = self._private.widgets[index] table.insert(ret, data.widget) end return #ret > 0 and ret or nil end --- Replace old widget by new widget, spanning the same columns and rows. -- @method replace_widget -- @tparam widget old The widget to remove -- @tparam widget new The widget to add -- @treturn boolean If the operation is successful (widget found) function grid:replace_widget(old, new) -- check if the new object is a widget local status = pcall(function () base.check_widget(new) end) if not status then return false end -- find the old widget local index = find_widget(self._private.widgets, old) if index == nil then return false end -- get old widget position local data = self._private.widgets[index] local row, col, row_span, col_span = data.row, data.col, data.row_span, data.col_span table.remove(self._private.widgets, index) return self:add_widget_at(new, row, col, row_span, col_span) end -- Update position of the widgets when inserting, adding or removing a row or a column. -- @tparam table table_widgets Table of widgets -- @tparam string orientation Orientation of the line: "horizontal" -> column, "vertical" -> row -- @tparam number index Index of the line -- @tparam string mode insert, extend or remove -- @tparam boolean after Add the line after the index instead of inserting it before. -- @tparam boolean extend Extend the line at index instead of inserting an empty line. local function update_widgets_position(table_widgets, orientation, index, mode) local t = orientation == "horizontal" and "col" or "row" local to_remove = {} -- inc : Index increment or decrement -- first : Offset index for top-left cell of the widgets to shift -- last : Offset index for bottom-right cell of the widgets to resize local inc, first, last if mode == "remove" then inc, first, last = -1, 1, 1 elseif mode == "insert" then inc, first, last = 1, 0, 0 elseif mode == "extend" then inc, first, last = 1, 1, 0 else return end for i, data in ipairs(table_widgets) do -- single widget in the line if mode == "remove" and data[t] == index and data[t .. "_span"] == 1 then table.insert(to_remove, i) -- widgets to shift elseif data[t] >= index + first then data[t] = data[t] + inc -- widgets to resize elseif data[t] + data[t .. "_span"] - 1 >= index + last then data[t .. "_span"] = data[t .. "_span"] + inc end end if mode == "remove" then -- reverse sort to remove table.sort(to_remove, function(a,b) return a>b end) -- Remove widgets for _,i in ipairs(to_remove) do table.remove(table_widgets, i) end end end --- Insert column at index. -- --@DOC_wibox_layout_grid_insert_column_EXAMPLE@ -- -- @method insert_column -- @tparam number|nil index Insert the new column at index. If `nil`, the column is added at the end. -- @treturn number The index of the inserted column function grid:insert_column(index) if index == nil or index > self._private.num_cols + 1 or index < 1 then index = self._private.num_cols + 1 end -- Update widget positions update_widgets_position(self._private.widgets, "horizontal", index, "insert") -- Recalculate number of rows and columns self._private.num_cols = self._private.num_cols + 1 return index end --- Extend column at index. --@DOC_wibox_layout_grid_extend_column_EXAMPLE@ -- -- @method extend_column -- @tparam number|nil index Extend the column at index. If `nil`, the last column is extended. -- @treturn number The index of the extended column function grid:extend_column(index) if index == nil or index > self._private.num_cols or index < 1 then index = self._private.num_cols end -- Update widget positions update_widgets_position(self._private.widgets, "horizontal", index, "extend") -- Recalculate number of rows and columns self._private.num_cols = self._private.num_cols + 1 return index end --- Remove column at index. -- --@DOC_wibox_layout_grid_remove_column_EXAMPLE@ -- -- @method remove_column -- @tparam number|nil index Remove column at index. If `nil`, the last column is removed. -- @treturn number The index of the removed column function grid:remove_column(index) if index == nil or index > self._private.num_cols or index < 1 then index = self._private.num_cols end -- Update widget positions update_widgets_position(self._private.widgets, "horizontal", index, "remove") -- Recalculate number of rows and columns update_dimension(self) return index end --- Insert row at index. -- -- see `insert_column` -- -- @method insert_row -- @tparam number|nil index Insert the new row at index. If `nil`, the row is added at the end. -- @treturn number The index of the inserted row function grid:insert_row(index) if index == nil or index > self._private.num_rows + 1 or index < 1 then index = self._private.num_rows + 1 end -- Update widget positions update_widgets_position(self._private.widgets, "vertical", index, "insert") -- Recalculate number of rows and columns self._private.num_rows = self._private.num_rows + 1 return index end --- Extend row at index. -- -- see `extend_column` -- -- @method extend_row -- @tparam number|nil index Extend the row at index. If `nil`, the last row is extended. -- @treturn number The index of the extended row function grid:extend_row(index) if index == nil or index > self._private.num_rows or index < 1 then index = self._private.num_rows end -- Update widget positions update_widgets_position(self._private.widgets, "vertical", index, "extend") -- Recalculate number of rows and columns self._private.num_rows = self._private.num_rows + 1 return index end --- Remove row at index. -- -- see `remove_column` -- -- @method remove_row -- @tparam number|nil index Remove row at index. If `nil`, the last row is removed. -- @treturn number The index of the removed row function grid:remove_row(index) if index == nil or index > self._private.num_rows or index < 1 then index = self._private.num_rows end -- Update widget positions update_widgets_position(self._private.widgets, "vertical", index, "remove") -- Recalculate number of rows and columns update_dimension(self) return index end -- Return list of children function grid:get_children() local ret = {} for _, data in ipairs(self._private.widgets) do table.insert(ret, data.widget) end return ret end -- Add list of children to the layout. function grid:set_children(children) self:reset() if #children > 0 then self:add(unpack(children)) end end -- Set the preferred orientation of the grid layout. function grid:set_orientation(val) if self._private.orientation ~= val and (val == "horizontal" or val == "vertical") then self._private.orientation = val end end -- Set the minimum size for the columns. function grid:set_min_cols_size(val) if self._private.min_cols_size ~= val and val >= 0 then self._private.min_cols_size = val end end -- Set the minimum size for the rows. function grid:set_min_rows_size(val) if self._private.min_rows_size ~= val and val >= 0 then self._private.min_rows_size = val end end -- Force the number of columns of the layout. function grid:set_forced_num_cols(val) if self._private.forced_num_cols ~= val then self._private.forced_num_cols = val update_dimension(self) self:emit_signal("widget::layout_changed") end end -- Force the number of rows of the layout. function grid:set_forced_num_rows(val) if self._private.forced_num_rows ~= val then self._private.forced_num_rows = val update_dimension(self) self:emit_signal("widget::layout_changed") end end -- Set the grid properties for _, prop in ipairs(properties) do if not grid["set_" .. prop] then grid["set_"..prop] = function(self, value) if self._private[prop] ~= value then self._private[prop] = value self:emit_signal("widget::layout_changed") end end end if not grid["get_" .. prop] then grid["get_"..prop] = function(self) return self._private[prop] end end end -- Set the directional grid properties -- create a couple of properties by prepending `horizontal_` and `vertical_` -- create a common property for the two directions: -- setting the common property sets both directional properties -- getting the common property returns the directional property -- defined by the `orientation` property for _, prop in ipairs(dir_properties) do for _,dir in ipairs{"horizontal", "vertical"} do local dir_prop = dir .. "_" .. prop grid["set_"..dir_prop] = function(self, value) if self._private[dir_prop] ~= value then self._private[dir_prop] = value self:emit_signal("widget::layout_changed") end end grid["get_"..dir_prop] = function(self) return self._private[dir_prop] end end -- Non-directional options grid["set_"..prop] = function(self, value) if self._private["horizontal_"..prop] ~= value or self._private["vertical_"..prop] ~= value then self._private["horizontal_"..prop] = value self._private["vertical_"..prop] = value self:emit_signal("widget::layout_changed") end end grid["get_"..prop] = function(self) return self._private[self._private.orientation .. "_" .. prop] end end -- Return two tables of the fitted sizes of the rows and columns -- @treturn table,table Tables of row heights and column widths local function get_grid_sizes(self, context, orig_width, orig_height) local rows_size = {} local cols_size = {} -- Set the row and column sizes to the minimum value for i = 1,self._private.num_rows do rows_size[i] = self._private.min_rows_size end for j = 1,self._private.num_cols do cols_size[j] = self._private.min_cols_size end -- Calculate cell sizes for _, data in ipairs(self._private.widgets) do local w, h = base.fit_widget(self, context, data.widget, orig_width, orig_height) h = math.max( self._private.min_rows_size, h / data.row_span ) w = math.max( self._private.min_cols_size, w / data.col_span ) -- update the row and column maximum size for i = data.row, data.row + data.row_span - 1 do if h > rows_size[i] then rows_size[i] = h end end for j = data.col, data.col + data.col_span - 1 do if w > cols_size[j] then cols_size[j] = w end end end return rows_size, cols_size end -- Fit the grid 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 grid:fit(context, orig_width, orig_height) local width, height = orig_width, orig_height -- Calculate the space needed local function fit_direction(dir, sizes) local m = 0 if self._private[dir .. "_homogeneous"] then -- all the columns/rows have the same size m = #sizes * max_value(sizes) + (#sizes - 1) * self._private[dir .. "_spacing"] else -- sum the columns/rows size for _,s in ipairs(sizes) do m = m + s + self._private[dir .. "_spacing"] end m = m - self._private[dir .. "_spacing"] end return m end -- fit matrix cells local rows_size, cols_size = get_grid_sizes(self, context, width, height) -- compute the width local used_width_max = fit_direction("horizontal", cols_size) local used_height_max = fit_direction("vertical", rows_size) return used_width_max, used_height_max end -- Layout a grid layout. -- @param context The context in which we are drawn. -- @param width The available width. -- @param height The available height. function grid:layout(context, width, height) local result = {} local hspacing, vspacing = self._private.horizontal_spacing, self._private.vertical_spacing -- Fit matrix cells local rows_size, cols_size = get_grid_sizes(self, context, width, height) local total_expected_width, total_expected_height = sum_values(cols_size), sum_values(rows_size) -- Figure out the maximum size we can give out to sub-widgets local single_width, single_height = max_value(cols_size), max_value(rows_size) if self._private.horizontal_expand then single_width = (width - (self._private.num_cols-1)*hspacing) / self._private.num_cols end if self._private.vertical_expand then single_height = (height - (self._private.num_rows-1)*vspacing) / self._private.num_rows end -- Calculate the position and size to place the widgets local cumul_width, cumul_height = {}, {} local cw, ch = 0, 0 for j = 1, #cols_size do cumul_width[j] = cw if self._private.horizontal_homogeneous then cols_size[j] = math.max(self._private.min_cols_size, single_width) elseif self._private.horizontal_expand then local hpercent = self._private.num_cols * single_width * cols_size[j] / total_expected_width cols_size[j] = math.max(self._private.min_cols_size, hpercent) end cw = cw + cols_size[j] + hspacing end cumul_width[#cols_size + 1] = cw for i = 1, #rows_size do cumul_height[i] = ch if self._private.vertical_homogeneous then rows_size[i] = math.max(self._private.min_rows_size, single_height) elseif self._private.vertical_expand then local vpercent = self._private.num_rows * single_height * rows_size[i] / total_expected_height rows_size[i] = math.max(self._private.min_rows_size, vpercent) end ch = ch + rows_size[i] + vspacing end cumul_height[#rows_size + 1] = ch -- Place widgets local fill_space = true -- should be fill_space property? for _, v in pairs(self._private.widgets) do local x, y, w, h -- Round numbers to avoid decimals error, force to place tight widgets -- and avoid redraw glitches x = math.floor(cumul_width[v.col]) y = math.floor(cumul_height[v.row]) w = math.floor(cumul_width[v.col + v.col_span] - hspacing - x) h = math.floor(cumul_height[v.row + v.row_span] - vspacing - y) -- Recalculate the width so the last widget fits if (fill_space or self._private.horizontal_expand) and x + w > width then w = math.floor(math.max(self._private.min_cols_size, width - x)) end -- Recalculate the height so the last widget fits if (fill_space or self._private.vertical_expand) and y + h > height then h = math.floor(math.max(self._private.min_rows_size, height - y)) end -- Place the widget if it fits in the area if x + w <= width and y + h <= height then table.insert(result, base.place_widget_at(v.widget, x, y, w, h)) end end return result end --- Reset the grid layout. -- Remove all widgets and reset row and column counts -- -- @method reset -- @emits reset -- @noreturn function grid:reset() self._private.widgets = {} -- reset the number of columns and rows to the forced value or to 0 self._private.num_rows = self._private.forced_num_rows ~= nil and self._private.forced_num_rows or 0 self._private.num_cols = self._private.forced_num_cols ~= nil and self._private.forced_num_cols or 0 -- emit signals self:emit_signal("widget::layout_changed") self:emit_signal("widget::reset") end --- When the layout is reset. -- This signal is emitted when the layout has been reset, -- all the widgets removed and the row and column counts reset. -- @signal widget::reset --- Return a new grid layout. -- -- A grid layout sets widgets in a grids of custom number of rows and columns. -- @tparam[opt="y"] string orientation The preferred grid extension direction. -- @constructorfct wibox.layout.grid local function new(orientation) -- Preference for vertical direction: fill rows first, extend grid with new row local dir = (orientation == "horizontal"or orientation == "vertical") and orientation or "vertical" local ret = base.make_widget(nil, nil, {enable_properties = true}) gtable.crush(ret, grid, true) ret._private.orientation = dir ret._private.widgets = {} ret._private.num_rows = 0 ret._private.num_cols = 0 ret._private.rows_size = {} ret._private.cols_size = {} ret._private.superpose = false ret._private.forced_num_rows = nil ret._private.forced_num_cols = nil ret._private.min_rows_size = 0 ret._private.min_cols_size = 0 ret._private.horizontal_homogeneous = true ret._private.vertical_homogeneous = true ret._private.horizontal_expand = false ret._private.vertical_expand = false ret._private.horizontal_spacing = 0 ret._private.vertical_spacing = 0 return ret end --- Return a new horizontal grid layout. -- -- Each new widget is positioned below the last widget on the same column -- up to `forced_num_rows`. Then the next column is filled, creating it if it doesn't exist. -- @tparam number|nil forced_num_rows Forced number of rows (`nil` for automatic). -- @tparam widget ... Widgets that should be added to the layout. -- @constructorfct wibox.layout.grid.horizontal function grid.horizontal(forced_num_rows, widget, ...) local ret = new("horizontal") ret:set_forced_num_rows(forced_num_rows) if widget then ret:add(widget, ...) end return ret end --- Return a new vertical grid layout. -- -- Each new widget is positioned left of the last widget on the same row -- up to `forced_num_cols`. Then the next row is filled, creating it if it doesn't exist. -- @tparam number|nil forced_num_cols Forced number of columns (`nil` for automatic). -- @tparam widget ... Widgets that should be added to the layout. -- @constructorfct wibox.layout.grid.vertical function grid.vertical(forced_num_cols, widget, ...) local ret = new("vertical") ret:set_forced_num_cols(forced_num_cols) if widget then ret:add(widget, ...) end return ret end function grid.mt:__call(...) return new(...) end --@DOC_fixed_COMMON@ return setmetatable(grid, grid.mt) -- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80