naughty: Add an icon widget.
This tracks the notification icon and add some resize strategies.
This commit is contained in:
parent
3592d33b6c
commit
f31afd8cb7
|
@ -0,0 +1,158 @@
|
||||||
|
----------------------------------------------------------------------------
|
||||||
|
--- A notification square icon.
|
||||||
|
--
|
||||||
|
-- This widget is a specialized `wibox.widget.imagebox` with the following extra
|
||||||
|
-- features:
|
||||||
|
--
|
||||||
|
-- * Honor the `beautiful` notification variables.
|
||||||
|
-- * Restrict the size avoid huge notifications
|
||||||
|
-- * Provides some strategies to handle small icons
|
||||||
|
-- * React to the `naughty.notification` object icon changes.
|
||||||
|
--
|
||||||
|
--@DOC_wibox_nwidget_icon_simple_EXAMPLE@
|
||||||
|
--
|
||||||
|
-- @author Emmanuel Lepage Vallee <elv1313@gmail.com>
|
||||||
|
-- @copyright 2017 Emmanuel Lepage Vallee
|
||||||
|
-- @widgetmod naughty.widget.icon
|
||||||
|
-- @see wibox.widget.imagebox
|
||||||
|
----------------------------------------------------------------------------
|
||||||
|
local imagebox = require("wibox.widget.imagebox")
|
||||||
|
local gtable = require("gears.table")
|
||||||
|
local beautiful = require("beautiful")
|
||||||
|
local dpi = require("beautiful.xresources").apply_dpi
|
||||||
|
|
||||||
|
local icon = {}
|
||||||
|
|
||||||
|
-- The default way to resize the icon.
|
||||||
|
-- @beautiful beautiful.notification_icon_resize_strategy
|
||||||
|
-- @param number
|
||||||
|
|
||||||
|
function icon:fit(_, width, height)
|
||||||
|
-- Until someone complains, adding a "leave blank space" isn't supported
|
||||||
|
if not self._private.image then return 0, 0 end
|
||||||
|
|
||||||
|
local maximum = math.min(width, height)
|
||||||
|
local strategy = self._private.resize_strategy or "resize"
|
||||||
|
local optimal = math.min(beautiful.notification_icon_size or dpi(48), maximum)
|
||||||
|
|
||||||
|
local w = self._private.image:get_width()
|
||||||
|
local h = self._private.image:get_height()
|
||||||
|
|
||||||
|
if strategy == "resize" then
|
||||||
|
return math.min(w, optimal, maximum), math.min(h, optimal, maximum)
|
||||||
|
else
|
||||||
|
return optimal, optimal
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
function icon:draw(_, cr, width, height)
|
||||||
|
if not self._private.image then return end
|
||||||
|
if width == 0 or height == 0 then return end
|
||||||
|
|
||||||
|
-- Let's scale the image so that it fits into (width, height)
|
||||||
|
local strategy = self._private.resize_strategy or "resize"
|
||||||
|
local w = self._private.image:get_width()
|
||||||
|
local h = self._private.image:get_height()
|
||||||
|
local aspect = width / w
|
||||||
|
local aspect_h = height / h
|
||||||
|
|
||||||
|
if aspect > aspect_h then aspect = aspect_h end
|
||||||
|
|
||||||
|
if aspect < 1 or (strategy == "scale" and (w < width or h < height)) then
|
||||||
|
cr:scale(aspect, aspect)
|
||||||
|
end
|
||||||
|
|
||||||
|
local x, y = 0, 0
|
||||||
|
|
||||||
|
if (strategy == "center" and aspect > 1) or strategy == "resize" then
|
||||||
|
x = math.floor((width - w) / 2)
|
||||||
|
y = math.floor((height - h) / 2)
|
||||||
|
end
|
||||||
|
|
||||||
|
cr:set_source_surface(self._private.image, x, y)
|
||||||
|
cr:paint()
|
||||||
|
end
|
||||||
|
|
||||||
|
--- The attached notification.
|
||||||
|
-- @property notification
|
||||||
|
-- @tparam naughty.notification notification
|
||||||
|
|
||||||
|
function icon:set_notification(notif)
|
||||||
|
if self._private.notification == notif then return end
|
||||||
|
|
||||||
|
if self._private.notification then
|
||||||
|
self._private.notification:disconnect_signal("destroyed",
|
||||||
|
self._private.icon_changed_callback)
|
||||||
|
end
|
||||||
|
|
||||||
|
self:set_image(notif.icon)
|
||||||
|
|
||||||
|
self._private.notification = notif
|
||||||
|
|
||||||
|
notif:connect_signal("poperty::icon", self._private.icon_changed_callback)
|
||||||
|
end
|
||||||
|
|
||||||
|
local valid_strategies = {
|
||||||
|
scale = true,
|
||||||
|
center = true,
|
||||||
|
resize = true,
|
||||||
|
}
|
||||||
|
|
||||||
|
--- How small icons are handled.
|
||||||
|
--
|
||||||
|
-- Valid values are:
|
||||||
|
--
|
||||||
|
-- * **scale**: Scale the icon up to the optimal size.
|
||||||
|
-- * **center**: Keep the icon size and draw it in the center
|
||||||
|
-- * **resize**: Change the size of the widget itself (*default*).
|
||||||
|
--
|
||||||
|
-- Note that the size upper bound is defined by
|
||||||
|
-- `beautiful.notification_icon_size`.
|
||||||
|
--
|
||||||
|
--@DOC_wibox_nwidget_icon_strategy_EXAMPLE@
|
||||||
|
--
|
||||||
|
-- @property resize_strategy
|
||||||
|
-- @param string
|
||||||
|
|
||||||
|
function icon:set_resize_strategy(strategy)
|
||||||
|
assert(valid_strategies[strategy], "Invalid strategy")
|
||||||
|
|
||||||
|
self._private.resize_strategy = strategy
|
||||||
|
|
||||||
|
self:emit_signal("widget::redraw_needed")
|
||||||
|
end
|
||||||
|
|
||||||
|
|
||||||
|
function icon:get_resize_strategy()
|
||||||
|
return self._private.resize_strategy
|
||||||
|
or beautiful.notification_icon_resize_strategy
|
||||||
|
or "resize"
|
||||||
|
end
|
||||||
|
|
||||||
|
--- Create a new naughty.widget.icon.
|
||||||
|
-- @tparam table args
|
||||||
|
-- @tparam naughty.notification args.notification The notification.
|
||||||
|
-- @constructorfct naughty.widget.icon
|
||||||
|
|
||||||
|
local function new(args)
|
||||||
|
args = args or {}
|
||||||
|
local tb = imagebox()
|
||||||
|
|
||||||
|
gtable.crush(tb, icon, true)
|
||||||
|
|
||||||
|
function tb._private.icon_changed_callback()
|
||||||
|
tb:set_image(tb._private.notification.icon)
|
||||||
|
end
|
||||||
|
|
||||||
|
if args.notification then
|
||||||
|
tb:set_notification(args.notification)
|
||||||
|
end
|
||||||
|
|
||||||
|
return tb
|
||||||
|
end
|
||||||
|
|
||||||
|
--@DOC_widget_COMMON@
|
||||||
|
|
||||||
|
--@DOC_object_COMMON@
|
||||||
|
|
||||||
|
return setmetatable(icon, {__call = function(_, ...) return new(...) end})
|
|
@ -6,4 +6,5 @@
|
||||||
|
|
||||||
return {
|
return {
|
||||||
title = require( "naughty.widget.title" );
|
title = require( "naughty.widget.title" );
|
||||||
|
icon = require( "naughty.widget.icon" );
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue