2016-08-05 00:12:38 +02:00
|
|
|
---------------------------------------------------------------------------
|
|
|
|
-- An interactive mouse based slider widget.
|
|
|
|
--
|
|
|
|
--@DOC_wibox_widget_defaults_slider_EXAMPLE@
|
|
|
|
--
|
|
|
|
-- @author Grigory Mishchenko <grishkokot@gmail.com>
|
|
|
|
-- @author Emmanuel Lepage Vallee <elv1313@gmail.com>
|
|
|
|
-- @copyright 2015 Grigory Mishchenko, 2016 Emmanuel Lepage Vallee
|
2019-06-06 08:15:53 +02:00
|
|
|
-- @widgetmod wibox.widget.slider
|
2021-03-27 20:41:28 +01:00
|
|
|
-- @supermodule wibox.widget.base
|
2016-08-05 00:12:38 +02:00
|
|
|
---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
local setmetatable = setmetatable
|
|
|
|
local type = type
|
|
|
|
local color = require("gears.color")
|
2017-03-08 21:18:33 +01:00
|
|
|
local gtable = require("gears.table")
|
2016-08-05 00:12:38 +02:00
|
|
|
local beautiful = require("beautiful")
|
|
|
|
local base = require("wibox.widget.base")
|
|
|
|
local shape = require("gears.shape")
|
|
|
|
local capi = {
|
|
|
|
mouse = mouse,
|
|
|
|
mousegrabber = mousegrabber,
|
|
|
|
root = root,
|
|
|
|
}
|
|
|
|
|
|
|
|
local slider = {mt={}}
|
|
|
|
|
|
|
|
--- The slider handle shape.
|
|
|
|
--
|
|
|
|
--@DOC_wibox_widget_slider_handle_shape_EXAMPLE@
|
|
|
|
--
|
|
|
|
-- @property handle_shape
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @tparam shape|nil handle_shape
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propemits true false
|
|
|
|
-- @propbeautiful
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @see gears.shape
|
|
|
|
|
2022-10-03 04:26:23 +02:00
|
|
|
--- The slider handle widget.
|
|
|
|
--
|
2022-10-01 04:41:40 +02:00
|
|
|
--@DOC_wibox_widget_slider_handle_widget_EXAMPLE@
|
2022-10-03 04:26:23 +02:00
|
|
|
--
|
2022-10-01 04:41:40 +02:00
|
|
|
-- @property handle_widget
|
2022-10-03 04:26:23 +02:00
|
|
|
-- @tparam[opt=nil] widget|nil handle_widget The handle widget
|
2022-10-01 04:41:40 +02:00
|
|
|
|
2016-08-05 00:12:38 +02:00
|
|
|
--- The slider handle color.
|
|
|
|
--
|
|
|
|
--@DOC_wibox_widget_slider_handle_color_EXAMPLE@
|
|
|
|
--
|
|
|
|
-- @property handle_color
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propbeautiful
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @tparam color|nil handle_color
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propemits true false
|
2016-08-05 00:12:38 +02:00
|
|
|
|
|
|
|
--- The slider handle margins.
|
|
|
|
--
|
|
|
|
--@DOC_wibox_widget_slider_handle_margins_EXAMPLE@
|
|
|
|
--
|
|
|
|
-- @property handle_margins
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @tparam[opt={}] table|number|nil handle_margins
|
|
|
|
-- @tparam[opt=0] number handle_margins.left
|
|
|
|
-- @tparam[opt=0] number handle_margins.right
|
|
|
|
-- @tparam[opt=0] number handle_margins.top
|
|
|
|
-- @tparam[opt=0] number handle_margins.bottom
|
|
|
|
-- @propertyunit pixel
|
|
|
|
-- @propertytype number A single value used for all sides.
|
|
|
|
-- @propertytype table A different value for each side. The side names are:
|
|
|
|
-- @negativeallowed true
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propemits true false
|
|
|
|
-- @propbeautiful
|
2016-08-05 00:12:38 +02:00
|
|
|
|
|
|
|
--- The slider handle width.
|
|
|
|
--
|
|
|
|
--@DOC_wibox_widget_slider_handle_width_EXAMPLE@
|
|
|
|
--
|
|
|
|
-- @property handle_width
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @tparam number|nil handle_width
|
|
|
|
-- @negativeallowed false
|
|
|
|
-- @propertyunit pixel
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propemits true false
|
|
|
|
-- @propbeautiful
|
2016-08-05 00:12:38 +02:00
|
|
|
|
|
|
|
--- The handle border_color.
|
|
|
|
--
|
|
|
|
--@DOC_wibox_widget_slider_handle_border_EXAMPLE@
|
|
|
|
--
|
|
|
|
-- @property handle_border_color
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @tparam color|nil handle_border_color
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propemits true false
|
|
|
|
-- @propbeautiful
|
2016-08-05 00:12:38 +02:00
|
|
|
|
|
|
|
--- The handle border width.
|
|
|
|
-- @property handle_border_width
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @tparam[opt=0] number|nil handle_border_width
|
|
|
|
-- @propertyunit pixel
|
|
|
|
-- @negativeallowed false
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propemits true false
|
|
|
|
-- @propbeautiful
|
2016-08-05 00:12:38 +02:00
|
|
|
|
|
|
|
--- The bar (background) shape.
|
|
|
|
--
|
|
|
|
--@DOC_wibox_widget_slider_bar_shape_EXAMPLE@
|
|
|
|
--
|
|
|
|
-- @property bar_shape
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @tparam shape|nil bar_shape
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propemits true false
|
|
|
|
-- @propbeautiful
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @see gears.shape
|
|
|
|
|
2022-10-03 04:26:23 +02:00
|
|
|
--- The bar widget
|
|
|
|
--
|
2022-10-01 04:41:40 +02:00
|
|
|
--@DOC_wibox_widget_slider_bar_widget_EXAMPLE@
|
|
|
|
--
|
|
|
|
-- @property bar_widget
|
2022-10-03 04:26:23 +02:00
|
|
|
-- @tparam[opt=nil] widget|nil bar_widget The widget that spans the active bar segment
|
2022-10-01 04:41:40 +02:00
|
|
|
|
2016-08-05 00:12:38 +02:00
|
|
|
--- The bar (background) height.
|
|
|
|
--
|
|
|
|
--@DOC_wibox_widget_slider_bar_height_EXAMPLE@
|
|
|
|
--
|
|
|
|
-- @property bar_height
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @tparam number|nil bar_height
|
|
|
|
-- @propertyunit pixel
|
|
|
|
-- @negativeallowed false
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propbeautiful
|
|
|
|
-- @propemits true false
|
2016-08-05 00:12:38 +02:00
|
|
|
|
|
|
|
--- The bar (background) color.
|
|
|
|
--
|
|
|
|
--@DOC_wibox_widget_slider_bar_color_EXAMPLE@
|
|
|
|
--
|
|
|
|
-- @property bar_color
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @tparam color|nil bar_color
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propbeautiful
|
|
|
|
-- @propemits true false
|
2016-08-05 00:12:38 +02:00
|
|
|
|
2019-04-24 05:13:32 +02:00
|
|
|
--- The bar (active) color.
|
|
|
|
--
|
|
|
|
--@DOC_wibox_widget_slider_bar_active_color_EXAMPLE@
|
|
|
|
--
|
|
|
|
-- Only works when both `bar_active_color` and `bar_color` are passed as hex color string
|
|
|
|
-- @property bar_active_color
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @tparam color|nil bar_active_color
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propbeautiful
|
|
|
|
-- @propemits true false
|
2019-04-24 05:13:32 +02:00
|
|
|
|
2016-08-05 00:12:38 +02:00
|
|
|
--- The bar (background) margins.
|
|
|
|
--
|
|
|
|
--@DOC_wibox_widget_slider_bar_margins_EXAMPLE@
|
|
|
|
--
|
|
|
|
-- @property bar_margins
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @tparam[opt={}] table|number|nil bar_margins
|
|
|
|
-- @tparam[opt=0] number bar_margins.left
|
|
|
|
-- @tparam[opt=0] number bar_margins.right
|
|
|
|
-- @tparam[opt=0] number bar_margins.top
|
|
|
|
-- @tparam[opt=0] number bar_margins.bottom
|
|
|
|
-- @propertyunit pixel
|
|
|
|
-- @propertytype number A single value used for all sides.
|
|
|
|
-- @propertytype table A different value for each side. The side names are:
|
|
|
|
-- @negativeallowed true
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propbeautiful
|
|
|
|
-- @propemits true false
|
2016-08-05 00:12:38 +02:00
|
|
|
|
|
|
|
--- The bar (background) border width.
|
|
|
|
-- @property bar_border_width
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @tparam[opt=0] number|nil bar_border_width
|
|
|
|
-- @propertyunit pixel
|
|
|
|
-- @negativeallowed false
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propemits true false
|
2021-12-21 06:54:15 +01:00
|
|
|
-- @propbeautiful
|
2016-08-05 00:12:38 +02:00
|
|
|
|
|
|
|
--- The bar (background) border_color.
|
|
|
|
--
|
|
|
|
--@DOC_wibox_widget_slider_bar_border_EXAMPLE@
|
|
|
|
--
|
|
|
|
-- @property bar_border_color
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @tparam color|nil bar_border_color
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propbeautiful
|
|
|
|
-- @propemits true false
|
2016-08-05 00:12:38 +02:00
|
|
|
|
|
|
|
--- The slider value.
|
|
|
|
--
|
|
|
|
--@DOC_wibox_widget_slider_value_EXAMPLE@
|
|
|
|
--
|
|
|
|
-- @property value
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @tparam[opt=0] number value
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @negativeallowed true
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propemits true false
|
2016-08-05 00:12:38 +02:00
|
|
|
|
|
|
|
--- The slider minimum value.
|
2019-11-29 06:33:44 +01:00
|
|
|
--
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @property minimum
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @tparam[opt=0] number minimum
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @negativeallowed true
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propemits true false
|
2016-08-05 00:12:38 +02:00
|
|
|
|
|
|
|
--- The slider maximum value.
|
2019-11-29 06:33:44 +01:00
|
|
|
--
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @property maximum
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @tparam[opt=100] number maximum
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @negativeallowed true
|
2019-11-29 06:33:44 +01:00
|
|
|
-- @propemits true false
|
2016-08-05 00:12:38 +02:00
|
|
|
|
|
|
|
--- The bar (background) border width.
|
2019-11-29 06:33:44 +01:00
|
|
|
--
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @beautiful beautiful.slider_bar_border_width
|
|
|
|
-- @param number
|
|
|
|
|
|
|
|
--- The bar (background) border color.
|
2019-11-29 06:33:44 +01:00
|
|
|
--
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @beautiful beautiful.slider_bar_border_color
|
|
|
|
-- @param color
|
|
|
|
|
|
|
|
--- The handle border_color.
|
2019-11-29 06:33:44 +01:00
|
|
|
--
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @beautiful beautiful.slider_handle_border_color
|
|
|
|
-- @param color
|
|
|
|
|
|
|
|
--- The handle border width.
|
2019-11-29 06:33:44 +01:00
|
|
|
--
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @beautiful beautiful.slider_handle_border_width
|
|
|
|
-- @param number
|
|
|
|
|
2019-11-29 06:33:44 +01:00
|
|
|
--- The handle width.
|
|
|
|
--
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @beautiful beautiful.slider_handle_width
|
|
|
|
-- @param number
|
|
|
|
|
2019-11-29 06:33:44 +01:00
|
|
|
--- The handle color.
|
|
|
|
--
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @beautiful beautiful.slider_handle_color
|
|
|
|
-- @param color
|
|
|
|
|
|
|
|
--- The handle shape.
|
2019-11-29 06:33:44 +01:00
|
|
|
--
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @beautiful beautiful.slider_handle_shape
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @tparam[opt=gears.shape.rectangle] gears.shape shape
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @see gears.shape
|
|
|
|
|
|
|
|
--- The bar (background) shape.
|
2019-11-29 06:33:44 +01:00
|
|
|
--
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @beautiful beautiful.slider_bar_shape
|
2022-08-22 08:02:27 +02:00
|
|
|
-- @tparam[opt=gears.shape.rectangle] gears.shape shape
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @see gears.shape
|
|
|
|
|
|
|
|
--- The bar (background) height.
|
2019-11-29 06:33:44 +01:00
|
|
|
--
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @beautiful beautiful.slider_bar_height
|
|
|
|
-- @param number
|
|
|
|
|
|
|
|
--- The bar (background) margins.
|
2019-11-29 06:33:44 +01:00
|
|
|
--
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @beautiful beautiful.slider_bar_margins
|
|
|
|
-- @tparam[opt={}] table margins
|
|
|
|
-- @tparam[opt=0] number margins.left
|
|
|
|
-- @tparam[opt=0] number margins.right
|
|
|
|
-- @tparam[opt=0] number margins.top
|
|
|
|
-- @tparam[opt=0] number margins.bottom
|
|
|
|
|
|
|
|
--- The slider handle margins.
|
2019-11-29 06:33:44 +01:00
|
|
|
--
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @beautiful beautiful.slider_handle_margins
|
|
|
|
-- @tparam[opt={}] table margins
|
|
|
|
-- @tparam[opt=0] number margins.left
|
|
|
|
-- @tparam[opt=0] number margins.right
|
|
|
|
-- @tparam[opt=0] number margins.top
|
|
|
|
-- @tparam[opt=0] number margins.bottom
|
|
|
|
|
|
|
|
--- The bar (background) color.
|
2019-11-29 06:33:44 +01:00
|
|
|
--
|
2016-08-05 00:12:38 +02:00
|
|
|
-- @beautiful beautiful.slider_bar_color
|
|
|
|
-- @param color
|
|
|
|
|
2019-04-24 05:13:32 +02:00
|
|
|
--- The bar (active) color.
|
2019-11-29 06:33:44 +01:00
|
|
|
--
|
2019-04-24 05:13:32 +02:00
|
|
|
-- Only works when both `beautiful.slider_bar_color` and `beautiful.slider_bar_active_color` are hex color strings
|
|
|
|
-- @beautiful beautiful.slider_bar_active_color
|
|
|
|
-- @param color
|
|
|
|
|
|
|
|
|
2016-08-05 00:12:38 +02:00
|
|
|
local properties = {
|
|
|
|
-- Handle
|
|
|
|
handle_shape = shape.rectangle,
|
2022-10-03 04:26:23 +02:00
|
|
|
handle_widget = false,
|
2016-08-05 00:12:38 +02:00
|
|
|
handle_color = false,
|
|
|
|
handle_margins = {},
|
|
|
|
handle_width = false,
|
|
|
|
handle_border_width = 0,
|
|
|
|
handle_border_color = false,
|
|
|
|
|
|
|
|
-- Bar
|
|
|
|
bar_shape = shape.rectangle,
|
|
|
|
bar_height = false,
|
|
|
|
bar_color = false,
|
2019-04-24 05:13:32 +02:00
|
|
|
bar_active_color = false,
|
2016-08-05 00:12:38 +02:00
|
|
|
bar_margins = {},
|
|
|
|
bar_border_width = 0,
|
|
|
|
bar_border_color = false,
|
2022-09-27 01:49:19 +02:00
|
|
|
bar_widget = false,
|
2016-08-05 00:12:38 +02:00
|
|
|
|
|
|
|
-- Content
|
|
|
|
value = 0,
|
|
|
|
minimum = 0,
|
|
|
|
maximum = 100,
|
|
|
|
}
|
|
|
|
|
|
|
|
-- Create the accessors
|
|
|
|
for prop in pairs(properties) do
|
|
|
|
slider["set_"..prop] = function(self, value)
|
|
|
|
local changed = self._private[prop] ~= value
|
|
|
|
self._private[prop] = value
|
|
|
|
|
|
|
|
if changed then
|
2019-11-29 06:33:44 +01:00
|
|
|
self:emit_signal("property::"..prop, value)
|
2016-08-05 00:12:38 +02:00
|
|
|
self:emit_signal("widget::redraw_needed")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
slider["get_"..prop] = function(self)
|
|
|
|
-- Ignoring the false's is on purpose
|
|
|
|
return self._private[prop] == nil
|
|
|
|
and properties[prop]
|
|
|
|
or self._private[prop]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Add some validation to set_value
|
|
|
|
function slider:set_value(value)
|
|
|
|
value = math.min(value, self:get_maximum())
|
|
|
|
value = math.max(value, self:get_minimum())
|
|
|
|
local changed = self._private.value ~= value
|
|
|
|
|
|
|
|
self._private.value = value
|
|
|
|
|
|
|
|
if changed then
|
2019-11-29 06:33:44 +01:00
|
|
|
self:emit_signal( "property::value", value)
|
2016-08-05 00:12:38 +02:00
|
|
|
self:emit_signal( "widget::redraw_needed" )
|
2022-10-01 04:41:40 +02:00
|
|
|
if self._private.handle_widget or self._private.bar_widget then
|
2022-09-27 01:49:19 +02:00
|
|
|
self:emit_signal("widget::layout_changed")
|
|
|
|
end
|
2016-08-05 00:12:38 +02:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local function get_extremums(self)
|
|
|
|
local min = self._private.minimum or properties.minimum
|
|
|
|
local max = self._private.maximum or properties.maximum
|
|
|
|
local interval = max - min
|
|
|
|
|
|
|
|
return min, max, interval
|
|
|
|
end
|
|
|
|
|
2022-09-27 01:49:19 +02:00
|
|
|
--TODO Document Margins
|
|
|
|
|
|
|
|
function slider:set_handle_margins(value)
|
|
|
|
if type(value) == "number" then
|
|
|
|
value = {
|
|
|
|
top = value,
|
|
|
|
bottom = value,
|
|
|
|
left = value,
|
|
|
|
right = value,
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
self._private.handle_margins = value
|
|
|
|
self:emit_signal( "property::handle_margins", value)
|
|
|
|
self:emit_signal( "widget::redraw_needed" )
|
|
|
|
self:emit_signal("widget::layout_changed")
|
|
|
|
end
|
|
|
|
|
|
|
|
function slider:set_bar_margins(value)
|
|
|
|
if type(value) == "number" then
|
|
|
|
value = {
|
|
|
|
top = value,
|
|
|
|
bottom = value,
|
|
|
|
left = value,
|
|
|
|
right = value,
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
self._private.bar_margins = value
|
|
|
|
self:emit_signal( "property::bar_margins", value)
|
|
|
|
self:emit_signal( "widget::redraw_needed" )
|
|
|
|
self:emit_signal("widget::layout_changed")
|
|
|
|
end
|
|
|
|
|
2016-08-05 00:12:38 +02:00
|
|
|
function slider:draw(_, cr, width, height)
|
2019-04-24 05:13:32 +02:00
|
|
|
local value = self._private.value or self._private.min or 0
|
|
|
|
|
|
|
|
local maximum = self._private.maximum
|
|
|
|
or properties.maximum
|
|
|
|
|
|
|
|
local minimum = self._private.minimum
|
|
|
|
or properties.minimum
|
|
|
|
|
|
|
|
local range = maximum - minimum
|
|
|
|
local active_rate = (value - minimum) / range
|
|
|
|
|
|
|
|
local handle_height, handle_width = height, self._private.handle_width
|
|
|
|
or beautiful.slider_handle_width
|
2021-04-19 10:12:16 +02:00
|
|
|
or math.floor(height/2)
|
2019-04-24 05:13:32 +02:00
|
|
|
|
|
|
|
local handle_border_width = self._private.handle_border_width
|
|
|
|
or beautiful.slider_handle_border_width
|
|
|
|
or properties.handle_border_width or 0
|
|
|
|
|
2016-08-05 00:12:38 +02:00
|
|
|
local bar_height = self._private.bar_height
|
|
|
|
|
|
|
|
-- If there is no background, then skip this
|
|
|
|
local bar_color = self._private.bar_color
|
|
|
|
or beautiful.slider_bar_color
|
|
|
|
|
2019-04-24 05:13:32 +02:00
|
|
|
local bar_active_color = self._private.bar_active_color
|
|
|
|
or beautiful.slider_bar_active_color
|
|
|
|
|
2016-08-05 00:12:38 +02:00
|
|
|
if bar_color then
|
|
|
|
cr:set_source(color(bar_color))
|
|
|
|
end
|
|
|
|
|
|
|
|
local margins = self._private.bar_margins
|
|
|
|
or beautiful.slider_bar_margins
|
|
|
|
|
|
|
|
local x_offset, right_margin, y_offset = 0, 0
|
|
|
|
|
|
|
|
if margins then
|
2022-09-27 01:49:19 +02:00
|
|
|
bar_height = bar_height or (
|
|
|
|
height - (margins.top or 0) - (margins.bottom or 0)
|
|
|
|
)
|
2022-10-01 04:41:40 +02:00
|
|
|
x_offset, y_offset = margins.left or 0,
|
|
|
|
margins.top or 0
|
2022-09-27 01:49:19 +02:00
|
|
|
right_margin = margins.right or 0
|
2016-08-05 00:12:38 +02:00
|
|
|
else
|
|
|
|
bar_height = bar_height or beautiful.slider_bar_height or height
|
2021-04-19 10:12:16 +02:00
|
|
|
y_offset = math.floor((height - bar_height)/2)
|
2016-08-05 00:12:38 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
cr:translate(x_offset, y_offset)
|
|
|
|
|
|
|
|
local bar_shape = self._private.bar_shape
|
|
|
|
or beautiful.slider_bar_shape
|
|
|
|
or properties.bar_shape
|
|
|
|
|
|
|
|
local bar_border_width = self._private.bar_border_width
|
|
|
|
or beautiful.slider_bar_border_width
|
|
|
|
or properties.bar_border_width
|
|
|
|
|
|
|
|
bar_shape(cr, width - x_offset - right_margin, bar_height or height)
|
|
|
|
|
2019-04-24 05:13:32 +02:00
|
|
|
if bar_active_color and type(bar_color) == "string" and type(bar_active_color) == "string" then
|
2021-04-19 10:12:16 +02:00
|
|
|
local bar_active_width = math.floor(
|
|
|
|
active_rate * (width - x_offset - right_margin)
|
2019-04-24 05:13:32 +02:00
|
|
|
- (handle_width - handle_border_width/2) * (active_rate - 0.5)
|
2021-04-19 10:12:16 +02:00
|
|
|
)
|
2019-04-24 05:13:32 +02:00
|
|
|
cr:set_source(color.create_pattern{
|
|
|
|
type = "linear",
|
|
|
|
from = {0,0},
|
|
|
|
to = {bar_active_width, 0},
|
|
|
|
stops = {{0.99, bar_active_color}, {0.99, bar_color}}
|
|
|
|
})
|
|
|
|
end
|
|
|
|
|
2016-08-05 00:12:38 +02:00
|
|
|
if bar_color then
|
|
|
|
if bar_border_width == 0 then
|
|
|
|
cr:fill()
|
|
|
|
else
|
|
|
|
cr:fill_preserve()
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Draw the bar border
|
|
|
|
if bar_border_width > 0 then
|
|
|
|
local bar_border_color = self._private.bar_border_color
|
|
|
|
or beautiful.slider_bar_border_color
|
|
|
|
or properties.bar_border_color
|
|
|
|
|
|
|
|
cr:set_line_width(bar_border_width)
|
|
|
|
|
|
|
|
if bar_border_color then
|
|
|
|
cr:save()
|
|
|
|
cr:set_source(color(bar_border_color))
|
|
|
|
cr:stroke()
|
|
|
|
cr:restore()
|
|
|
|
else
|
|
|
|
cr:stroke()
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
cr:translate(-x_offset, -y_offset)
|
|
|
|
|
|
|
|
-- Paint the handle
|
|
|
|
local handle_color = self._private.handle_color
|
|
|
|
or beautiful.slider_handle_color
|
|
|
|
|
|
|
|
-- It is ok if there is no color, it will be inherited
|
|
|
|
if handle_color then
|
|
|
|
cr:set_source(color(handle_color))
|
|
|
|
end
|
|
|
|
|
|
|
|
local handle_shape = self._private.handle_shape
|
|
|
|
or beautiful.slider_handle_shape
|
|
|
|
or properties.handle_shape
|
|
|
|
|
|
|
|
-- Lets get the margins for the handle
|
|
|
|
margins = self._private.handle_margins
|
|
|
|
or beautiful.slider_handle_margins
|
|
|
|
|
|
|
|
x_offset, y_offset = 0, 0
|
|
|
|
|
|
|
|
if margins then
|
2022-10-01 04:41:40 +02:00
|
|
|
x_offset, y_offset = margins.left or 0,
|
|
|
|
margins.top or 0
|
2022-09-27 01:49:19 +02:00
|
|
|
handle_width = handle_width -
|
|
|
|
(margins.left or 0) - (margins.right or 0)
|
|
|
|
handle_height = handle_height -
|
|
|
|
(margins.top or 0) - (margins.bottom or 0)
|
2016-08-05 00:12:38 +02:00
|
|
|
end
|
|
|
|
|
|
|
|
-- Get the widget size back to it's non-transfored value
|
|
|
|
local min, _, interval = get_extremums(self)
|
2021-04-19 10:12:16 +02:00
|
|
|
local rel_value = math.floor(((value-min)/interval) * (width-handle_width))
|
2016-08-05 00:12:38 +02:00
|
|
|
|
|
|
|
cr:translate(x_offset + rel_value, y_offset)
|
|
|
|
|
|
|
|
handle_shape(cr, handle_width, handle_height)
|
|
|
|
|
|
|
|
if handle_border_width > 0 then
|
|
|
|
cr:fill_preserve()
|
|
|
|
else
|
|
|
|
cr:fill()
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Draw the handle border
|
|
|
|
if handle_border_width > 0 then
|
|
|
|
local handle_border_color = self._private.handle_border_color
|
|
|
|
or beautiful.slider_handle_border_color
|
|
|
|
or properties.handle_border_color
|
|
|
|
|
|
|
|
if handle_border_color then
|
|
|
|
cr:set_source(color(handle_border_color))
|
|
|
|
end
|
|
|
|
|
|
|
|
cr:set_line_width(handle_border_width)
|
|
|
|
cr:stroke()
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
function slider:fit(_, width, height)
|
|
|
|
-- Use all the space, this should be used with a constraint widget
|
|
|
|
return width, height
|
|
|
|
end
|
|
|
|
|
2022-09-19 21:21:39 +02:00
|
|
|
function slider:layout(context, width, height)
|
2022-09-27 01:49:19 +02:00
|
|
|
local result = {}
|
|
|
|
local handle_widget = self._private.handle_widget
|
|
|
|
or nil
|
|
|
|
local bar_widget = self._private.bar_widget
|
|
|
|
or nil
|
|
|
|
--[[ if not handle_widget or not bar_widget then
|
|
|
|
return {}
|
|
|
|
end ]]
|
|
|
|
|
|
|
|
|
|
|
|
local value = self._private.value or self._private.min or 0
|
|
|
|
local min, _, interval = get_extremums(self)
|
|
|
|
local bar_height, bar_width = self._private.bar_height or height, width
|
2022-09-19 21:21:39 +02:00
|
|
|
local handle_height, handle_width = height, self._private.handle_width
|
|
|
|
or beautiful.slider_handle_width
|
2022-10-01 04:41:40 +02:00
|
|
|
or width
|
2022-09-27 01:49:19 +02:00
|
|
|
|
|
|
|
if bar_widget then
|
|
|
|
bar_width = (((value - min) / interval) * width + handle_width / 2 - ((value - min) / interval ) * handle_width)
|
|
|
|
local w, h = base.fit_widget(self, context, bar_widget, bar_width, bar_height)
|
|
|
|
local margins = self._private.bar_margins
|
|
|
|
or beautiful.slider_bar_margins
|
|
|
|
|
|
|
|
local x_offset, y_offset = 0, 0
|
|
|
|
|
|
|
|
if margins then
|
|
|
|
x_offset, y_offset = margins.left or 0, margins.top or 0
|
|
|
|
bar_width = bar_width -
|
|
|
|
(margins.left or 0) - (margins.right or 0)
|
|
|
|
end
|
|
|
|
local x, y = 0 + x_offset, 0 + y_offset
|
|
|
|
table.insert(result, base.place_widget_at(bar_widget, x, y, w, h))
|
|
|
|
end
|
2022-09-19 21:21:39 +02:00
|
|
|
if handle_widget then
|
2022-09-27 01:49:19 +02:00
|
|
|
local w, h = base.fit_widget(self, context, handle_widget, handle_width, handle_height)
|
2022-09-19 21:21:39 +02:00
|
|
|
local margins = self._private.handle_margins
|
|
|
|
or beautiful.slider_handle_margins
|
|
|
|
|
|
|
|
local x_offset, y_offset = 0, 0
|
|
|
|
|
|
|
|
if margins then
|
|
|
|
x_offset, y_offset = margins.left or 0, margins.top or 0
|
|
|
|
handle_width = handle_width -
|
|
|
|
(margins.left or 0) - (margins.right or 0)
|
|
|
|
end
|
|
|
|
|
|
|
|
x_offset = x_offset + handle_width / 2 - w / 2
|
|
|
|
|
2022-09-27 01:49:19 +02:00
|
|
|
local x = (((value - min) / interval) * (width - handle_width) + x_offset)
|
|
|
|
local y = (height / 2) + y_offset - h / 2
|
|
|
|
table.insert(result, base.place_widget_at(handle_widget, x, y, w, h))
|
2022-09-19 21:21:39 +02:00
|
|
|
end
|
2022-09-27 01:49:19 +02:00
|
|
|
|
|
|
|
return result
|
2022-09-19 21:21:39 +02:00
|
|
|
end
|
|
|
|
|
2016-08-05 00:12:38 +02:00
|
|
|
-- Move the handle to the correct location
|
|
|
|
local function move_handle(self, width, x, _)
|
|
|
|
local _, _, interval = get_extremums(self)
|
|
|
|
self:set_value(math.floor((x*interval)/width))
|
|
|
|
end
|
|
|
|
|
|
|
|
local function mouse_press(self, x, y, button_id, _, geo)
|
|
|
|
if button_id ~= 1 then return end
|
|
|
|
|
|
|
|
local matrix_from_device = geo.hierarchy:get_matrix_from_device()
|
|
|
|
|
|
|
|
-- Sigh. geo.width/geo.height is in device space. We need it in our own
|
|
|
|
-- coordinate system
|
|
|
|
local width = geo.widget_width
|
|
|
|
|
|
|
|
move_handle(self, width, x, y)
|
|
|
|
|
|
|
|
-- Calculate a matrix transforming from screen coordinates into widget coordinates
|
|
|
|
local wgeo = geo.drawable.drawable:geometry()
|
|
|
|
local matrix = matrix_from_device:translate(-wgeo.x, -wgeo.y)
|
|
|
|
|
|
|
|
capi.mousegrabber.run(function(mouse)
|
|
|
|
if not mouse.buttons[1] then
|
|
|
|
return false
|
|
|
|
end
|
|
|
|
|
|
|
|
-- Calculate the point relative to the widget
|
|
|
|
move_handle(self, width, matrix:transform_point(mouse.x, mouse.y))
|
|
|
|
|
|
|
|
return true
|
|
|
|
end,"fleur")
|
|
|
|
end
|
|
|
|
|
|
|
|
--- Create a slider widget.
|
2021-12-21 06:54:15 +01:00
|
|
|
--
|
2019-06-07 20:59:34 +02:00
|
|
|
-- @constructorfct wibox.widget.slider
|
2021-12-21 06:54:15 +01:00
|
|
|
-- @tparam[opt={}] table args
|
|
|
|
-- @tparam[opt] gears.shape args.handle_shape The slider handle shape.
|
2022-09-27 01:49:19 +02:00
|
|
|
-- @tparam[opt] widget args.handle_widget The slider handle widget.
|
2021-12-21 06:54:15 +01:00
|
|
|
-- @tparam[opt] color args.handle_color The slider handle color.
|
|
|
|
-- @tparam[opt] table args.handle_margins The slider handle margins.
|
|
|
|
-- @tparam[opt] number args.handle_width The slider handle width.
|
|
|
|
-- @tparam[opt] color args.handle_border_color The handle border_color.
|
|
|
|
-- @tparam[opt] number args.handle_border_width The handle border width.
|
|
|
|
-- @tparam[opt] gears.shape args.bar_shape The bar (background) shape.
|
2022-09-27 01:49:19 +02:00
|
|
|
-- @tparam[opt] widget args.bar_widget The active bar widget.
|
2021-12-21 06:54:15 +01:00
|
|
|
-- @tparam[opt] number args.bar_height The bar (background) height.
|
|
|
|
-- @tparam[opt] color args.bar_color The bar (background) color.
|
|
|
|
-- @tparam[opt] color args.bar_active_color The bar (active) color.
|
|
|
|
-- @tparam[opt] table args.bar_margins The bar (background) margins.
|
|
|
|
-- @tparam[opt] number args.bar_border_width The bar (background) border width.
|
|
|
|
-- @tparam[opt] color args.bar_border_color The bar (background) border_color.
|
|
|
|
-- @tparam[opt] number args.value The slider value.
|
|
|
|
-- @tparam[opt] number args.minimum The slider minimum value.
|
|
|
|
-- @tparam[opt] number args.maximum The slider maximum value.
|
2016-08-05 00:12:38 +02:00
|
|
|
local function new(args)
|
|
|
|
local ret = base.make_widget(nil, nil, {
|
|
|
|
enable_properties = true,
|
|
|
|
})
|
|
|
|
|
2017-03-08 21:18:33 +01:00
|
|
|
gtable.crush(ret, slider, true)
|
2016-08-05 00:12:38 +02:00
|
|
|
|
2022-09-27 01:49:19 +02:00
|
|
|
gtable.crush(ret, args or {})
|
|
|
|
|
2016-08-05 00:12:38 +02:00
|
|
|
ret:connect_signal("button::press", mouse_press)
|
|
|
|
|
|
|
|
return ret
|
|
|
|
end
|
|
|
|
|
2021-03-08 18:20:23 +01:00
|
|
|
function slider.mt:__call(...)
|
2016-08-05 00:12:38 +02:00
|
|
|
return new(...)
|
|
|
|
end
|
|
|
|
|
|
|
|
return setmetatable(slider, slider.mt)
|
|
|
|
|
|
|
|
-- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80
|