352 lines
11 KiB
Lua
352 lines
11 KiB
Lua
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 underlay = require( "radical.widgets.underlay" )
|
|
local theme = require( "radical.theme" )
|
|
local util = require( "awful.util" )
|
|
local margins2 = require( "radical.margins")
|
|
local shape = require( "gears.shape" )
|
|
local surface = require( "gears.surface" )
|
|
|
|
local module = {}
|
|
|
|
|
|
-- Add [F1], [F2] ... to items
|
|
function module:setup_fkey(item,data)
|
|
item.set_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,menu)
|
|
data.visible = false
|
|
end)
|
|
end
|
|
item.get_f_key = function() return item._internal.f_key end
|
|
end
|
|
|
|
-- Like an overlay, but under
|
|
function module.paint_underlay(data,item,cr,width,height, name)
|
|
name = name or "underlay"
|
|
cr:save()
|
|
local state = item.state or {}
|
|
local current_state = state._current_key or nil
|
|
local state_name = theme.colors_by_id[current_state] or ""
|
|
|
|
local bg_color = item[name.."_bg_"..state_name] or data[name.."_bg_"..state_name] or data[name.."_bg"]
|
|
local style = item[name.."_style" ] or data[name.."_style" ]
|
|
local alpha = item[name.."_alpha" ] or data[name.."_alpha" ]
|
|
local align = item[name.."_align" ] or data[name.."_align" ]
|
|
|
|
local udl = underlay.draw(item[name],{style=data.underlay_style,height=height,bg=bg_color, style = style})
|
|
|
|
if align == "center" then
|
|
local offset = (width-udl:get_width()-6)/2
|
|
cr:set_source_surface(udl,3 + offset)
|
|
else
|
|
cr:set_source_surface(udl,width-udl:get_width()-3)
|
|
end
|
|
cr:paint_with_alpha(data[name.."_alpha"])
|
|
cr:restore()
|
|
end
|
|
|
|
function module.after_draw_children(self, context, cr, width, height)
|
|
if self._item.overlay then
|
|
module.paint_underlay(self._data, self._item, cr, width, height, "overlay")
|
|
end
|
|
|
|
if self._item.overlay_draw then
|
|
self._item.overlay_draw(context,self._item,cr,width,height)
|
|
end
|
|
|
|
-- Draw the border, if any
|
|
if self._after_draw_children then
|
|
self._after_draw_children(self, context, cr, width, height)
|
|
end
|
|
end
|
|
|
|
-- Apply icon transformation
|
|
function module.set_icon(self,image)
|
|
if self._data.icon_transformation then
|
|
self._item._original_icon = image
|
|
image = self._data.icon_transformation(image,self._data,self._item)
|
|
end
|
|
wibox.widget.imagebox.set_image(self,image)
|
|
end
|
|
|
|
-- Setup the item icon
|
|
function module:setup_icon(item,data)
|
|
local icon = wibox.widget.imagebox()
|
|
icon._data = data
|
|
icon._item = item
|
|
icon.set_image = module.set_icon
|
|
if item.icon then
|
|
icon:set_image(item.icon)
|
|
end
|
|
|
|
item.set_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.get_checked = function()
|
|
if type(item._private_data.checked) == "function" then
|
|
return item._private_data.checked(data,item)
|
|
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.set_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
|
|
|
|
-- Setup hover
|
|
function module:setup_hover(item,data)
|
|
item.set_hover = function(_,value)
|
|
local item_style = item.item_style or data.item_style
|
|
item.state[-1] = value and true or nil
|
|
item_style(item,{})
|
|
end
|
|
end
|
|
|
|
-- Create sub_menu arrows
|
|
local sub_arrow = nil
|
|
function module:setup_sub_menu_arrow(item,data)
|
|
if (item._private_data.sub_menu_f or item._private_data.sub_menu_m) and not data.disable_submenu_icon then
|
|
if not sub_arrow then
|
|
sub_arrow = wibox.widget.imagebox() --TODO, make global
|
|
sub_arrow.fit = function(box, context,w, h) return (sub_arrow._image and sub_arrow._image:get_width() or 0),item.height end
|
|
|
|
if beautiful.menu_submenu_icon then
|
|
sub_arrow:set_image( beautiful.menu_submenu_icon )
|
|
else
|
|
local h = data.item_height
|
|
sub_arrow:set_image(surface.load_from_shape(7, h,
|
|
shape.transform(shape.isosceles_triangle) : rotate_at(3.5,h/2,math.pi/2),
|
|
beautiful.menu_fg_normal or beautiful.menu_fg or beautiful.fg_normal
|
|
))
|
|
end
|
|
end
|
|
return sub_arrow
|
|
end
|
|
end
|
|
|
|
-- Proxy all events to the parent
|
|
function module.setup_event(data,item,widget)
|
|
local widget = widget or item.widget
|
|
|
|
-- Setup data signals
|
|
widget:connect_signal("button::press",function(_,__,___,id,mod,geo)
|
|
local mods_invert = {}
|
|
for k,v in ipairs(mod) do
|
|
mods_invert[v] = i
|
|
end
|
|
|
|
item.state[4] = true
|
|
data:emit_signal("button::press",item,id,mods_invert,geo)
|
|
item:emit_signal("button::press",data,id,mods_invert,geo)
|
|
end)
|
|
widget:connect_signal("button::release",function(wdg,__,___,id,mod,geo)
|
|
local mods_invert = {}
|
|
for k,v in ipairs(mod) do
|
|
mods_invert[v] = i
|
|
end
|
|
item.state[4] = nil
|
|
data:emit_signal("button::release",item,id,mods_invert,geo)
|
|
item:emit_signal("button::release",data,id,mods_invert,geo)
|
|
end)
|
|
widget:connect_signal("mouse::enter",function(b,mod,geo)
|
|
data:emit_signal("mouse::enter",item,mod,geo)
|
|
item:emit_signal("mouse::enter",data,mod,geo)
|
|
end)
|
|
widget:connect_signal("mouse::leave",function(b,mod,geo)
|
|
data:emit_signal("mouse::leave",item,mod,geo)
|
|
item:emit_signal("mouse::leave",data,mod,geo)
|
|
end)
|
|
|
|
-- Always tracking mouse::move is expensive, only do it when necessary
|
|
-- local function conn(b,t)
|
|
-- item:emit_signal("mouse::move",item)
|
|
-- end
|
|
-- item:connect_signal("connection",function(_,name,count)
|
|
-- if name == "mouse::move" then
|
|
-- widget:connect_signal("mouse::move",conn)
|
|
-- end
|
|
-- end)
|
|
-- item:connect_signal("disconnection",function(_,name,count)
|
|
-- if count == 0 then
|
|
-- widget:connect_signal("mouse::move",conn)
|
|
-- end
|
|
-- end)
|
|
end
|
|
|
|
-- Use all the space, let "align_fit" compute the right size
|
|
local function textbox_fit(box,context,w,h)
|
|
return w,h
|
|
end
|
|
|
|
-- Force the width or compute the minimum space
|
|
local function align_fit(box,context,w,h)
|
|
local mar = util.table.join(box._data.item_style.margins,box._data.default_item_margins)
|
|
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(context,w,h)+wibox.widget.textbox.fit(box.second,context,w,h)+box.third:fit(context,w,h),h
|
|
end
|
|
|
|
-- Create the actual widget
|
|
local function create_item(item,data,args)
|
|
|
|
-- F keys
|
|
module:setup_fkey(item,data)
|
|
|
|
-- Icon
|
|
local icon = module:setup_icon(item,data)
|
|
icon.fit = function(...)
|
|
local w,h = wibox.widget.imagebox.fit(...)
|
|
return w+3,h
|
|
end
|
|
|
|
if data.icon_per_state == true then
|
|
item:connect_signal("state::changed",function(i,d,st)
|
|
if item._original_icon and data.icon_transformation then
|
|
wibox.widget.imagebox.set_image(icon,data.icon_transformation(item._original_icon,data,item))
|
|
end
|
|
end)
|
|
end
|
|
|
|
-- Text
|
|
local tb = wibox.widget.textbox()
|
|
tb.fit = data._internal.text_fit or textbox_fit
|
|
tb.draw = function(self, context, cr, width, height)
|
|
if item.underlay then
|
|
module.paint_underlay(data,item,cr,width,height)
|
|
end
|
|
wibox.widget.textbox.draw(self, context, cr, width, height)
|
|
end
|
|
tb:set_text(item.text)
|
|
item.set_text = function (_,value)
|
|
if data.disable_markup then
|
|
tb:set_text(value)
|
|
else
|
|
tb:set_markup(value)
|
|
end
|
|
item._private_data.text = value
|
|
end
|
|
|
|
-- Hover
|
|
module:setup_hover(item,data)
|
|
|
|
-- Overlay
|
|
item.set_overlay = function(_,value)
|
|
item._private_data.overlay = value
|
|
item.widget:emit_signal("widget::updated")
|
|
end
|
|
|
|
-- Define the item layout
|
|
item.widget = wibox.widget.base.make_widget_declarative {
|
|
-- Widgets
|
|
{
|
|
-- Widget
|
|
{
|
|
-- This is where the content is placed
|
|
|
|
-- Widgets
|
|
{
|
|
-- The prefixes
|
|
|
|
-- Widget
|
|
data.fkeys_prefix and fkey(data,item) or nil,
|
|
icon ,
|
|
args.prefix_widget ,
|
|
|
|
-- Attributes
|
|
layout = wibox.layout.fixed.horizontal
|
|
},
|
|
tb,
|
|
{
|
|
-- Suffixes
|
|
|
|
-- Widget
|
|
module:setup_checked(item,data) ,
|
|
module:setup_sub_menu_arrow(item,data),
|
|
args.suffix_widget ,
|
|
|
|
-- Attributes
|
|
layout = wibox.layout.fixed.horizontal
|
|
},
|
|
|
|
-- Attributes
|
|
_item = item ,
|
|
_data = data ,
|
|
id = "main_align" ,
|
|
layout = wibox.layout.align.horizontal,
|
|
},
|
|
|
|
-- Attributes
|
|
id = "main_margin" ,
|
|
layout = wibox.layout.margin,
|
|
},
|
|
|
|
-- Attributes
|
|
fg = item._private_data.fg ,
|
|
tooltip = item.tooltip ,
|
|
_item = item ,
|
|
_data = data ,
|
|
widget = wibox.widget.background,
|
|
}
|
|
|
|
-- Make some widgets easier to access
|
|
item._internal.margin_w = item.widget:get_children_by_id("main_margin")[1]
|
|
item._internal.align = item.widget:get_children_by_id("main_align" )[1]
|
|
|
|
-- Override some methods
|
|
item.widget._after_draw_children = item.widget.after_draw_children
|
|
item.widget.after_draw_children = module.after_draw_children
|
|
item._internal.align.fit = data._internal.align_fit or align_fit
|
|
item._internal.text_w = tb
|
|
item._internal.icon_w = icon
|
|
|
|
-- Export the margin
|
|
local mrgns = margins2(
|
|
item._internal.margin_w,
|
|
util.table.join(
|
|
(item.item_style or data.item_style).margins,data.default_item_margins
|
|
)
|
|
)
|
|
|
|
function item:get_margins()
|
|
return mrgns
|
|
end
|
|
|
|
-- Draw
|
|
local item_style = item.style or data.item_style
|
|
item_style(item,{})
|
|
|
|
-- Setup dynamic underlay
|
|
item:connect_signal("underlay::changed",function(_,udl)
|
|
item.widget:emit_signal("widget::updated")
|
|
end)
|
|
|
|
-- Setup events
|
|
module.setup_event(data,item)
|
|
|
|
return item.widget
|
|
end
|
|
|
|
return setmetatable(module, { __call = function(_, ...) return create_item(...) end })
|
|
-- kate: space-indent on; indent-width 4; replace-tabs on;
|