From 0376cfaf319075daa9378e487de1268697688221 Mon Sep 17 00:00:00 2001 From: Emmanuel Lepage Vallee Date: Sun, 2 Feb 2014 16:53:09 -0500 Subject: [PATCH] Add new 'item_layout' concept to fight code duplication between layouts --- bar.lua | 92 +-------------------- item_layout/horizontal.lua | 163 +++++++++++++++++++++++++++++++++++++ layout/vertical.lua | 67 ++------------- style/arrow.lua | 2 +- 4 files changed, 171 insertions(+), 153 deletions(-) create mode 100644 item_layout/horizontal.lua diff --git a/bar.lua b/bar.lua index e6e5c3d..1c02d1e 100644 --- a/bar.lua +++ b/bar.lua @@ -11,6 +11,7 @@ local button = require( "awful.button" ) local checkbox = require( "radical.widgets.checkbox" ) local item_style = require( "radical.item_style.arrow_single" ) local vertical = require( "radical.layout.vertical" ) +local item_layout= require( "radical.item_layout.horizontal" ) local capi,module = { mouse = mouse , screen = screen, keygrabber = keygrabber },{} @@ -58,95 +59,6 @@ local function setup_drawable(data) data.draw = internal.margin.draw end --- Use all the space, let "align_fit" compute the right size -local function textbox_fit(box,w,h) - return w,h -end - --- Force the width or compute the minimum space -local function align_fit(box,w,h) - if box._item.width then return box._item.width - box._data.item_style.margins.LEFT - box._data.item_style.margins.RIGHT,h end - return box.first:fit(w,h)+wibox.widget.textbox.fit(box.second,w,h)+box.third:fit(w,h),h -end - --- Create the actual widget -local function create_item(item,data,args) - -- Background - local bg = wibox.widget.background() - - -- Margins - local m = wibox.layout.margin(la) - m:set_margins (0) - m:set_left ( data.item_style.margins.LEFT ) - m:set_right ( data.item_style.margins.RIGHT ) - m:set_top ( data.item_style.margins.TOP ) - m:set_bottom( data.item_style.margins.BOTTOM ) - - -- Layout (left) - local layout = wibox.layout.fixed.horizontal() - bg:set_widget(m) - - -- Layout (right) - local right = wibox.layout.fixed.horizontal() - - -- F keys - vertical:setup_fkey(item,data) - if data.fkeys_prefix == true then - layout:add(fkey(data,item)) - end - - -- Icon - layout:add(vertical:setup_icon(item,data)) - - -- Prefix - if args.prefix_widget then - layout:add(args.prefix_widget) - end - - -- Text - local tb = wibox.widget.textbox() - tb.fit = textbox_fit - tb.draw = function(self,w, cr, width, height) - if item.underlay then - vertical.paint_underlay(data,item,cr,width,height) - end - wibox.widget.textbox.draw(self,w, cr, width, height) - end - item.widget = bg - tb:set_text(item.text) - - -- Checkbox - local ck = vertical:setup_checked(item,data) - if ck then - right:add(ck) - end - - -- Suffix - if args.suffix_widget then - right:add(args.suffix_widget) - end - - -- Layout (align) - local align = wibox.layout.align.horizontal() - align:set_middle( tb ) - align:set_left ( layout ) - align:set_right ( right ) - m:set_widget ( align ) - align._item = item - align._data = data - align.fit = align_fit - item._internal.align = align - - -- Tooltip - item.widget:set_tooltip(item.tooltip) - - -- Draw - data.item_style(data,item,{}) - item.widget:set_fg(item._private_data.fg) - - return bg -end - local function setup_buttons(data,item,args) local buttons = {} for i=1,10 do @@ -178,7 +90,7 @@ end local function setup_item(data,item,args) -- Add widgets - data._internal.layout:add(create_item(item,data,args)) + data._internal.layout:add(item_layout(item,data,args)) item.widget:connect_signal("mouse::enter", function() item.selected = true end) item.widget:connect_signal("mouse::leave", function() item.selected = false end) diff --git a/item_layout/horizontal.lua b/item_layout/horizontal.lua new file mode 100644 index 0000000..47a1a07 --- /dev/null +++ b/item_layout/horizontal.lua @@ -0,0 +1,163 @@ +local setmetatable = setmetatable +local beautiful = require( "beautiful" ) +local color = require( "gears.color" ) +local cairo = require( "lgi" ).cairo +local wibox = require( "wibox" ) +local checkbox = require( "radical.widgets.checkbox" ) +local fkey = require( "radical.widgets.fkey" ) + +local module = {} + + +-- Add [F1], [F2] ... to items +function module:setup_fkey(item,data) + item._internal.set_map.f_key = function(value) + item._internal.has_changed = true + item._internal.f_key = value + data:remove_key_hook("F"..value) + data:add_key_hook({}, "F"..value , "press", function() + item.button1() + data.visible = false + end) + end + item._internal.get_map.f_key = function() return item._internal.f_key end +end + +-- Like an overlay, but under +function module.paint_underlay(data,item,cr,width,height) + cr:save() + local udl = underlay.draw(item.underlay) + cr:set_source_surface(udl,width-udl:get_width()-3) + cr:paint_with_alpha(data.underlay_alpha) + cr:restore() +end + +-- Setup the item icon +function module:setup_icon(item,data) + local icon = wibox.widget.imagebox() + icon.fit = function(...) + local w,h = wibox.widget.imagebox.fit(...) + return w+3,h + end + if item.icon then + icon:set_image(item.icon) + end + + item._internal.set_map.icon = function (value) + icon:set_image(value) + end + return icon +end + +-- Show the checkbox +function module:setup_checked(item,data) + if item.checkable then + item._internal.get_map.checked = function() + if type(item._private_data.checked) == "function" then + return item._private_data.checked() + else + return item._private_data.checked + end + end + local ck = wibox.widget.imagebox() + ck:set_image(item.checked and checkbox.checked() or checkbox.unchecked()) + item._internal.set_map.checked = function (value) + item._private_data.checked = value + ck:set_image(item.checked and checkbox.checked() or checkbox.unchecked()) + item._internal.has_changed = true + end + return ck + end +end + +-- Use all the space, let "align_fit" compute the right size +local function textbox_fit(box,w,h) + return w,h +end + +-- Force the width or compute the minimum space +local function align_fit(box,w,h) + if box._item.width then return box._item.width - box._data.item_style.margins.LEFT - box._data.item_style.margins.RIGHT,h end + return box.first:fit(w,h)+wibox.widget.textbox.fit(box.second,w,h)+box.third:fit(w,h),h +end + +-- Create the actual widget +local function create_item(item,data,args) + -- Background + local bg = wibox.widget.background() + + -- Margins + local m = wibox.layout.margin(la) + m:set_margins (0) + m:set_left ( data.item_style.margins.LEFT ) + m:set_right ( data.item_style.margins.RIGHT ) + m:set_top ( data.item_style.margins.TOP ) + m:set_bottom( data.item_style.margins.BOTTOM ) + + -- Layout (left) + local layout = wibox.layout.fixed.horizontal() + bg:set_widget(m) + + -- Layout (right) + local right = wibox.layout.fixed.horizontal() + + -- F keys + module:setup_fkey(item,data) + if data.fkeys_prefix == true then + layout:add(fkey(data,item)) + end + + -- Icon + layout:add(module:setup_icon(item,data)) + + -- Prefix + if args.prefix_widget then + layout:add(args.prefix_widget) + end + + -- Text + local tb = wibox.widget.textbox() + tb.fit = textbox_fit + tb.draw = function(self,w, cr, width, height) + if item.underlay then + module.paint_underlay(data,item,cr,width,height) + end + wibox.widget.textbox.draw(self,w, cr, width, height) + end + item.widget = bg + tb:set_text(item.text) + + -- Checkbox + local ck = module:setup_checked(item,data) + if ck then + right:add(ck) + end + + -- Suffix + if args.suffix_widget then + right:add(args.suffix_widget) + end + + -- Layout (align) + local align = wibox.layout.align.horizontal() + align:set_middle( tb ) + align:set_left ( layout ) + align:set_right ( right ) + m:set_widget ( align ) + align._item = item + align._data = data + align.fit = align_fit + item._internal.align = align + + -- Tooltip + item.widget:set_tooltip(item.tooltip) + + -- Draw + data.item_style(data,item,{}) + item.widget:set_fg(item._private_data.fg) + + return bg +end + +return setmetatable(module, { __call = function(_, ...) return create_item(...) end }) +-- kate: space-indent on; indent-width 2; replace-tabs on; diff --git a/layout/vertical.lua b/layout/vertical.lua index 82dd593..93ea81e 100644 --- a/layout/vertical.lua +++ b/layout/vertical.lua @@ -11,6 +11,7 @@ local beautiful = require("beautiful" ) local wibox = require( "wibox" ) local color = require( "gears.color" ) local cairo = require( "lgi" ).cairo +local item_layout= require( "radical.item_layout.horizontal" ) local module = {} @@ -65,15 +66,6 @@ local function item_fit(data,item,...) return w, item._private_data.height or h end --- Like an overlay, but under -function module.paint_underlay(data,item,cr,width,height) - cr:save() - local udl = underlay.draw(item.underlay) - cr:set_source_surface(udl,width-udl:get_width()-3) - cr:paint_with_alpha(data.underlay_alpha) - cr:restore() -end - -- As of July 2013, LGI is too slow to redraw big menus at ok speed -- This do a pixmap cache to allow pre-rendering local function cache_pixmap(item) @@ -96,61 +88,12 @@ local function cache_pixmap(item) end end -function module:setup_fkey(item,data) - item._internal.set_map.f_key = function(value) - item._internal.has_changed = true - item._internal.f_key = value - data:remove_key_hook("F"..value) - data:add_key_hook({}, "F"..value , "press", function() - item.button1() - data.visible = false - end) - end - item._internal.get_map.f_key = function() return item._internal.f_key end -end - -function module:setup_checked(item,data) - if item.checkable then - item._internal.get_map.checked = function() - if type(item._private_data.checked) == "function" then - return item._private_data.checked() - else - return item._private_data.checked - end - end - local ck = wibox.widget.imagebox() - ck:set_image(item.checked and checkbox.checked() or checkbox.unchecked()) - item._internal.set_map.checked = function (value) - item._private_data.checked = value - ck:set_image(item.checked and checkbox.checked() or checkbox.unchecked()) - item._internal.has_changed = true - end - return ck - end -end - -function module:setup_icon(item,data) - local icon = wibox.widget.imagebox() - icon.fit = function(...) - local w,h = wibox.widget.imagebox.fit(...) - return w+3,h - end - if item.icon then - icon:set_image(item.icon) - end - - item._internal.set_map.icon = function (value) - icon:set_image(value) - end - return icon -end - function module:setup_text(item,data) local text_w = wibox.widget.textbox() text_w.draw = function(self,w, cr, width, height) if item.underlay then - module.paint_underlay(data,item,cr,width,height) + item_layout.paint_underlay(data,item,cr,width,height) end wibox.widget.textbox.draw(self,w, cr, width, height) end @@ -241,11 +184,11 @@ function module:setup_item(data,item,args) end -- Icon - local icon = module:setup_icon(item,data) + local icon = item_layout:setup_icon(item,data) l:add(icon) -- Checkbox - local ck = module:setup_checked(item,data) + local ck = item_layout:setup_checked(item,data) if ck then lr:add(ck) end @@ -279,7 +222,7 @@ function module:setup_item(data,item,args) end -- F keys - module:setup_fkey(item,data) + item_layout:setup_fkey(item,data) -- Enable scrollbar if necessary if data._internal.scroll_w and data.rowcount > data.max_items then diff --git a/style/arrow.lua b/style/arrow.lua index 2efbd8d..2d3d7ce 100644 --- a/style/arrow.lua +++ b/style/arrow.lua @@ -94,7 +94,7 @@ local function draw(data,args) --END set_arrow set_direction(data,direction) - data._internal.set_position(data) +-- data._internal.set_position(data) --TODO DEAD CODE? local margins = data.margins local margin = data._internal.margin