2016-05-29 18:01:40 +02:00
|
|
|
--[[
|
2017-09-04 12:43:00 +02:00
|
|
|
|
|
|
|
Licensed under GNU General Public License v2
|
2017-09-11 06:51:52 +02:00
|
|
|
* (c) 2013, Luca CPZ
|
2017-09-04 12:43:00 +02:00
|
|
|
* (c) 2013, Rman
|
|
|
|
|
2016-05-29 18:01:40 +02:00
|
|
|
--]]
|
|
|
|
|
2023-09-05 22:29:10 +02:00
|
|
|
local requirePrefix = tostring(...):match(".*lain") or ""
|
|
|
|
if requirePrefix then
|
|
|
|
requirePrefix = requirePrefix .. "."
|
|
|
|
end
|
|
|
|
|
|
|
|
local helpers = require(requirePrefix .. "helpers")
|
2018-11-28 20:43:04 +01:00
|
|
|
local awful = require("awful")
|
|
|
|
local naughty = require("naughty")
|
|
|
|
local wibox = require("wibox")
|
|
|
|
local math = math
|
|
|
|
local string = string
|
|
|
|
local type = type
|
|
|
|
local tonumber = tonumber
|
2016-05-29 18:01:40 +02:00
|
|
|
|
2017-09-13 12:33:40 +02:00
|
|
|
-- PulseAudio volume bar
|
2017-02-08 14:15:48 +01:00
|
|
|
-- lain.widget.pulsebar
|
2016-05-29 18:01:40 +02:00
|
|
|
|
2017-02-08 20:45:11 +01:00
|
|
|
local function factory(args)
|
2017-04-02 19:35:03 +02:00
|
|
|
local pulsebar = {
|
|
|
|
colors = {
|
2019-10-13 21:52:27 +02:00
|
|
|
background = "#000000",
|
|
|
|
mute_background = "#000000",
|
|
|
|
mute = "#EB8F8F",
|
|
|
|
unmute = "#A4CE8A"
|
2017-04-02 19:35:03 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
_current_level = 0,
|
2017-06-25 13:11:30 +02:00
|
|
|
_mute = "no",
|
2017-09-13 12:33:40 +02:00
|
|
|
device = "N/A"
|
2017-04-02 19:35:03 +02:00
|
|
|
}
|
|
|
|
|
2020-11-29 22:48:32 +01:00
|
|
|
args = args or {}
|
|
|
|
|
2016-05-29 18:01:40 +02:00
|
|
|
local timeout = args.timeout or 5
|
|
|
|
local settings = args.settings or function() end
|
|
|
|
local width = args.width or 63
|
2018-10-28 13:59:00 +01:00
|
|
|
local height = args.height or 1
|
|
|
|
local margins = args.margins or 1
|
|
|
|
local paddings = args.paddings or 1
|
2016-05-29 18:01:40 +02:00
|
|
|
local ticks = args.ticks or false
|
|
|
|
local ticks_size = args.ticks_size or 7
|
2019-08-10 19:27:17 +02:00
|
|
|
local tick = args.tick or "|"
|
|
|
|
local tick_pre = args.tick_pre or "["
|
|
|
|
local tick_post = args.tick_post or "]"
|
2019-08-10 19:34:15 +02:00
|
|
|
local tick_none = args.tick_none or " "
|
2016-05-29 18:01:40 +02:00
|
|
|
|
2017-06-25 13:11:30 +02:00
|
|
|
pulsebar.colors = args.colors or pulsebar.colors
|
|
|
|
pulsebar.followtag = args.followtag or false
|
|
|
|
pulsebar.notification_preset = args.notification_preset
|
2017-09-13 03:46:43 +02:00
|
|
|
pulsebar.devicetype = args.devicetype or "sink"
|
|
|
|
pulsebar.cmd = args.cmd or "pacmd list-" .. pulsebar.devicetype .. "s | sed -n -e '/*/,$!d' -e '/index/p' -e '/base volume/d' -e '/volume:/p' -e '/muted:/p' -e '/device\\.string/p'"
|
2017-01-23 22:27:37 +01:00
|
|
|
|
|
|
|
if not pulsebar.notification_preset then
|
2017-09-13 12:33:40 +02:00
|
|
|
pulsebar.notification_preset = {
|
|
|
|
font = "Monospace 10"
|
|
|
|
}
|
2017-01-23 22:27:37 +01:00
|
|
|
end
|
2016-05-29 18:01:40 +02:00
|
|
|
|
2017-01-05 13:53:47 +01:00
|
|
|
pulsebar.bar = wibox.widget {
|
|
|
|
color = pulsebar.colors.unmute,
|
|
|
|
background_color = pulsebar.colors.background,
|
2018-10-28 13:59:00 +01:00
|
|
|
forced_height = height,
|
|
|
|
forced_width = width,
|
|
|
|
margins = margins,
|
2018-11-04 20:29:12 +01:00
|
|
|
paddings = paddings,
|
2017-01-05 13:53:47 +01:00
|
|
|
ticks = ticks,
|
|
|
|
ticks_size = ticks_size,
|
|
|
|
widget = wibox.widget.progressbar,
|
|
|
|
}
|
2016-05-29 18:01:40 +02:00
|
|
|
|
|
|
|
pulsebar.tooltip = awful.tooltip({ objects = { pulsebar.bar } })
|
|
|
|
|
2017-01-23 22:27:37 +01:00
|
|
|
function pulsebar.update(callback)
|
2017-09-13 12:33:40 +02:00
|
|
|
helpers.async({ awful.util.shell, "-c", type(pulsebar.cmd) == "string" and pulsebar.cmd or pulsebar.cmd() },
|
|
|
|
function(s)
|
2017-01-23 22:27:37 +01:00
|
|
|
volume_now = {
|
2017-09-13 12:33:40 +02:00
|
|
|
index = string.match(s, "index: (%S+)") or "N/A",
|
2017-09-13 03:46:43 +02:00
|
|
|
device = string.match(s, "device.string = \"(%S+)\"") or "N/A",
|
2017-09-13 12:33:40 +02:00
|
|
|
muted = string.match(s, "muted: (%S+)") or "N/A"
|
2017-01-23 22:27:37 +01:00
|
|
|
}
|
|
|
|
|
2017-02-28 10:55:48 +01:00
|
|
|
pulsebar.device = volume_now.index
|
2017-02-24 13:48:13 +01:00
|
|
|
|
2017-01-23 22:27:37 +01:00
|
|
|
local ch = 1
|
|
|
|
volume_now.channel = {}
|
|
|
|
for v in string.gmatch(s, ":.-(%d+)%%") do
|
|
|
|
volume_now.channel[ch] = v
|
|
|
|
ch = ch + 1
|
|
|
|
end
|
|
|
|
|
|
|
|
volume_now.left = volume_now.channel[1] or "N/A"
|
|
|
|
volume_now.right = volume_now.channel[2] or "N/A"
|
|
|
|
|
|
|
|
local volu = volume_now.left
|
|
|
|
local mute = volume_now.muted
|
|
|
|
|
2017-06-25 13:11:30 +02:00
|
|
|
if volu:match("N/A") or mute:match("N/A") then return end
|
|
|
|
|
|
|
|
if volu ~= pulsebar._current_level or mute ~= pulsebar._mute then
|
|
|
|
pulsebar._current_level = tonumber(volu)
|
2017-01-23 22:27:37 +01:00
|
|
|
pulsebar.bar:set_value(pulsebar._current_level / 100)
|
2017-06-25 13:11:30 +02:00
|
|
|
if pulsebar._current_level == 0 or mute == "yes" then
|
|
|
|
pulsebar._mute = mute
|
2017-09-13 12:33:40 +02:00
|
|
|
pulsebar.tooltip:set_text ("[muted]")
|
2017-01-23 22:27:37 +01:00
|
|
|
pulsebar.bar.color = pulsebar.colors.mute
|
2019-10-13 21:52:27 +02:00
|
|
|
pulsebar.bar.background_color = pulsebar.colors.mute_background
|
2017-01-23 22:27:37 +01:00
|
|
|
else
|
2017-06-25 13:11:30 +02:00
|
|
|
pulsebar._mute = "no"
|
2017-09-13 12:33:40 +02:00
|
|
|
pulsebar.tooltip:set_text(string.format("%s %s: %s", pulsebar.devicetype, pulsebar.device, volu))
|
2017-01-23 22:27:37 +01:00
|
|
|
pulsebar.bar.color = pulsebar.colors.unmute
|
2019-10-13 21:52:27 +02:00
|
|
|
pulsebar.bar.background_color = pulsebar.colors.background
|
2017-01-23 22:27:37 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
settings()
|
|
|
|
|
|
|
|
if type(callback) == "function" then callback() end
|
|
|
|
end
|
|
|
|
end)
|
|
|
|
end
|
|
|
|
|
|
|
|
function pulsebar.notify()
|
|
|
|
pulsebar.update(function()
|
|
|
|
local preset = pulsebar.notification_preset
|
2017-09-04 12:43:00 +02:00
|
|
|
|
2017-09-13 12:33:40 +02:00
|
|
|
preset.title = string.format("%s %s - %s%%", pulsebar.devicetype, pulsebar.device, pulsebar._current_level)
|
2017-01-23 22:27:37 +01:00
|
|
|
|
2017-06-25 13:11:30 +02:00
|
|
|
if pulsebar._mute == "yes" then
|
2017-09-13 12:33:40 +02:00
|
|
|
preset.title = preset.title .. " muted"
|
2016-05-29 18:01:40 +02:00
|
|
|
end
|
2017-01-23 22:27:37 +01:00
|
|
|
|
2018-02-17 14:28:31 +01:00
|
|
|
-- tot is the maximum number of ticks to display in the notification
|
|
|
|
-- fallback: default horizontal wibox height
|
|
|
|
local wib, tot = awful.screen.focused().mywibox, 20
|
|
|
|
|
|
|
|
-- if we can grab mywibox, tot is defined as its height if
|
|
|
|
-- horizontal, or width otherwise
|
|
|
|
if wib then
|
|
|
|
if wib.position == "left" or wib.position == "right" then
|
|
|
|
tot = wib.width
|
|
|
|
else
|
|
|
|
tot = wib.height
|
|
|
|
end
|
2018-02-16 19:49:13 +01:00
|
|
|
end
|
|
|
|
|
2020-11-29 22:48:32 +01:00
|
|
|
local int = math.modf((pulsebar._current_level / 100) * tot)
|
2019-08-10 19:27:17 +02:00
|
|
|
preset.text = string.format(
|
|
|
|
"%s%s%s%s",
|
|
|
|
tick_pre,
|
|
|
|
string.rep(tick, int),
|
2019-08-10 19:34:15 +02:00
|
|
|
string.rep(tick_none, tot - int),
|
2019-08-10 19:27:17 +02:00
|
|
|
tick_post
|
|
|
|
)
|
2017-01-23 22:27:37 +01:00
|
|
|
|
|
|
|
if pulsebar.followtag then preset.screen = awful.screen.focused() end
|
|
|
|
|
2017-01-25 18:35:30 +01:00
|
|
|
if not pulsebar.notification then
|
|
|
|
pulsebar.notification = naughty.notify {
|
|
|
|
preset = preset,
|
|
|
|
destroy = function() pulsebar.notification = nil end
|
|
|
|
}
|
|
|
|
else
|
|
|
|
naughty.replace_text(pulsebar.notification, preset.title, preset.text)
|
|
|
|
end
|
2017-01-23 22:27:37 +01:00
|
|
|
end)
|
2016-05-29 18:01:40 +02:00
|
|
|
end
|
|
|
|
|
2019-06-28 09:24:01 +02:00
|
|
|
helpers.newtimer(string.format("pulsebar-%s-%s", pulsebar.devicetype, pulsebar.device), timeout, pulsebar.update)
|
2016-05-29 18:01:40 +02:00
|
|
|
|
|
|
|
return pulsebar
|
|
|
|
end
|
|
|
|
|
2017-04-03 00:53:31 +02:00
|
|
|
return factory
|