------------------------------------------------- -- The Ultimate Volume Widget for Awesome Window Manager -- More details could be found here: -- https://github.com/streetturtle/awesome-wm-widgets/tree/master/volume-widget -- @author Pavel Makhov -- @copyright 2020 Pavel Makhov ------------------------------------------------- local awful = require("awful") local wibox = require("wibox") local spawn = require("awful.spawn") local gears = require("gears") local beautiful = require("beautiful") local watch = require("awful.widget.watch") local utils = require("awesome-wm-widgets.volume-widget.utils") local LIST_DEVICES_CMD = [[sh -c "pacmd list-sinks; pacmd list-sources"]] local function GET_VOLUME_CMD(card, device, mixctrl, value_type) return "amixer -c " .. card .. " -D " .. device .. " sget " .. mixctrl .. " " .. value_type end local function INC_VOLUME_CMD(card, device, mixctrl, value_type, step) return "amixer -c " .. card .. " -D " .. device .. " sset " .. mixctrl .. " " .. value_type .. " " .. step .. "%+" end -- luacheck: ignore local function DEC_VOLUME_CMD(card, device, mixctrl, value_type, step) return "amixer -c " .. card .. " -D " .. device .. " sset " .. mixctrl .. " " .. value_type .. " " .. step .. "%-" end -- luacheck: ignore local function TOG_VOLUME_CMD(card, device, mixctrl) return "amixer -c " .. card .. " -D " .. device .. " sset " .. mixctrl .. " toggle" end -- luacheck: ignore local widget_types = { icon_and_text = require("awesome-wm-widgets.volume-widget.widgets.icon-and-text-widget"), icon = require("awesome-wm-widgets.volume-widget.widgets.icon-widget"), arc = require("awesome-wm-widgets.volume-widget.widgets.arc-widget"), horizontal_bar = require("awesome-wm-widgets.volume-widget.widgets.horizontal-bar-widget"), vertical_bar = require("awesome-wm-widgets.volume-widget.widgets.vertical-bar-widget"), } local volume = {} local rows = { layout = wibox.layout.fixed.vertical } local popup = awful.popup({ bg = beautiful.bg_normal, fg = beautiful.fg_normal, ontop = true, visible = false, shape = gears.shape.rounded_rect, border_width = 1, border_color = beautiful.bg_focus, maximum_width = 400, offset = { y = 5 }, widget = {}, }) local function build_main_line(device) if device.active_port ~= nil and device.ports[device.active_port] ~= nil then return device.properties.device_description .. " ยท " .. device.ports[device.active_port] else return device.properties.device_description end end local function build_rows(devices, on_checkbox_click, device_type) local device_rows = { layout = wibox.layout.fixed.vertical } for _, device in pairs(devices) do local checkbox = wibox.widget({ checked = device.is_default, color = beautiful.fg_normal, paddings = 2, shape = gears.shape.circle, forced_width = 20, forced_height = 20, check_color = beautiful.fg_normal, widget = wibox.widget.checkbox, }) checkbox:connect_signal("button::press", function() spawn.easy_async( string.format([[sh -c 'pacmd set-default-%s "%s"']], device_type, device.name), function() on_checkbox_click() end ) end) local row = wibox.widget({ { { { checkbox, valign = "center", layout = wibox.container.place, }, { { text = build_main_line(device), align = "left", widget = wibox.widget.textbox, }, left = 10, layout = wibox.container.margin, }, spacing = 8, layout = wibox.layout.align.horizontal, }, margins = 4, layout = wibox.container.margin, }, bg = beautiful.bg_normal, fg = beautiful.fg_normal, widget = wibox.container.background, }) row:connect_signal("mouse::enter", function(c) checkbox:set_color(beautiful.fg_focus) checkbox:set_check_color(beautiful.fg_focus) c:set_fg(beautiful.fg_focus) c:set_bg(beautiful.bg_focus) end) row:connect_signal("mouse::leave", function(c) checkbox:set_color(beautiful.fg_normal) checkbox:set_check_color(beautiful.fg_normal) c:set_fg(beautiful.fg_normal) c:set_bg(beautiful.bg_normal) end) local old_cursor, old_wibox row:connect_signal("mouse::enter", function() local wb = mouse.current_wibox old_cursor, old_wibox = wb.cursor, wb wb.cursor = "hand1" end) row:connect_signal("mouse::leave", function() if old_wibox then old_wibox.cursor = old_cursor old_wibox = nil end end) row:connect_signal("button::press", function() spawn.easy_async( string.format([[sh -c 'pacmd set-default-%s "%s"']], device_type, device.name), function() on_checkbox_click() end ) end) table.insert(device_rows, row) end return device_rows end local function build_header_row(text) return wibox.widget({ { markup = "" .. text .. "", align = "center", widget = wibox.widget.textbox, }, bg = beautiful.bg_normal, fg = beautiful.fg_normal, widget = wibox.container.background, }) end local function rebuild_popup() spawn.easy_async(LIST_DEVICES_CMD, function(stdout) local sinks, sources = utils.extract_sinks_and_sources(stdout) for i = 0, #rows do rows[i] = nil end table.insert(rows, build_header_row("SINKS")) table.insert( rows, build_rows(sinks, function() rebuild_popup() end, "sink") ) table.insert(rows, build_header_row("SOURCES")) table.insert( rows, build_rows(sources, function() rebuild_popup() end, "source") ) popup:setup(rows) end) end local function worker(user_args) local args = user_args or {} local mixer_cmd = args.mixer_cmd or "pavucontrol" local widget_type = args.widget_type local refresh_rate = args.refresh_rate or 1 local step = args.step or 5 local card = args.card or 1 local device = args.device or "pulse" local mixctrl = args.mixctrl or "Master" local value_type = args.value_type or "-M" local toggle_cmd = args.toggle_cmd or nil if widget_types[widget_type] == nil then volume.widget = widget_types["icon_and_text"].get_widget(args.icon_and_text_args) else volume.widget = widget_types[widget_type].get_widget(args) end local function update_graphic(widget, stdout) local mute = string.match(stdout, "%[(o%D%D?)%]") -- \[(o\D\D?)\] - [on] or [off] if mute == "off" then widget:mute() elseif mute == "on" then widget:unmute() end local volume_level = string.match(stdout, "(%d?%d?%d)%%") -- (\d?\d?\d)\%) volume_level = string.format("% 3d", volume_level) widget:set_volume_level(volume_level) end function volume:inc(s) spawn.easy_async(INC_VOLUME_CMD(card, device, mixctrl, value_type, s or step), function(stdout) update_graphic(volume.widget, stdout) end) end function volume:dec(s) spawn.easy_async(DEC_VOLUME_CMD(card, device, mixctrl, value_type, s or step), function(stdout) update_graphic(volume.widget, stdout) end) end function volume:toggle() if toggle_cmd == nil then spawn.easy_async(TOG_VOLUME_CMD(card, device, mixctrl), function(stdout) update_graphic(volume.widget, stdout) end) else spawn.easy_async(toggle_cmd, function(_stdout) spawn.easy_async(GET_VOLUME_CMD(card, device, mixctrl, value_type), function(stdout) update_graphic(volume.widget, stdout) end) end) end end function volume:mixer() if mixer_cmd then spawn.easy_async(mixer_cmd) end end volume.widget:buttons(awful.util.table.join( awful.button({}, 3, function() if popup.visible then popup.visible = not popup.visible else rebuild_popup() popup:move_next_to(mouse.current_widget_geometry) end end), awful.button({}, 4, function() volume:inc() end), awful.button({}, 5, function() volume:dec() end), awful.button({}, 2, function() volume:mixer() end), awful.button({}, 1, function() volume:toggle() end) )) watch(GET_VOLUME_CMD(card, device, mixctrl, value_type), refresh_rate, update_graphic, volume.widget) return volume.widget end return setmetatable(volume, { __call = function(_, ...) return worker(...) end, })