2010-10-06 12:42:56 +02:00
|
|
|
---------------------------------------------------------------------------
|
|
|
|
-- @author Uli Schlachter
|
|
|
|
-- @copyright 2010 Uli Schlachter
|
|
|
|
-- @release @AWESOME_VERSION@
|
|
|
|
---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
local setmetatable = setmetatable
|
|
|
|
local table = table
|
|
|
|
local pairs = pairs
|
|
|
|
local type = type
|
|
|
|
local base = require("wibox.layout.base")
|
|
|
|
local widget_base = require("wibox.widget.base")
|
|
|
|
|
2012-06-12 15:29:52 +02:00
|
|
|
-- wibox.layout.align
|
|
|
|
local align = {}
|
2010-10-06 12:42:56 +02:00
|
|
|
|
2013-01-05 16:12:47 +01:00
|
|
|
--- Draw an align layout.
|
|
|
|
-- @param wibox The wibox that this widget is drawn to.
|
|
|
|
-- @param cr The cairo context to use.
|
|
|
|
-- @param width The available width.
|
|
|
|
-- @param height The available height.
|
|
|
|
-- @return The total space needed by the layout.
|
|
|
|
function align:draw(wibox, cr, width, height)
|
2010-10-06 12:42:56 +02:00
|
|
|
local size_first = 0
|
|
|
|
local size_third = 0
|
2013-01-05 16:12:47 +01:00
|
|
|
local size_limit = self.dir == "y" and height or width
|
2010-10-06 12:42:56 +02:00
|
|
|
|
2013-01-05 16:12:47 +01:00
|
|
|
if self.first then
|
2012-06-16 21:12:26 +02:00
|
|
|
local w, h, _ = width, height, nil
|
2013-01-05 16:12:47 +01:00
|
|
|
if self.dir == "y" then
|
|
|
|
_, h = self.first:fit(w, h)
|
2010-10-06 12:42:56 +02:00
|
|
|
size_first = h
|
|
|
|
else
|
2013-01-05 16:12:47 +01:00
|
|
|
w, _ = self.first:fit(w, h)
|
2010-10-06 12:42:56 +02:00
|
|
|
size_first = w
|
|
|
|
end
|
2013-01-05 16:12:47 +01:00
|
|
|
base.draw_widget(wibox, cr, self.first, 0, 0, w, h)
|
2010-10-06 12:42:56 +02:00
|
|
|
end
|
|
|
|
|
2013-01-05 16:12:47 +01:00
|
|
|
if self.third and size_first < size_limit then
|
2012-06-16 21:12:26 +02:00
|
|
|
local w, h, x, y, _
|
2013-01-05 16:12:47 +01:00
|
|
|
if self.dir == "y" then
|
2010-10-06 12:42:56 +02:00
|
|
|
w, h = width, height - size_first
|
2013-01-05 16:12:47 +01:00
|
|
|
_, h = self.third:fit(w, h)
|
2010-10-06 12:42:56 +02:00
|
|
|
x, y = 0, height - h
|
|
|
|
size_third = h
|
|
|
|
else
|
|
|
|
w, h = width - size_first, height
|
2013-01-05 16:12:47 +01:00
|
|
|
w, _ = self.third:fit(w, h)
|
2010-10-06 12:42:56 +02:00
|
|
|
x, y = width - w, 0
|
|
|
|
size_third = w
|
|
|
|
end
|
2013-01-05 16:12:47 +01:00
|
|
|
base.draw_widget(wibox, cr, self.third, x, y, w, h)
|
2010-10-06 12:42:56 +02:00
|
|
|
end
|
|
|
|
|
2013-01-05 16:12:47 +01:00
|
|
|
if self.second and size_first + size_third < size_limit then
|
2010-10-06 12:42:56 +02:00
|
|
|
local x, y, w, h
|
2013-01-05 16:12:47 +01:00
|
|
|
if self.dir == "y" then
|
2010-10-06 12:42:56 +02:00
|
|
|
x, y = 0, size_first
|
|
|
|
w, h = width, size_limit - size_first - size_third
|
|
|
|
else
|
|
|
|
x, y = size_first, 0
|
|
|
|
w, h = size_limit - size_first - size_third, height
|
|
|
|
end
|
2013-01-05 16:12:47 +01:00
|
|
|
base.draw_widget(wibox, cr, self.second, x, y, w, h)
|
2010-10-06 12:42:56 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function widget_changed(layout, old_w, new_w)
|
|
|
|
if old_w then
|
|
|
|
old_w:disconnect_signal("widget::updated", layout._emit_updated)
|
|
|
|
end
|
|
|
|
if new_w then
|
|
|
|
widget_base.check_widget(new_w)
|
|
|
|
new_w:connect_signal("widget::updated", layout._emit_updated)
|
|
|
|
end
|
|
|
|
layout._emit_updated()
|
|
|
|
end
|
|
|
|
|
|
|
|
--- Set the layout's first widget. This is the widget that is at the left/top
|
2012-11-18 20:44:03 +01:00
|
|
|
function align:set_first(widget)
|
|
|
|
widget_changed(self, self.first, widget)
|
|
|
|
self.first = widget
|
2010-10-06 12:42:56 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
--- Set the layout's second widget. This is the centered one.
|
2012-11-18 20:44:03 +01:00
|
|
|
function align:set_second(widget)
|
|
|
|
widget_changed(self, self.second, widget)
|
|
|
|
self.second = widget
|
2010-10-06 12:42:56 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
--- Set the layout's third widget. This is the widget that is at the right/bottom
|
2012-11-18 20:44:03 +01:00
|
|
|
function align:set_third(widget)
|
|
|
|
widget_changed(self, self.third, widget)
|
|
|
|
self.third = widget
|
2010-10-06 12:42:56 +02:00
|
|
|
end
|
|
|
|
|
2013-01-05 16:12:49 +01:00
|
|
|
--- Fit the align layout into the given space. The align layout will
|
|
|
|
-- take all available space in its direction and the maximum size of
|
|
|
|
-- it's all three inner widgets in the other axis.
|
|
|
|
-- @param orig_width The available width.
|
|
|
|
-- @param orig_height The available height.
|
|
|
|
function align:fit(orig_width, orig_height)
|
|
|
|
local used_max = 0
|
|
|
|
|
|
|
|
for k, v in pairs{self.first, self.second, self.third} do
|
|
|
|
local w, h = v:fit(orig_width, orig_height)
|
|
|
|
|
|
|
|
local max = self.dir == "y" and w or h
|
|
|
|
|
|
|
|
if max > used_max then
|
|
|
|
used_max = max
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
if self.dir == "y" then
|
|
|
|
return used_max, orig_height
|
|
|
|
end
|
|
|
|
return orig_width, used_max
|
|
|
|
end
|
|
|
|
|
2012-11-18 20:44:03 +01:00
|
|
|
function align:reset()
|
2010-10-06 12:42:56 +02:00
|
|
|
for k, v in pairs({ "first", "second", "third" }) do
|
2012-11-18 20:44:03 +01:00
|
|
|
self[v] = nil
|
2010-10-06 12:42:56 +02:00
|
|
|
end
|
2012-11-18 20:44:03 +01:00
|
|
|
self:emit_signal("widget::updated")
|
2010-10-06 12:42:56 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
local function get_layout(dir)
|
|
|
|
local ret = widget_base.make_widget()
|
2013-01-05 16:12:47 +01:00
|
|
|
ret.dir = dir
|
2010-10-06 12:42:56 +02:00
|
|
|
ret._emit_updated = function()
|
|
|
|
ret:emit_signal("widget::updated")
|
|
|
|
end
|
|
|
|
|
2012-06-12 15:29:52 +02:00
|
|
|
for k, v in pairs(align) do
|
2010-10-06 12:42:56 +02:00
|
|
|
if type(v) == "function" then
|
|
|
|
ret[k] = v
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
return ret
|
|
|
|
end
|
|
|
|
|
|
|
|
--- Returns a new horizontal align layout. An align layout can display up to
|
|
|
|
-- three widgets. The widget set via :set_left() is left-aligned. :set_right()
|
|
|
|
-- sets a widget which will be right-aligned. The remaining space between those
|
|
|
|
-- two will be given to the widget set via :set_middle().
|
2012-06-12 15:29:52 +02:00
|
|
|
function align.horizontal()
|
2010-10-06 12:42:56 +02:00
|
|
|
local ret = get_layout("x")
|
|
|
|
|
|
|
|
ret.set_left = ret.set_first
|
|
|
|
ret.set_middle = ret.set_second
|
|
|
|
ret.set_right = ret.set_third
|
|
|
|
|
|
|
|
return ret
|
|
|
|
end
|
|
|
|
|
|
|
|
--- Returns a new vertical align layout. An align layout can display up to
|
|
|
|
-- three widgets. The widget set via :set_top() is top-aligned. :set_bottom()
|
|
|
|
-- sets a widget which will be bottom-aligned. The remaining space between those
|
|
|
|
-- two will be given to the widget set via :set_middle().
|
2012-06-12 15:29:52 +02:00
|
|
|
function align.vertical()
|
2010-10-06 12:42:56 +02:00
|
|
|
local ret = get_layout("y")
|
|
|
|
|
|
|
|
ret.set_top = ret.set_first
|
|
|
|
ret.set_middle = ret.set_second
|
|
|
|
ret.set_bottom = ret.set_third
|
|
|
|
|
|
|
|
return ret
|
|
|
|
end
|
|
|
|
|
2012-06-12 15:29:52 +02:00
|
|
|
return align
|
|
|
|
|
2011-09-11 16:50:01 +02:00
|
|
|
-- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80
|