local setmetatable = setmetatable local print,pairs = print,pairs local unpack=unpack local util = require( "awful.util" ) local button = require( "awful.button" ) local checkbox = require( "radical.widgets.checkbox" ) local wibox = require( "wibox" ) local item_layout = require("radical.item.layout.icon") local base = nil local module = {} local function left(data) if data._current_item._tmp_menu then data = data._current_item._tmp_menu data.items.selected = true return true,data end end local function right(data) if data.parent_geometry.is_menu then for k,v in ipairs(data.items) do if v._tmp_menu == data or v.sub_menu_m == data then v.selected = true end end data.visible = false data = data.parent_geometry return true,data end end local function up(data) data.previous_item.selected = true end local function down(data) data.next_item.selected = true end function module:setup_key_hooks(data) data:add_key_hook({}, "Up" , "press", up ) data:add_key_hook({}, "&" , "press", up ) data:add_key_hook({}, "Down" , "press", down ) data:add_key_hook({}, "KP_Enter", "press", down ) data:add_key_hook({}, "Left" , "press", left ) data:add_key_hook({}, "\"" , "press", left ) data:add_key_hook({}, "Right" , "press", right ) data:add_key_hook({}, "#" , "press", right ) end local function setup_event(data,item,args) --Event handling if data.select_on == base.event.HOVER then item.widget:connect_signal("mouse::enter", function() item.selected = true end) item.widget:connect_signal("mouse::leave", function() item.selected = false end) end data._internal.layout:add(item) local buttons = {} for i=1,10 do if args["button"..i] then buttons[#buttons+1] = button({},i,args["button"..i]) end end if not buttons[3] then --Hide on right click buttons[#buttons+1] = button({},3,function() data.visible = false if data.parent_geometry and data.parent_geometry.is_menu then data.parent_geometry.visible = false end end) end --Be sure to always hide sub menus, even when data.visible is set manually data:connect_signal("visible::changed",function(_,vis) if data._tmp_menu and data.visible == false then data._tmp_menu.visible = false end end) data:connect_signal("parent_geometry::changed",function(_,vis) local fit_w,fit_h = data._internal.layout:fit() data.height = fit_h if data.style then data.style(data) end end) item.widget:buttons( util.table.join(unpack(buttons))) end function module:setup_item(data,item,args) local bg = item_layout(item,data,args) -- Set size local fit_w,fit_h = data._internal.layout:fit() data.width = fit_w data.height = fit_h if data.style then data.style(data) end local text_w = item._internal.text_w local icon_w = item._internal.icon_w -- Setup text item.set_text = function (_,value) if data.disable_markup then text_w:set_text(value) else text_w:set_markup(value) end if data.auto_resize then local fit_w,fit_h = text_w:get_preferred_size() local is_largest = item == data._internal.largest_item_h --TODO find new largest is item is smaller if not data._internal.largest_item_h_v or data._internal.largest_item_h_v < fit_h then data._internal.largest_item_h =item data._internal.largest_item_h_v = fit_h end end end item.set_icon = function (_,value) icon_w:set_image(value) end item:set_text(item._private_data.text) -- Setup tooltip bg:set_tooltip(item.tooltip) -- Set widget item.widget = bg data.item_style(item,{}) setup_event(data,item,args) end --Get preferred item geometry local function item_fit(data,item,self, content, width, height) if not data.visible then return 1,1 end local w, h = item._private_data._fit(self,content,width,height) --TODO port to new context API return data.item_width or 70, item._private_data.height or h end local function new(data) if not base then base = require( "radical.base" ) end -- Define the item layout local real_l = wibox.widget.base.make_widget_declarative { spacing = data.spacing , item_fit = item_fit , setup_key_hooks = module.setup_key_hooks , setup_item = module.setup_item , layout = wibox.layout.fixed.horizontal, } data:connect_signal("widget::added",function(_,item,widget) wibox.layout.fixed.add(real_l, item.widget) real_l:emit_signal("widget::updated") end) function real_l:add(item) return wibox.layout.fixed.add(self, item.widget) end -- Hack fit local new_fit new_fit = function(self,context,w,h,force_values) --TODO use the context instead of extra argument -- Get the original fit, the function need to be replaced to avoir a stack overflow real_l.fit = real_l._fit local result,r2 = self:get_preferred_size(context, force_values and w, force_values and h) real_l.fit = new_fit local w,h if data.auto_resize and data._internal.largest_item_h then w,h = data.rowcount*(data.item_width or data.default_width),data._internal.largest_item_h_v > data.item_height and data._internal.largest_item_h_v or data.item_height else w,h = data.rowcount*(data.item_width or data.default_width),data.item_height end data:emit_signal("layout_size",w,h) return w,h end real_l._fit = real_l.fit real_l.fit = new_fit return real_l end return setmetatable(module, { __call = function(_, ...) return new(...) end }) -- kate: space-indent on; indent-width 4; replace-tabs on;