-- Standard awesome library
local gears = require("gears")
local awful = require("awful")
require("awful.autofocus")
-- Widget and layout library
local wibox = require("wibox")
-- Theme handling library
local beautiful = require("beautiful")
-- Notification library
local naughty = require("naughty")
local gio = require("lgi").Gio
-- local modules
local bindings = require("bindings")
local commondefs = require("commondefs")
local utils = require("utils")
local workspace = require("workspace")
-- local widgets
local BatteryManager = require("battery_mng")
local pager = require("pager")
local urgent_wgt = require("urgent_wgt")
-- {{{ Error handling
-- Check if awesome encountered an error during startup and fell back to
-- another config (This code will only ever execute for the fallback config)
if awesome.startup_errors then
naughty.notify({ preset = naughty.config.presets.critical,
title = "Oops, there were errors during startup!",
text = awesome.startup_errors })
end
-- Handle runtime errors after startup
do
local in_error = false
awesome.connect_signal("debug::error", function (err)
-- Make sure we don't go into an endless error loop
if in_error then return end
in_error = true
naughty.notify({ preset = naughty.config.presets.critical,
title = "Oops, an error happened!",
text = tostring(err) })
in_error = false
end)
end
-- }}}
-- {{{ Variable definitions
-- Themes define colours, icons, font and wallpapers.
beautiful.init(awful.util.getdir("config") .. "/theme.lua")
-- Table of layouts to cover with awful.layout.inc, order matters.
local layouts =
{
awful.layout.suit.tile,
awful.layout.suit.tile.bottom,
}
-- }}}
-- Panel
-- clock, shared among all screens
local clock_time = wibox.widget.textclock("%H:%M", 20)
local clock_date = wibox.widget.textclock("%a %F", 60)
clock_time:set_align("center")
clock_date:set_align("center")
-- systray
local tray_icon_size = 24
local systray = wibox.widget.systray()
local function update_systray_iconsize()
systray:set_base_size(beautiful.xresources.apply_dpi(tray_icon_size, screen.primary))
end
screen.connect_signal('primary_changed', update_systray_iconsize)
update_systray_iconsize()
-- widget indicating a desk with urgent clients
-- global, shown only on the primary screen
local u_wgt = urgent_wgt.UrgentWidget:new()
local b_mng = BatteryManager()
-- Create a wibox for each screen and add it
mywibox = {}
mypromptbox = {}
local wsp = workspace.Workspace:new(layouts)
awful.screen.connect_for_each_screen(function(s)
utils.log('screen', 'Setting up screen %d:', s.index)
for k, v in pairs(s.outputs) do
utils.log('screen', ' %s', k)
end
local panel_width = beautiful.xresources.apply_dpi(128, s)
-- Create a promptbox for each screen
mypromptbox[s] = awful.widget.prompt()
-- Create an imagebox widget which will contains an icon indicating which layout we're using.
local layoutbox = awful.widget.layoutbox(s)
layoutbox:buttons(gears.table.join(
awful.button({ }, 1, function () awful.layout.inc(layouts, 1) end),
awful.button({ }, 3, function () awful.layout.inc(layouts, -1) end),
awful.button({ }, 4, function () awful.layout.inc(layouts, 1) end),
awful.button({ }, 5, function () awful.layout.inc(layouts, -1) end)))
local layout_utils = wibox.layout.fixed.horizontal()
layout_utils:add(layoutbox)
layout_utils:add(b_mng)
layout_utils:add(u_wgt.widget)
local utils_bar = wibox.container.constraint(layout_utils, 'max', nil, beautiful.xresources.apply_dpi(tray_icon_size))
-- Create a tasklist widget
local pgr = pager.Pager:new(wsp, s)
-- Widgets that are aligned to the bottom of the tool bar
local layout_bottom = wibox.layout.fixed.vertical()
layout_bottom:add(systray)
layout_bottom:add(utils_bar)
layout_bottom:add(clock_time)
layout_bottom:add(clock_date)
-- Now bring it all together (with the tasklist in the middle)
local layout = wibox.layout.align.vertical()
layout:set_top(mypromptbox[s])
layout:set_middle(pgr.widget)
layout:set_bottom(layout_bottom)
-- Create the wibox
mywibox[s] = awful.wibar({ position = "right", screen = s, width = panel_width })
mywibox[s]:set_widget(layout)
-- clean up on screen removal
s:connect_signal("removed",
function (s)
mywibox[s] = nil
mypromptbox[s] = nil
end)
-- show desktop on th screen
gears.timer.delayed_call(function(s) wsp:view(s, s.index, 1) end, s)
end)
clientbuttons = gears.table.join(
awful.button({ }, 1, function (c) client.focus = c; c:raise() end),
awful.button({ commondefs.modkey }, 1, awful.mouse.client.move),
awful.button({ commondefs.modkey }, 3, awful.mouse.client.resize))
local globalkeys, clientkeys = bindings.create(wsp)
-- Set keys
root.keys(globalkeys)
-- {{{ Rules
-- Rules to apply to new clients (through the "manage" signal).
awful.rules.rules = {
-- All clients will match this rule.
{ rule = { },
properties = { border_width = 3,
border_color = "#000000",
focus = awful.client.focus.filter,
raise = true,
keys = clientkeys,
buttons = clientbuttons } },
{ rule = { class = "mpv" },
properties = { floating = true } },
}
-- }}}
-- {{{ Signals
-- Signal function to execute when a new client appears.
client.connect_signal("manage", function (c, startup)
-- Prevent clients from being unreachable
awful.placement.no_offscreen(c)
if startup then
return
end
-- Put windows in a smart way, only if they does not set an initial position.
if not c.size_hints.user_position and not c.size_hints.program_position then
awful.placement.no_overlap(c)
end
-- put the client where the mouse is
if c.screen ~= mouse.screen then
c:move_to_screen(mouse.screen)
end
-- put new clients at the end
awful.client.setslave(c)
-- give the new client focus if it spawns on the focused page
if c.focusable and c.first_tag == mouse.screen.selected_tag then
-- use pcall() because the client might be gone by the time we get to it
gears.timer.delayed_call(pcall, function() c:jump_to(false) end)
end
end)
client.connect_signal("focus", function(c) c.border_color = "#b0e2ff" end)
client.connect_signal("unfocus", function(c) c.border_color = "#000000" end)
-- Enable sloppy focus
client.connect_signal("mouse::enter",
function(c)
if awful.layout.get(c.screen) ~= awful.layout.suit.magnifier
and awful.client.focus.filter(c) then
client.focus = c
end
end)
-- }}}
-- show a notification on changing tag properties
do
local tag_prop_notify = {
master_count = 'Master count',
column_count = 'Column count',
master_width_factor = 'Master width factor',
}
for name, desc in pairs(tag_prop_notify) do
tag.connect_signal('property::' .. name,
function(tag)
utils.notify_singleton(wsp, name,
{ title = desc, text = tostring(tag[name]), timeout = 2 })
end)
end
end
-- show notification on display brightness changes
local function brightness_stdout(line)
local action, device = string.match(line, '^KERNEL%[%d+.%d*%]%s+(%a+)%s+([^%s]+)%s+%(backlight%)$')
if action ~= 'change' then
return
end
local function readnum(name)
local file = gio.File.new_for_path('/sys/' .. device .. '/' .. name)
local contents = file:load_contents(nil)
if contents then
contents = string.gsub(contents, "%s$", "")
return string.len(contents) > 0 and tonumber(contents) or nil
end
return nil
end
local brightness = readnum('brightness')
local max_brightness = readnum('max_brightness')
if brightness and max_brightness then
utils.notify_singleton(wsp, 'brightness',
{ title = 'Brightness',
text = tostring(100 * brightness // max_brightness) .. '%'})
end
return nil
end
local function brightness_exit(reason, code)
utils.warn('brightness', 'udevadm monitor exited: %s/%d', reason, code)
end
awful.spawn.with_line_callback('udevadm monitor --kernel --subsystem-match=backlight',
{ stdout = brightness_stdout, exit = brightness_exit })