1113 lines
39 KiB
Lua
1113 lines
39 KiB
Lua
---------------------------------------------------------------------------
|
|
--- A graph widget.
|
|
--
|
|
-- The graph goes from left to right. To change this to right to left, use
|
|
-- a `wibox.container.mirror` widget. This can also be used to have data
|
|
-- shown from top to bottom.
|
|
--
|
|
-- To add text on top of the graph, use a `wibox.layout.stack` and a
|
|
-- `wibox.container.align` widgets.
|
|
--
|
|
-- To display the graph vertically, use a `wibox.container.rotate` widget.
|
|
--
|
|
--@DOC_wibox_widget_defaults_graph_EXAMPLE@
|
|
-- @author Julien Danjou <julien@danjou.info>
|
|
-- @copyright 2009 Julien Danjou
|
|
-- @widgetmod wibox.widget.graph
|
|
-- @supermodule wibox.widget.base
|
|
---------------------------------------------------------------------------
|
|
|
|
local setmetatable = setmetatable
|
|
local ipairs = ipairs
|
|
local math = math
|
|
local math_max = math.max
|
|
local math_min = math.min
|
|
local table = table
|
|
local type = type
|
|
local color = require("gears.color")
|
|
local gdebug = require("gears.debug")
|
|
local gtable = require("gears.table")
|
|
local base = require("wibox.widget.base")
|
|
local beautiful = require("beautiful")
|
|
|
|
local graph = { mt = {} }
|
|
|
|
--- Set the graph border_width.
|
|
--
|
|
--@DOC_wibox_widget_graph_border_width_EXAMPLE@
|
|
--
|
|
-- @property border_width
|
|
-- @tparam number border_width
|
|
-- @propemits true false
|
|
-- @see border_color
|
|
|
|
--- Set the graph border color.
|
|
--
|
|
--@DOC_wibox_widget_graph_border_color_EXAMPLE@
|
|
--
|
|
-- @property border_color
|
|
-- @tparam gears.color border_color The border color to set.
|
|
-- @propbeautiful
|
|
-- @propemits true false
|
|
-- @see gears.color
|
|
|
|
--- Set the graph foreground color.
|
|
--
|
|
-- This color is used, when `group_colors` isn't set.
|
|
--
|
|
--@DOC_wibox_widget_graph_color_EXAMPLE@
|
|
--
|
|
-- @property color
|
|
-- @tparam color color The graph color.
|
|
-- @usebeautiful beautiful.graph_fg
|
|
-- @propemits true false
|
|
-- @see gears.color
|
|
|
|
--- Set the graph background color.
|
|
--
|
|
--@DOC_wibox_widget_graph_background_color_EXAMPLE@
|
|
--
|
|
-- @property background_color
|
|
-- @tparam gears.color background_color The graph background color.
|
|
-- @usebeautiful beautiful.graph_bg
|
|
-- @propemits true false
|
|
-- @see gears.color
|
|
|
|
--- Set the colors for data groups.
|
|
--
|
|
-- Colors in this table are used to paint respective data groups.
|
|
-- When this property is unset (default), the `color` property is used
|
|
-- instead for all data groups.
|
|
-- When this property is set, but there's no color for a data group in it
|
|
-- (i.e. `group_colors`[group] is nil or false), then the respective
|
|
-- data group is disabled, i.e. not drawn.
|
|
--
|
|
-- @DOC_wibox_widget_graph_stacked_group_disable_EXAMPLE@
|
|
--
|
|
-- @property group_colors
|
|
-- @tparam table colors A table with colors for data groups.
|
|
-- @see gears.color
|
|
|
|
--- The maximum value the graph should handle.
|
|
--
|
|
-- This value corresponds to the top of the graph.
|
|
-- If `scale` is also set, the graph never scales up below this value, but it
|
|
-- automatically scales down to make all data fit.
|
|
-- If `scale` and `max_value` are unset, `max_value` defaults to 1.
|
|
--
|
|
-- @DOC_wibox_widget_graph_max_value_EXAMPLE@
|
|
--
|
|
-- @property max_value
|
|
-- @tparam number max_value
|
|
-- @propemits true false
|
|
|
|
--- The minimum value the graph should handle.
|
|
--
|
|
-- This value corresponds to the bottom of the graph.
|
|
-- If `scale` is also set, the graph never scales up above this value, but it
|
|
-- automatically scales down to make all data fit.
|
|
-- If `scale` and `min_value` are unset, `min_value` defaults to 0.
|
|
--
|
|
-- @DOC_wibox_widget_graph_min_value_EXAMPLE@
|
|
--
|
|
-- @property min_value
|
|
-- @tparam number min_value
|
|
-- @propemits true false
|
|
|
|
--- Set the graph to automatically scale its values. Default is false.
|
|
--
|
|
-- If this property is set to true, the graph calculates
|
|
-- effective `min_value` and `max_value` based on the displayed data,
|
|
-- so that all data fits on screen. The properties themselves aren't changed,
|
|
-- but the graph is drawn as though `min_value`(`max_value`) were equal to
|
|
-- the minimum(maximum) value among itself and the currently drawn values.
|
|
-- If `min_value`(`max_value`) is unset, then only the drawn values
|
|
-- are considered in this calculation.
|
|
--
|
|
-- @DOC_wibox_widget_graph_scale1_EXAMPLE@
|
|
--
|
|
-- @property scale
|
|
-- @tparam boolean scale
|
|
-- @propemits true false
|
|
|
|
--- Clamp graph bars to keep them inside the widget for out-of-range values.
|
|
--
|
|
-- Drawing values outside the [`min_value`, `max_value`] range leads to
|
|
-- bar shapes that exceed physical widget dimensions.
|
|
-- Most of the time this doesn't matter, because bar shapes are rectangles
|
|
-- and bar heights aren't large enough to trigger errors in the drawing system.
|
|
-- However for some shapes and values it does make a difference and leads
|
|
-- to visibly different and/or invalid result.
|
|
--
|
|
-- When this property is set to true (the default), the graph clamps
|
|
-- bars' heights to keep them within the graph.
|
|
--
|
|
-- @DOC_wibox_widget_graph_clamp_bars_EXAMPLE@
|
|
--
|
|
-- @property clamp_bars
|
|
-- @tparam boolean clamp_bars
|
|
-- @propemits true false
|
|
|
|
--- The value corresponding to the starting point of graph bars. Default is 0.
|
|
--
|
|
-- @DOC_wibox_widget_graph_baseline_value_EXAMPLE@
|
|
--
|
|
-- @property baseline_value
|
|
-- @tparam number baseline_value
|
|
-- @propemits true false
|
|
|
|
--- Set the width or the individual steps.
|
|
--
|
|
--@DOC_wibox_widget_graph_step_EXAMPLE@
|
|
--
|
|
-- @property step_width
|
|
-- @tparam[opt=1] number step_width
|
|
-- @propemits true false
|
|
|
|
--- Set the spacing between the steps.
|
|
--
|
|
--@DOC_wibox_widget_graph_step_spacing_EXAMPLE@
|
|
--
|
|
-- @property step_spacing
|
|
-- @tparam[opt=0] number step_spacing
|
|
-- @propemits true false
|
|
|
|
--- The step shape.
|
|
--
|
|
-- If `step_hook` property is also set, this property is ignored.
|
|
--
|
|
--@DOC_wibox_widget_graph_step_shape_EXAMPLE@
|
|
--
|
|
-- @property step_shape
|
|
-- @tparam[opt=rectangle] gears.shape|function step_shape
|
|
-- @propemits true false
|
|
-- @see gears.shape
|
|
|
|
--- Set the graph to draw stacks. Default is false.
|
|
--
|
|
-- When set to true, bars of each successive data group are drawn on top of
|
|
-- bars of previous groups, instead of the baseline.
|
|
-- This necessitates all data values to be non-negative.
|
|
-- Negative values, if present, will trigger @{nan_color|NaN indication}.
|
|
--
|
|
-- @DOC_wibox_widget_graph_normal_vs_stacked_EXAMPLE@
|
|
--
|
|
-- @property stack
|
|
-- @tparam boolean stack
|
|
-- @propemits true false
|
|
|
|
--- Display NaN indication. Default is true.
|
|
--
|
|
-- When the data contains [NaN](https://en.wikipedia.org/wiki/NaN) values,
|
|
-- and `nan_indication` is set, the corresponding area,
|
|
-- where the value bar should have been drawn, is filled
|
|
-- with the `nan_color` from top to bottom.
|
|
-- The painting is done after all other data is rendered,
|
|
-- to make sure that it won't be overpainted and go unnoticed.
|
|
--
|
|
-- @DOC_wibox_widget_graph_nan_color_EXAMPLE@
|
|
--
|
|
-- @property nan_indication
|
|
-- @tparam boolean nan_indication
|
|
-- @propemits true false
|
|
|
|
--- The color of NaN indication.
|
|
--
|
|
-- The color used when `nan_indication` is set.
|
|
-- Defaults to a yellow-black diagonal stripes pattern.
|
|
--
|
|
-- @DOC_wibox_widget_graph_stacked_nan_EXAMPLE@
|
|
--
|
|
-- @property nan_color
|
|
-- @tparam gears.color nan_color The color of NaN indication.
|
|
-- @propemits true false
|
|
-- @see gears.color
|
|
|
|
--- Control what is done before drawing each data group.
|
|
--
|
|
-- This property can be set to a `group_start_callback` function,
|
|
-- a number, or nil.
|
|
--
|
|
-- When `group_start` is nil (default), the default implementation
|
|
-- is used, which does `cr:set_source(data group's color)`.
|
|
-- It is user's responsibility to set colors how they wish,
|
|
-- if they set this property to their own function.
|
|
--
|
|
-- As a convenience, `group_start` property can be set to a number,
|
|
-- then this number will be used for bar shifting of all groups as
|
|
-- described for the return value of `group_start_callback`.
|
|
-- The group's color will then be set appropriately too,
|
|
-- as with group_start = nil.
|
|
--
|
|
-- @DOC_wibox_widget_graph_custom_draw_outline_EXAMPLE@
|
|
--
|
|
-- @property group_start
|
|
-- @within Advanced drawing properties
|
|
-- @tparam function|number|nil group_start
|
|
-- @propemits true false
|
|
|
|
--- User callback for preparing to draw a data group.
|
|
--
|
|
-- This callback is called once for each drawn data group, when its
|
|
-- values are about to be drawn.
|
|
--
|
|
-- `group_start_callback` may return a number, which would be added
|
|
-- to the x coordinate of all bars of the current group,
|
|
-- i.e. shifting its bars horizontally. This can be useful e.g.
|
|
-- for ensuring sharpness of 1px-thin vertical lines or ensuring
|
|
-- that the bars from different groups don't obscure each other.
|
|
-- User mustn't return more values from the callback than one,
|
|
-- they are reserved for future use.
|
|
--
|
|
-- @DOC_wibox_widget_graph_custom_draw_group_shift_EXAMPLE@
|
|
--
|
|
-- @callback group_start_callback
|
|
-- @within Advanced drawing callbacks
|
|
-- @tparam cairo.Context cr Cairo context
|
|
-- @tparam number group_idx The index of the currently drawn data group
|
|
-- @tparam @{draw_callback_options} options Additional info (@{draw_callback_options})
|
|
-- @treturn number|nil common horizontal offset for all bars of the group
|
|
-- @see group_start
|
|
|
|
--- Control what is done after drawing each data group.
|
|
--
|
|
-- This property can be set to a `group_finish_callback` function or nil.
|
|
--
|
|
-- When `group_finish` is nil (default), the default implementation
|
|
-- is used, which does `cr:fill()`.
|
|
--
|
|
-- @DOC_wibox_widget_graph_custom_draw_line_EXAMPLE@
|
|
--
|
|
-- @property group_finish
|
|
-- @within Advanced drawing properties
|
|
-- @tparam function|nil group_finish
|
|
-- @propemits true false
|
|
|
|
--- User callback for completing drawing a data group.
|
|
--
|
|
-- This callback is called once for each drawn data group, right after its
|
|
-- values were drawn.
|
|
--
|
|
-- At the moment of calling this callback, the cairo context is populated
|
|
-- with paths corresponding to bar shapes of the data group, but no actual
|
|
-- painting has taken place. This callback is supposed to do something about it.
|
|
--
|
|
-- @callback group_finish_callback
|
|
-- @within Advanced drawing callbacks
|
|
-- @tparam cairo.Context cr Cairo context
|
|
-- @tparam number group_idx The index of the currently drawn data group
|
|
-- @tparam @{draw_callback_options} options Additional info (@{draw_callback_options})
|
|
-- @see group_finish
|
|
|
|
--- Control what is done to draw each value.
|
|
--
|
|
-- This property can be set to a `step_hook_callback` function or nil.
|
|
--
|
|
-- When `step_hook` is nil (default), the default implementation
|
|
-- is used, which draws the values as `step_shape` bars.
|
|
--
|
|
-- @DOC_wibox_widget_graph_bezier_curve_EXAMPLE@
|
|
--
|
|
-- @property step_hook
|
|
-- @within Advanced drawing properties
|
|
-- @tparam function|nil step_hook
|
|
-- @propemits true false
|
|
-- @see gears.math
|
|
|
|
--- User callback for drawing a value.
|
|
--
|
|
-- This function is called in succession to draw the values of a data group,
|
|
-- starting with the newest value.
|
|
--
|
|
-- Prior to this, the `group_start_callback` will be called once for
|
|
-- the data group. And after `step_hook` was called for all values
|
|
-- necessary, the `group_finish_callback` will be called once.
|
|
--
|
|
-- A drawing session consists of repeating the above steps for each drawn
|
|
-- data group. The order of drawing of data groups is unspecified.
|
|
--
|
|
-- This function can assume, that nothing except itself interferes with
|
|
-- the widget or cairo context state between the paired
|
|
-- `group_start_callback`/`group_finish_callback` calls.
|
|
--
|
|
-- The coordinate system of the cairo context during all calls is the
|
|
-- one with (0,0) and (width, height) corresponding to the top-left
|
|
-- and the bottom-right corner of the graph's value drawing area respectively.
|
|
--
|
|
-- The parameters of this callback are pretty straightforward.
|
|
-- It deserves a note though, that the baseline\_y parameter could vary between
|
|
-- successive calls, because e.g. for stacked graphs, baseline\_y corresponds
|
|
-- to the tops of the bars of the lower data group.
|
|
--
|
|
-- It's also not necessary that `baseline_y >= value_y`, the opposite holds e.g.
|
|
-- for graphs with values smaller than `baseline_value`.
|
|
--
|
|
-- @callback step_hook_callback
|
|
-- @within Advanced drawing callbacks
|
|
-- @tparam cairo.Context cr Cairo context
|
|
-- @tparam number x The horizontal coordinate of the left edge of the bar.
|
|
-- @tparam number value_y The vertical coordinate corresponding to the drawn value.
|
|
-- Can be a NaN.
|
|
-- @tparam number baseline_y The vertical coordinate corresponding to the baseline.
|
|
-- @tparam number step_width Same as `step_width` (but never nil).
|
|
-- @tparam @{draw_callback_options} options Additional info (@{draw_callback_options}).
|
|
-- @see step_hook
|
|
|
|
--- The graph foreground color
|
|
-- Used, when the `color` property isn't set.
|
|
--
|
|
-- @beautiful beautiful.graph_fg
|
|
-- @param color
|
|
|
|
--- The graph background color.
|
|
-- Used, when the `background_color` property isn't set.
|
|
--
|
|
-- @beautiful beautiful.graph_bg
|
|
-- @param color
|
|
|
|
--- The graph border color.
|
|
-- Used, when the `border_color` property isn't set.
|
|
--
|
|
-- @beautiful beautiful.graph_border_color
|
|
-- @param color
|
|
|
|
local properties = { "width", "height", "border_color", "stack",
|
|
"stack_colors", "color", "background_color",
|
|
"max_value", "scale", "min_value", "step_shape",
|
|
"step_spacing", "step_width", "border_width",
|
|
"clamp_bars", "baseline_value",
|
|
"capacity", "nan_color", "nan_indication",
|
|
"group_start", "group_finish", "step_hook",
|
|
"group_colors",
|
|
}
|
|
|
|
-- This is what the properties are set to on widget construction.
|
|
local prop_defaults = {
|
|
baseline_value = 0,
|
|
clamp_bars = true,
|
|
nan_indication = true,
|
|
step_width = 1,
|
|
step_spacing = 0,
|
|
|
|
-- These aren't very useful to set, and the docs don't distinguish between
|
|
-- "defaults to" (equals to in a fresh instance) and
|
|
-- "falls back to" (is assumed to be equal to, when nil) anyway.
|
|
-- scale = false,
|
|
-- stack = false,
|
|
}
|
|
|
|
-- This is what the properties are assumed to be in the code, when unset/falsy.
|
|
local prop_fallbacks = {
|
|
-- This one might become beautiful-themed in the future, so we can't set it
|
|
-- in the constructor.
|
|
border_width = 0,
|
|
|
|
-- These are better left unreplaced in code, because they're used only in one
|
|
-- place and the intent is more clear when the numbers are directly visible.
|
|
-- min_value = 0,
|
|
-- max_value = 1,
|
|
|
|
-- This one is set later. It's not in `prop_defaults`, because I don't
|
|
-- want to make it accessible through the getter, lest the user somehow mutates
|
|
-- the Cairo pattern and breaks NaN indication for all other graphs.
|
|
-- nan_color = make_fallback_nan_color()
|
|
}
|
|
|
|
-- All property defaults are also necessarily fallbacks.
|
|
gtable.crush(prop_fallbacks, prop_defaults, true)
|
|
|
|
-- These fallbacks are themed and can change on the fly.
|
|
setmetatable(prop_fallbacks, {
|
|
__index = function(_, key)
|
|
-- TODO: maybe theme graph.nan_color too?
|
|
if key == "background_color" then
|
|
return beautiful.graph_bg or "#000000aa"
|
|
elseif key == "border_color" then
|
|
return beautiful.graph_border_color or "#ffffff"
|
|
elseif key == "color" then
|
|
return beautiful.graph_fg or "#ff0000"
|
|
end
|
|
end
|
|
})
|
|
|
|
-- This function sets up default implementations for property getters/setters,
|
|
-- when none exist yet. It will be called at the end of this class module.
|
|
local function build_properties(prototype, prop_names)
|
|
for _, prop in ipairs(prop_names) do
|
|
if not prototype["set_" .. prop] then
|
|
prototype["set_" .. prop] = function(self, value)
|
|
if self._private[prop] ~= value then
|
|
self._private[prop] = value
|
|
self:emit_signal("widget::redraw_needed")
|
|
self:emit_signal("property::"..prop, value)
|
|
end
|
|
return self
|
|
end
|
|
end
|
|
if not prototype["get_" .. prop] then
|
|
prototype["get_" .. prop] = function(self)
|
|
return self._private[prop]
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
-- Creates a yellow-black danger stripe @{gears.color} for NaN indication.
|
|
local function build_fallback_nan_color()
|
|
local clr = color.create_pattern_uncached({
|
|
["type"] = "linear",
|
|
from = {0, 0}, to = {4, 4},
|
|
stops={
|
|
{0, "#000000"},
|
|
{0.25, "#000000"}, {0.25, "#ffff00"},
|
|
{0.50, "#ffff00"}, {0.50, "#000000"},
|
|
{0.75, "#000000"}, {0.75, "#ffff00"},
|
|
{1, "#ffff00"},
|
|
},
|
|
})
|
|
clr:set_extend("REPEAT")
|
|
return clr
|
|
end
|
|
|
|
-- Set up the nan_color fallback.
|
|
prop_fallbacks.nan_color = build_fallback_nan_color()
|
|
|
|
--
|
|
-- Module and prototype methods.
|
|
--
|
|
|
|
local function graph_gather_drawn_values_num_stats(self, new_value)
|
|
if not (new_value >= 0) then
|
|
return -- is negative or NaN
|
|
end
|
|
|
|
local last_value = self._private.last_drawn_values_num or 0
|
|
-- Grow instantly and shrink slow
|
|
if new_value < last_value then
|
|
self._private.last_drawn_values_num = last_value - 1
|
|
else
|
|
self._private.last_drawn_values_num = new_value
|
|
end
|
|
end
|
|
|
|
--- Determine the color to paint a data group with.
|
|
--
|
|
-- The graph uses this method to choose a color for a given data group.
|
|
-- The default implementation uses a color from the `group_colors` table,
|
|
-- if present, otherwise it falls back to `color`, then
|
|
-- `beautiful.graph_fg` and finally to color red (#ff0000).
|
|
--
|
|
-- @method pick_data_group_color
|
|
-- @tparam number group_idx The index of the data group.
|
|
-- @treturn gears.color The color to paint the data group's values with.
|
|
function graph:pick_data_group_color(group_idx)
|
|
-- Use an individual group color, if there's one
|
|
local data_group_colors = self._private.group_colors
|
|
local clr = data_group_colors and data_group_colors[group_idx]
|
|
-- Or fall back to some other colors
|
|
return clr or self._private.color or prop_fallbacks.color
|
|
end
|
|
|
|
--- Determine if a data group should be rendered.
|
|
--
|
|
-- The graph uses this method to decide whether the given data group
|
|
-- should get its values rendered.
|
|
--
|
|
-- The default implementation says yes to all data groups, unless
|
|
-- `group_colors` property is set, in which case only those groups are
|
|
-- rendered, for which there are colors in the `group_colors` table,
|
|
-- so one can e.g. disable groups by setting their colors to false.
|
|
--
|
|
-- @method should_draw_data_group
|
|
-- @tparam number group_idx The index of the data group.
|
|
-- @treturn boolean true if the group should be rendered, false otherwise.
|
|
-- @local I'm not confident, that this is good API, so I'm making it private.
|
|
local function graph_should_draw_data_group(self, group_idx)
|
|
-- This default implementation decides, whether a group should be drawn,
|
|
-- based on presence of colors, for reasons of backward compatibility.
|
|
local data_group_colors = self._private.group_colors
|
|
if not data_group_colors then
|
|
-- The colors table isn't set, all data groups are deemed enabled.
|
|
return true
|
|
end
|
|
|
|
-- A group is enabled if it has a color, i.e. nil color means "don't draw it"
|
|
return not not data_group_colors[group_idx]
|
|
end
|
|
|
|
local function graph_preprocess_values(self, values, drawn_values_num)
|
|
-- TODO: elevate to function documentation, if we decide to make it public API.
|
|
--- Preprocesses values before drawing them.
|
|
-- This function can return up to 2 values: drawn_values and scaling_values
|
|
-- The former will be used as values to draw in place of the original data.
|
|
-- The latter will be used as values to scan for min/max value for scaling.
|
|
-- Either can be nil, which means: use values as is.
|
|
|
|
-- This default implementation is only used to implement
|
|
-- presumming values for stacked graphs.
|
|
if not self._private.stack then
|
|
return
|
|
end
|
|
|
|
-- Prepare to draw a stacked graph
|
|
|
|
-- summed_values[i] = sum [1,#values] of values[c][i]
|
|
local summed_values = {}
|
|
-- drawn_values[c][i] = sum [1,c] of values[c][i]
|
|
local drawn_values = {}
|
|
|
|
local nan = 0/0
|
|
|
|
-- Add stacked values up to get values we need to render
|
|
for group_idx, group_values in ipairs(values) do
|
|
local drawn_row = {}
|
|
drawn_values[group_idx] = drawn_row
|
|
|
|
if graph_should_draw_data_group(self, group_idx) then
|
|
for idx, value in ipairs(group_values) do
|
|
if idx > drawn_values_num then
|
|
break
|
|
end
|
|
|
|
-- drawn_values will have NaN values in it due to negatives/NaNs in input.
|
|
-- we can't simply treat them like zeros during rendering,
|
|
-- in case step_shape() draws visible shapes for actual zero values too.
|
|
local acc = summed_values[idx] or 0
|
|
if value >= 0 then
|
|
acc = acc + value
|
|
drawn_row[idx] = acc
|
|
else
|
|
drawn_row[idx] = nan
|
|
end
|
|
summed_values[idx] = acc
|
|
end
|
|
end
|
|
end
|
|
|
|
-- In a stacked graph it's sufficient to examine only the last summed row
|
|
-- to determine the max_value, since all values are necessarily >= 0
|
|
-- and the min_value should be always at most 0
|
|
local scaling_values = { {0}, summed_values }
|
|
|
|
return drawn_values, scaling_values
|
|
end
|
|
|
|
local function graph_map_value_to_widget_coordinates(self, value, min_value, max_value, height)
|
|
-- Scale the value so that [min_value..max_value] maps to [0..1]
|
|
value = (value - min_value) / (max_value - min_value)
|
|
|
|
-- Check whether value is NaN
|
|
if value == value then
|
|
if self._private.clamp_bars then
|
|
-- Don't allow the bar to exceed widget's dimensions
|
|
value = math_min(1, math_max(0, value))
|
|
end
|
|
|
|
-- Drawing bars up from the lower edge of the widget
|
|
return height * (1 - value)
|
|
end
|
|
return value --NaN
|
|
end
|
|
|
|
local function graph_choose_coordinate_system(self, scaling_values, drawn_values_num, height)
|
|
local scale = self._private.scale
|
|
local max_value = self._private.max_value or (scale and -math.huge or 1)
|
|
local min_value = self._private.min_value or (scale and math.huge or 0)
|
|
|
|
if scale then
|
|
for _, group_values in ipairs(scaling_values) do
|
|
for idx, v in ipairs(group_values) do
|
|
-- Do not let off-screen values affect autoscaling
|
|
if idx > drawn_values_num then
|
|
break
|
|
end
|
|
|
|
-- We don't use math.min/max here to be sure that
|
|
-- min/max_value don't accidentally get assigned a NaN
|
|
if v > max_value then
|
|
max_value = v
|
|
end
|
|
if min_value > v then
|
|
min_value = v
|
|
end
|
|
end
|
|
end
|
|
if min_value == max_value then
|
|
-- If all values are equal in an autoscaled graph,
|
|
-- simply draw them in the middle
|
|
min_value, max_value = min_value - 1, max_value + 1
|
|
end
|
|
end
|
|
|
|
-- The position of the baseline in value coordinates
|
|
-- It defaults to the usual zero axis
|
|
local baseline_value = self._private.baseline_value or prop_fallbacks.baseline_value
|
|
|
|
-- Let's map it into widget coordinates
|
|
local main_baseline_y = graph_map_value_to_widget_coordinates(
|
|
self, baseline_value, min_value, max_value, height
|
|
)
|
|
|
|
return min_value, max_value, main_baseline_y
|
|
end
|
|
|
|
local function graph_draw_values(self, cr, width, height, drawn_values_num)
|
|
local step_spacing = self._private.step_spacing or prop_fallbacks.step_spacing
|
|
local step_width = self._private.step_width or prop_fallbacks.step_width
|
|
|
|
-- Cache methods used in the inner loop for a 3x performance boost
|
|
local cairo_rectangle = cr.rectangle
|
|
local map_coords = graph_map_value_to_widget_coordinates
|
|
|
|
local drawn_values, scaling_values = graph_preprocess_values(self, self._private.values, drawn_values_num)
|
|
-- If preprocessor returned drawn_values = nil, then simply draw the values we have
|
|
drawn_values = drawn_values or self._private.values
|
|
-- If preprocessor returned scaling_values = nil, then
|
|
-- all drawn values need to be examined to determine proper scaling
|
|
scaling_values = scaling_values or drawn_values
|
|
|
|
local min_value, max_value, main_baseline_y = graph_choose_coordinate_system(
|
|
self, scaling_values, drawn_values_num, height
|
|
)
|
|
|
|
--- A bag of potentially useful things passed into user's draw callbacks.
|
|
--
|
|
-- An fresh instance of this table is created each time
|
|
-- awesome asks the widget to redraw itself.
|
|
--
|
|
-- The user can also add their own values to this table to conveniently
|
|
-- pass data pertaining to the same drawing session between callbacks, but
|
|
-- all underscore-prefixed keys are reserved for future use by the widget.
|
|
--
|
|
-- @within Advanced drawing fields
|
|
-- @table draw_callback_options
|
|
-- @tfield wibox.widget.graph graph The graph widget itself.
|
|
-- @tfield number width The width it is being drawn with.
|
|
-- @tfield number height The height it is being drawn with.
|
|
-- @tfield number|nil group_idx Index of the currently drawn data group.
|
|
local options = setmetatable({}, { __index = {
|
|
graph = self,
|
|
width = width,
|
|
height = height,
|
|
group_idx = nil,
|
|
}})
|
|
|
|
-- The user callback to call before drawing each data group
|
|
local group_start = self._private.group_start
|
|
if not group_start or type(group_start) == "number" then
|
|
local offset_x = group_start
|
|
group_start = function(cr, group_idx) --luacheck: ignore 431 432
|
|
-- Set the data series' color early, in case the user
|
|
-- wants to do their own painting inside step_shape()
|
|
cr:set_source(color(self:pick_data_group_color(group_idx)))
|
|
-- Pass the user-given constant group offset, if any
|
|
return offset_x
|
|
end
|
|
end
|
|
|
|
-- The user callback to call after drawing each data group
|
|
local group_finish = self._private.group_finish
|
|
if not group_finish then
|
|
group_finish = function(cr, _group_idx, _options) --luacheck: ignore 432 212/_.*
|
|
cr:fill()
|
|
end
|
|
end
|
|
|
|
-- The user callback for drawing each data bar
|
|
local step_hook = self._private.step_hook
|
|
if not step_hook then
|
|
local step_shape = self._private.step_shape
|
|
if step_shape then
|
|
-- Preserve the transform centered at the top-left corner of the graph
|
|
local pristine_transform = cr:get_matrix()
|
|
local cairo_translate = cr.translate
|
|
local cairo_set_matrix = cr.set_matrix
|
|
step_hook = function(_, x, value_y, baseline_y, _step_width)
|
|
local step_height = baseline_y - value_y
|
|
if not (step_height == step_height) then
|
|
return -- is NaN
|
|
end
|
|
-- Shift to the bar beginning
|
|
cairo_translate(cr, x, value_y)
|
|
step_shape(cr, _step_width, step_height)
|
|
-- Undo the shift
|
|
cairo_set_matrix(cr, pristine_transform)
|
|
end
|
|
end
|
|
end
|
|
|
|
local nan_x = self._private.nan_indication and {}
|
|
local prev_y = self._private.stack and {}
|
|
|
|
for group_idx, group_values in ipairs(drawn_values) do
|
|
if graph_should_draw_data_group(self, group_idx) then
|
|
options.group_idx = group_idx
|
|
-- group_start() callback prepares context for drawing a data group.
|
|
-- It can give us a horizontal offset for all data group bars.
|
|
local offset_x = group_start(cr, group_idx, options)
|
|
offset_x = offset_x or 0
|
|
|
|
for i = 1, math_min(#group_values, drawn_values_num) do
|
|
local value = group_values[i]
|
|
|
|
local value_y = map_coords(self, value, min_value, max_value, height)
|
|
local not_nan = value_y == value_y
|
|
|
|
-- The coordinate of the i-th bar's left edge
|
|
local x = (i-1)*(step_width + step_spacing) + offset_x
|
|
|
|
local base_y = main_baseline_y
|
|
if prev_y then
|
|
-- Draw from where the previous stacked series left off
|
|
base_y = prev_y[i] or base_y
|
|
-- Save our y for the next stacked series
|
|
if not_nan then
|
|
prev_y[i] = value_y
|
|
end
|
|
end
|
|
|
|
if step_hook then
|
|
-- step_hook() is expected to handle NaNs itself
|
|
step_hook(cr, x, value_y, base_y, step_width, options)
|
|
elseif not_nan then
|
|
cairo_rectangle(cr, x, value_y, step_width, base_y - value_y)
|
|
end
|
|
|
|
if not not_nan and nan_x then
|
|
-- Keep the coordinate to draw NaN indication later
|
|
table.insert(nan_x, x)
|
|
end
|
|
end
|
|
|
|
-- group_finish() callback does what is needed to paint the data group
|
|
group_finish(cr, group_idx, options)
|
|
end
|
|
end
|
|
|
|
if nan_x and #nan_x > 0 then
|
|
cr:set_source(color(self._private.nan_color or prop_fallbacks.nan_color))
|
|
for _, x in ipairs(nan_x) do
|
|
-- Draw full-height rectangle with nan_color to indicate NaN
|
|
cairo_rectangle(cr, x, 0, step_width, height)
|
|
end
|
|
cr:fill()
|
|
end
|
|
end
|
|
|
|
function graph:draw(_, cr, width, height)
|
|
local border_width = self._private.border_width or prop_fallbacks.border_width
|
|
local drawn_values_num = self:compute_drawn_values_num(width-2*border_width)
|
|
|
|
-- Track our usage to help us guess the necessary values array capacity
|
|
graph_gather_drawn_values_num_stats(self, drawn_values_num)
|
|
|
|
-- Draw the background first
|
|
cr:set_source(color(self._private.background_color or prop_fallbacks.background_color))
|
|
cr:paint()
|
|
|
|
-- Draw the values
|
|
if drawn_values_num > 0 then
|
|
cr:save()
|
|
|
|
-- Account for the border width
|
|
if border_width > 0 then
|
|
cr:translate(border_width, border_width)
|
|
end
|
|
|
|
local values_width = width - 2*border_width
|
|
local values_height = height - 2*border_width
|
|
|
|
graph_draw_values(self, cr, values_width, values_height, drawn_values_num)
|
|
|
|
-- Undo the cr:translate() for the border and step shapes
|
|
cr:restore()
|
|
end
|
|
|
|
-- Draw the border last so that it overlaps already drawn values
|
|
if border_width > 0 then
|
|
cr:set_line_width(border_width)
|
|
cr:rectangle(border_width/2, border_width/2, width - border_width, height - border_width)
|
|
cr:set_source(color(self._private.border_color or prop_fallbacks.border_color))
|
|
cr:stroke()
|
|
end
|
|
end
|
|
|
|
function graph:fit(_, width, height)
|
|
return width, height
|
|
end
|
|
|
|
--- Determine how many values should be drawn for a given widget width.
|
|
--
|
|
-- The graph uses this method to determine the upper bound on the
|
|
-- number of values that will be drawn from each data group. This affects,
|
|
-- among other things, how many values will be considered for autoscaling,
|
|
-- when `scale` is true, and, indirectly, how many values will be kept in
|
|
-- the backing array, when `capacity` is unset.
|
|
--
|
|
-- The default implementation computes the minimum number that is enough
|
|
-- to completely cover the given width with `step_width` + `step_spacing`
|
|
-- intervals. The graph calls this method on every redraw and the width
|
|
-- passed is the width of the value drawing area, i.e the graph borders
|
|
-- are subtracted (2\*`border_width`).
|
|
--
|
|
-- @method compute_drawn_values_num
|
|
-- @tparam number usable_width
|
|
function graph:compute_drawn_values_num(usable_width)
|
|
if usable_width <= 0 then
|
|
return 0
|
|
end
|
|
local step_width = self._private.step_width or prop_fallbacks.step_width
|
|
local step_spacing = self._private.step_spacing or prop_fallbacks.step_spacing
|
|
return math.ceil(usable_width / (step_width + step_spacing))
|
|
end
|
|
|
|
local function guess_capacity(self)
|
|
local capacity = self._private.capacity
|
|
if capacity then
|
|
-- Ensure it's integer, no matter what the user sets.
|
|
return math.ceil(capacity)
|
|
end
|
|
|
|
local ldwn = self._private.last_drawn_values_num
|
|
if not ldwn then
|
|
-- We haven't been drawn even once yet,
|
|
-- maybe the user will push a ton of values now.
|
|
-- Our widget is 8K-display-ready.
|
|
return 8192
|
|
end
|
|
|
|
-- Calculate an appropriate capacity from drawn values num
|
|
-- with some wiggle room for widget resizes
|
|
return math.ceil(ldwn/64 + 1)*64
|
|
end
|
|
|
|
--- Add a value to the graph.
|
|
--
|
|
-- The graph widget keeps its values grouped in _data groups_. Each data group
|
|
-- is drawn with its own set of bars, starting with the latest value
|
|
-- in the data group at the left edge of the graph.
|
|
--
|
|
-- Simply calling this method with a particular data group index is the only
|
|
-- thing necessary and sufficient for creating a data group.
|
|
-- Any natural integer as a group number is ok, but the user is advised to keep
|
|
-- the group numbers low and consecutive for performance reasons.
|
|
--
|
|
-- There are no constraints on the value parameter, other than it should
|
|
-- be a number.
|
|
--
|
|
-- @method add_value
|
|
-- @tparam[opt=NaN] number value The value to be added to a graph's data group.
|
|
-- @tparam[opt=1] integer group The index of the data group.
|
|
function graph:add_value(value, group)
|
|
value = value or 0/0 -- default to NaN
|
|
group = group or 1
|
|
|
|
local values = self._private.values
|
|
if not values[group] then
|
|
-- Ensure that there are no gaps in the values array,
|
|
-- so that ipairs() can reach all data groups.
|
|
for i = #values+1, group do
|
|
values[i] = {}
|
|
end
|
|
-- If the above loop hasn't set it, then
|
|
-- `group` wasn't a non-negative integer.
|
|
if not values[group] then
|
|
error("Invalid data group index: " .. tostring(group))
|
|
end
|
|
end
|
|
values = values[group]
|
|
|
|
local capacity = guess_capacity(self)
|
|
-- Map negatives, NaNs and zero to nil
|
|
capacity = (capacity >= 1) and capacity
|
|
|
|
-- Remove old values over capacity
|
|
-- Invalid capacity means "remove everything"
|
|
local i = capacity or 1
|
|
while values[i] do
|
|
values[i] = nil
|
|
i = i + 1
|
|
end
|
|
|
|
if capacity then
|
|
table.insert(values, 1, value)
|
|
end
|
|
|
|
self:emit_signal("widget::redraw_needed")
|
|
return self
|
|
end
|
|
|
|
--- Clear the graph.
|
|
--
|
|
-- Removes all values from all data groups.
|
|
--
|
|
-- @method clear
|
|
function graph:clear()
|
|
self._private.values = {}
|
|
self:emit_signal("widget::redraw_needed")
|
|
return self
|
|
end
|
|
|
|
--- Set the graph capacity.
|
|
--
|
|
-- Since the typical uses of the graph widget imply that `add_value` will be
|
|
-- called an indefinite number of times, the widget needs a way to know, when
|
|
-- to start discarding old values from the backing array.
|
|
--
|
|
-- When `capacity` is set, it defines the maximum number of values to keep in
|
|
-- each data group.
|
|
--
|
|
-- When `capacity` is unset (default), the number is determined heuristically,
|
|
-- which is sufficient most of the time, unless the widget gets resized
|
|
-- too much too fast.
|
|
--
|
|
-- @property capacity
|
|
-- @tparam[opt=nil] integer|nil capacity The maximum number of values to keep
|
|
-- per data group (`nil` for automatic guess).
|
|
-- @propemits true false
|
|
function graph:set_capacity(capacity)
|
|
-- Property override to avoid emitting the "redraw_needed" signal,
|
|
-- because nothing visibly changes until the next add_value() call,
|
|
-- which emits the signal itself.
|
|
-- It might have been prudent to truncate the values array here
|
|
-- and emit the signal, but I don't think anyone really needs that.
|
|
if self._private.capacity ~= capacity then
|
|
self._private.capacity = capacity
|
|
self:emit_signal("property::capacity", capacity)
|
|
end
|
|
return self
|
|
end
|
|
|
|
--- Set the graph height.
|
|
--
|
|
-- This property is deprecated. Use a `wibox.container.constraint` widget or
|
|
-- `forced_height`.
|
|
---
|
|
-- @deprecatedproperty height
|
|
-- @tparam number height The height to set.
|
|
-- @renamedin 5.0 forced_height
|
|
-- @propemits true false
|
|
function graph:set_height(height)
|
|
gdebug.deprecate("Use a `wibox.container.constraint` widget or `forced_height`", {deprecated_in=5})
|
|
if awesome.api_level <= 5 then
|
|
if height >= 5 then
|
|
-- this sends "layout_changed" for us
|
|
self:set_forced_height(height)
|
|
-- signal, because we did it before
|
|
self:emit_signal("property::height", height)
|
|
end
|
|
return self
|
|
end
|
|
end
|
|
|
|
function graph:get_height()
|
|
gdebug.deprecate("Use `forced_height`", {deprecated_in=5})
|
|
return awesome.api_level <= 5 and self._private.forced_height or nil
|
|
end
|
|
|
|
--- Set the graph width.
|
|
--
|
|
-- This property is deprecated. Use a `wibox.container.constraint` widget or
|
|
-- `forced_width`.
|
|
---
|
|
-- @deprecatedproperty width
|
|
-- @tparam number width The width to set.
|
|
-- @renamedin 5.0 forced_width
|
|
-- @propemits true false
|
|
function graph:set_width(width)
|
|
gdebug.deprecate("Use a `wibox.container.constraint` widget or `forced_width`", {deprecated_in=5})
|
|
if awesome.api_level <= 5 then
|
|
if width >= 5 then
|
|
-- this sends "layout_changed" for us
|
|
self:set_forced_width(width)
|
|
-- signal, because we did it before
|
|
self:emit_signal("property::width", width)
|
|
end
|
|
return self
|
|
end
|
|
end
|
|
|
|
function graph:get_width()
|
|
gdebug.deprecate("Use `forced_width`", {deprecated_in=5})
|
|
return awesome.api_level <= 5 and self._private.forced_width or nil
|
|
end
|
|
|
|
--- Set the colors for data groups.
|
|
--
|
|
-- This property is deprecated. Use `group_colors` instead.
|
|
---
|
|
-- @deprecatedproperty stack_colors
|
|
-- @renamedin 5.0 group_colors
|
|
-- @tparam table colors A table with colors for data groups.
|
|
-- @see group_colors
|
|
function graph:set_stack_colors(colors)
|
|
gdebug.deprecate("Use `group_colors`", {deprecated_in=5})
|
|
if awesome.api_level <= 5 then
|
|
if self._private.group_colors ~= colors then
|
|
-- this sends "redraw_needed" for us
|
|
self:set_group_colors(colors)
|
|
-- signal, because we did it before
|
|
self:emit_signal("property::stack_colors", colors)
|
|
end
|
|
return self
|
|
end
|
|
end
|
|
|
|
function graph:get_stack_colors()
|
|
gdebug.deprecate("Use `group_colors`", {deprecated_in=5})
|
|
return awesome.api_level <= 5 and self._private.group_colors or nil
|
|
end
|
|
|
|
|
|
--- Create a graph widget.
|
|
--
|
|
-- @tparam table args Standard widget() arguments.
|
|
-- @treturn wibox.widget.graph A new graph widget.
|
|
-- @constructorfct wibox.widget.graph
|
|
function graph.new(args)
|
|
args = args or {}
|
|
|
|
local _graph = base.make_widget(nil, nil, {enable_properties = true})
|
|
|
|
if args.width or args.height then
|
|
gdebug.deprecate(
|
|
"`args.width` and `args.height` are deprecated. "..
|
|
"Use a `wibox.container.constraint` widget "..
|
|
"or `forced_width`/`forced_height`",
|
|
{deprecated_in=5, raw=true}
|
|
)
|
|
end
|
|
|
|
if awesome.api_level <= 5 then
|
|
local width = args.width or 100
|
|
local height = args.height or 20
|
|
|
|
if width < 5 or height < 5 then return end
|
|
|
|
_graph._private.forced_width = width
|
|
_graph._private.forced_height = height
|
|
end
|
|
|
|
-- Set initial values for properties.
|
|
gtable.crush(_graph._private, prop_defaults, true)
|
|
_graph._private.values = {}
|
|
-- Copy methods and properties over
|
|
gtable.crush(_graph, graph, true)
|
|
-- Except those, which don't belong in the widget instance
|
|
rawset(_graph, "new", nil)
|
|
rawset(_graph, "mt", nil)
|
|
|
|
return _graph
|
|
end
|
|
|
|
function graph.mt:__call(...)
|
|
return graph.new(...)
|
|
end
|
|
|
|
-- Setup default impls for property accessors that haven't been implemented explicitly.
|
|
build_properties(graph, properties)
|
|
|
|
return setmetatable(graph, graph.mt)
|
|
|
|
-- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80
|