170 lines
5.5 KiB
Lua
170 lines
5.5 KiB
Lua
---------------------------------------------------------------------------
|
|
--- Display the time (and date) in a text box.
|
|
--
|
|
-- The `wibox.widget.textclock` widget is part of the Awesome WM's widget
|
|
-- system (see @{03-declarative-layout.md}).
|
|
--
|
|
-- This widget displays a text clock formatted by the
|
|
-- [GLib Date Time format](https://developer.gnome.org/glib/stable/glib-GDateTime.html#g-date-time-format)
|
|
-- and [GTimeZone](https://developer.gnome.org/glib/stable/glib-GTimeZone.html#g-time-zone-new).
|
|
--
|
|
-- The `wibox.widget.textclock` inherits from `wibox.widget.textbox`. It means
|
|
-- that, once created, the user will receive a derivated instance of
|
|
-- `wibox.widget.textbox` associated with a private `gears.timer` to manage
|
|
-- timed updates of the displayed clock.
|
|
--
|
|
-- Use a `wibox.widget.textclock`
|
|
-- ---
|
|
--
|
|
-- @DOC_wibox_widget_defaults_textclock_EXAMPLE@
|
|
--
|
|
-- Alternatively, you can declare the `textclock` widget using the
|
|
-- declarative pattern (Both codes are strictly equivalent):
|
|
--
|
|
-- @DOC_wibox_widget_declarative-pattern_textclock_EXAMPLE@
|
|
--
|
|
-- The GLib DateTime format
|
|
-- ---
|
|
--
|
|
-- The time displayed by the textclock widget can be formated by the GLib
|
|
-- DateTime format.
|
|
--
|
|
-- Here is a short list with commonly used format specifiers (extracted from
|
|
-- the Glib API references):
|
|
--
|
|
--@DOC_glib_timedate_format_COMMON@
|
|
--
|
|
-- You can read more on the GLib DateTime format in the
|
|
-- [GLib documentation](https://developer.gnome.org/glib/stable/glib-GDateTime.html#g-date-time-format).
|
|
--
|
|
-- @author Julien Danjou <julien@danjou.info>
|
|
-- @copyright 2009 Julien Danjou
|
|
-- @widgetmod wibox.widget.textclock
|
|
-- @supermodule wibox.widget.textbox
|
|
---------------------------------------------------------------------------
|
|
|
|
local setmetatable = setmetatable
|
|
local os = os
|
|
local textbox = require("wibox.widget.textbox")
|
|
local timer = require("gears.timer")
|
|
local gtable = require("gears.table")
|
|
local glib = require("lgi").GLib
|
|
local DateTime = glib.DateTime
|
|
local TimeZone = glib.TimeZone
|
|
|
|
local textclock = { mt = {} }
|
|
|
|
local DateTime_new_now = DateTime.new_now
|
|
|
|
-- When $SOURCE_DATE_EPOCH and $SOURCE_DIRECTORY are both set, then this code is
|
|
-- most likely being run by the test runner. Ensure reproducible dates.
|
|
local source_date_epoch = tonumber(os.getenv("SOURCE_DATE_EPOCH"))
|
|
if source_date_epoch and os.getenv("SOURCE_DIRECTORY") then
|
|
DateTime_new_now = function()
|
|
return DateTime.new_from_unix_utc(source_date_epoch)
|
|
end
|
|
end
|
|
|
|
--- Set the clock's format.
|
|
--
|
|
-- For information about the format specifiers, see
|
|
-- [the GLib docs](https://developer.gnome.org/glib/stable/glib-GDateTime.html#g-date-time-format).
|
|
-- @property format
|
|
-- @tparam string format The new time format. This can contain pango markup.
|
|
|
|
function textclock:set_format(format)
|
|
self._private.format = format
|
|
self:force_update()
|
|
end
|
|
|
|
function textclock:get_format()
|
|
return self._private.format
|
|
end
|
|
|
|
--- Set the clock's timezone.
|
|
--
|
|
-- e.g. "Z" for UTC, "±hh:mm" or "Europe/Amsterdam". See
|
|
-- [GTimeZone](https://developer.gnome.org/glib/stable/glib-GTimeZone.html#g-time-zone-new).
|
|
-- @property timezone
|
|
-- @tparam string timezone
|
|
|
|
function textclock:set_timezone(tzid)
|
|
self._private.tzid = tzid
|
|
self._private.timezone = tzid and TimeZone.new(tzid)
|
|
self:force_update()
|
|
end
|
|
|
|
function textclock:get_timezone()
|
|
return self._private.tzid
|
|
end
|
|
|
|
--- Set the clock's refresh rate.
|
|
--
|
|
-- @property refresh
|
|
-- @tparam number refresh How often the clock is updated, in seconds
|
|
|
|
function textclock:set_refresh(refresh)
|
|
self._private.refresh = refresh or self._private.refresh
|
|
self:force_update()
|
|
end
|
|
|
|
function textclock:get_refresh()
|
|
return self._private.refresh
|
|
end
|
|
|
|
--- Force a textclock to update now.
|
|
--
|
|
-- @noreturn
|
|
-- @method force_update
|
|
function textclock:force_update()
|
|
self._timer:emit_signal("timeout")
|
|
end
|
|
|
|
--- This lowers the timeout so that it occurs "correctly". For example, a timeout
|
|
-- of 60 is rounded so that it occurs the next time the clock reads ":00 seconds".
|
|
local function calc_timeout(real_timeout)
|
|
return real_timeout - os.time() % real_timeout
|
|
end
|
|
|
|
--- Create a textclock widget. It draws the time it is in a textbox.
|
|
--
|
|
-- @tparam[opt=" %a %b %d, %H:%M "] string format The time [format](#format).
|
|
-- @tparam[opt=60] number refresh How often to update the time (in seconds).
|
|
-- @tparam[opt=local timezone] string timezone The [timezone](#timezone) to use.
|
|
-- @treturn table A textbox widget.
|
|
-- @constructorfct wibox.widget.textclock
|
|
local function new(format, refresh, tzid)
|
|
local w = textbox()
|
|
gtable.crush(w, textclock, true)
|
|
|
|
w._private.format = format or " %a %b %d, %H:%M "
|
|
w._private.refresh = refresh or 60
|
|
w._private.tzid = tzid
|
|
w._private.timezone = tzid and TimeZone.new(tzid)
|
|
|
|
function w._private.textclock_update_cb()
|
|
local str = DateTime_new_now(w._private.timezone or TimeZone.new_local()):format(w._private.format)
|
|
if str == nil then
|
|
require("gears.debug").print_warning("textclock: "
|
|
.. "g_date_time_format() failed for format "
|
|
.. "'" .. w._private.format .. "'")
|
|
end
|
|
w:set_markup(str)
|
|
w._timer.timeout = calc_timeout(w._private.refresh)
|
|
w._timer:again()
|
|
return true -- Continue the timer
|
|
end
|
|
|
|
w._timer = timer.weak_start_new(refresh, w._private.textclock_update_cb)
|
|
w:force_update()
|
|
return w
|
|
end
|
|
|
|
function textclock.mt:__call(...)
|
|
return new(...)
|
|
end
|
|
|
|
return setmetatable(textclock, textclock.mt)
|
|
|
|
-- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80
|