awesome/lib/wibox/layout/grid.lua

998 lines
34 KiB
Lua

---------------------------------------------------------------------------
--- 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@
--
-- Using the declarative system, widgets are automatically added next to each
-- other spanning only one cell.
--
--@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.
--
-- @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
-- Get the next empty coordinate to insert the widget
row, column = self:get_next_empty(row, column)
self:add_widget_at(args[i], row, column, 1, 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