+
+--]]
+
+local helpers = require("lain.helpers")
+local shell = require("awful.util").shell
+local escape_f = require("awful.util").escape
+local focused = require("awful.screen").focused
+local naughty = require("naughty")
+local wibox = require("wibox")
+local os = os
+local string = string
+
+-- MPD infos
+-- lain.widget.mpd
+
+local function factory(args)
+ local mpd = { widget = wibox.widget.textbox() }
+ local args = args or {}
+ local timeout = args.timeout or 2
+ local password = (args.password and #args.password > 0 and string.format("password %s\\n", args.password)) or ""
+ local host = args.host or os.getenv("MPD_HOST") or "127.0.0.1"
+ local port = args.port or os.getenv("MPD_PORT") or "6600"
+ local music_dir = args.music_dir or os.getenv("HOME") .. "/Music"
+ local cover_pattern = args.cover_pattern or "*\\.(jpg|jpeg|png|gif)$"
+ local cover_size = args.cover_size or 100
+ local default_art = args.default_art
+ local notify = args.notify or "on"
+ local followtag = args.followtag or false
+ local settings = args.settings or function() end
+
+ local mpdh = string.format("telnet://%s:%s", host, port)
+ local echo = string.format("printf \"%sstatus\\ncurrentsong\\nclose\\n\"", password)
+ local cmd = string.format("%s | curl --connect-timeout 1 -fsm 3 %s", echo, mpdh)
+
+ mpd_notification_preset = { title = "Now playing", timeout = 6 }
+
+ helpers.set_map("current mpd track", nil)
+
+ function mpd.update()
+ helpers.async({ shell, "-c", cmd }, function(f)
+ mpd_now = {
+ random_mode = false,
+ single_mode = false,
+ repeat_mode = false,
+ consume_mode = false,
+ pls_pos = "N/A",
+ pls_len = "N/A",
+ state = "N/A",
+ file = "N/A",
+ name = "N/A",
+ artist = "N/A",
+ title = "N/A",
+ album = "N/A",
+ genre = "N/A",
+ track = "N/A",
+ date = "N/A",
+ time = "N/A",
+ elapsed = "N/A"
+ }
+
+ for line in string.gmatch(f, "[^\n]+") do
+ for k, v in string.gmatch(line, "([%w]+):[%s](.*)$") do
+ if k == "state" then mpd_now.state = v
+ elseif k == "file" then mpd_now.file = v
+ elseif k == "Name" then mpd_now.name = escape_f(v)
+ elseif k == "Artist" then mpd_now.artist = escape_f(v)
+ elseif k == "Title" then mpd_now.title = escape_f(v)
+ elseif k == "Album" then mpd_now.album = escape_f(v)
+ elseif k == "Genre" then mpd_now.genre = escape_f(v)
+ elseif k == "Track" then mpd_now.track = escape_f(v)
+ elseif k == "Date" then mpd_now.date = escape_f(v)
+ elseif k == "Time" then mpd_now.time = v
+ elseif k == "elapsed" then mpd_now.elapsed = string.match(v, "%d+")
+ elseif k == "song" then mpd_now.pls_pos = v
+ elseif k == "playlistlength" then mpd_now.pls_len = v
+ elseif k == "repeat" then mpd_now.repeat_mode = v ~= "0"
+ elseif k == "single" then mpd_now.single_mode = v ~= "0"
+ elseif k == "random" then mpd_now.random_mode = v ~= "0"
+ elseif k == "consume" then mpd_now.consume_mode = v ~= "0"
+ end
+ end
+ end
+
+ mpd_notification_preset.text = string.format("%s (%s) - %s\n%s", mpd_now.artist,
+ mpd_now.album, mpd_now.date, mpd_now.title)
+ widget = mpd.widget
+ settings()
+
+ if mpd_now.state == "play" then
+ if notify == "on" and mpd_now.title ~= helpers.get_map("current mpd track") then
+ helpers.set_map("current mpd track", mpd_now.title)
+
+ if followtag then mpd_notification_preset.screen = focused() end
+
+ local common = {
+ preset = mpd_notification_preset,
+ icon = default_art,
+ icon_size = cover_size,
+ replaces_id = mpd.id
+ }
+
+ if not string.match(mpd_now.file, "http.*://") then -- local file instead of http stream
+ local path = string.format("%s/%s", music_dir, string.match(mpd_now.file, ".*/"))
+ local cover = string.format("find '%s' -maxdepth 1 -type f | egrep -i -m1 '%s'",
+ path:gsub("'", "'\\''"), cover_pattern)
+ helpers.async({ shell, "-c", cover }, function(current_icon)
+ common.icon = current_icon:gsub("\n", "")
+ if #common.icon == 0 then common.icon = nil end
+ mpd.id = naughty.notify(common).id
+ end)
+ else
+ mpd.id = naughty.notify(common).id
+ end
+
+ end
+ elseif mpd_now.state ~= "pause" then
+ helpers.set_map("current mpd track", nil)
+ end
+ end)
+ end
+
+ mpd.timer = helpers.newtimer("mpd", timeout, mpd.update, true, true)
+
+ return mpd
+end
+
+return factory
diff --git a/Phranque/awesome/lain/widget/net.lua b/Phranque/awesome/lain/widget/net.lua
new file mode 100644
index 00000000..805b5778
--- /dev/null
+++ b/Phranque/awesome/lain/widget/net.lua
@@ -0,0 +1,113 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2013, Luca CPZ
+ * (c) 2010-2012, Peter Hofmann
+
+--]]
+
+local helpers = require("lain.helpers")
+local naughty = require("naughty")
+local wibox = require("wibox")
+local string = string
+
+-- Network infos
+-- lain.widget.net
+
+local function factory(args)
+ local net = { widget = wibox.widget.textbox(), devices = {} }
+ local args = args or {}
+ local timeout = args.timeout or 2
+ local units = args.units or 1024 -- KB
+ local notify = args.notify or "on"
+ local wifi_state = args.wifi_state or "off"
+ local eth_state = args.eth_state or "off"
+ local screen = args.screen or 1
+ local settings = args.settings or function() end
+
+ -- Compatibility with old API where iface was a string corresponding to 1 interface
+ net.iface = (args.iface and (type(args.iface) == "string" and {args.iface}) or
+ (type(args.iface) == "table" and args.iface)) or {}
+
+ function net.get_device()
+ helpers.line_callback("ip link", function(line)
+ net.iface[#net.iface + 1] = not string.match(line, "LOOPBACK") and string.match(line, "(%w+): <") or nil
+ end)
+ end
+
+ if #net.iface == 0 then net.get_device() end
+
+ function net.update()
+ -- These are the totals over all specified interfaces
+ net_now = {
+ devices = {},
+ -- Bytes since last iteration
+ sent = 0,
+ received = 0
+ }
+
+ for _, dev in ipairs(net.iface) do
+ local dev_now = {}
+ local dev_before = net.devices[dev] or { last_t = 0, last_r = 0 }
+ local now_t = tonumber(helpers.first_line(string.format("/sys/class/net/%s/statistics/tx_bytes", dev)) or 0)
+ local now_r = tonumber(helpers.first_line(string.format("/sys/class/net/%s/statistics/rx_bytes", dev)) or 0)
+
+ dev_now.carrier = helpers.first_line(string.format("/sys/class/net/%s/carrier", dev)) or "0"
+ dev_now.state = helpers.first_line(string.format("/sys/class/net/%s/operstate", dev)) or "down"
+
+ dev_now.sent = (now_t - dev_before.last_t) / timeout / units
+ dev_now.received = (now_r - dev_before.last_r) / timeout / units
+
+ net_now.sent = net_now.sent + dev_now.sent
+ net_now.received = net_now.received + dev_now.received
+
+ dev_now.sent = string.format("%.1f", dev_now.sent)
+ dev_now.received = string.format("%.1f", dev_now.received)
+
+ dev_now.last_t = now_t
+ dev_now.last_r = now_r
+
+ if wifi_state == "on" and helpers.first_line(string.format("/sys/class/net/%s/uevent", dev)) == "DEVTYPE=wlan" and string.match(dev_now.carrier, "1") then
+ dev_now.wifi = true
+ dev_now.signal = tonumber(string.match(helpers.lines_from("/proc/net/wireless")[3], "(%-%d+%.)")) or nil
+ end
+
+ if eth_state == "on" and helpers.first_line(string.format("/sys/class/net/%s/uevent", dev)) ~= "DEVTYPE=wlan" and string.match(dev_now.carrier, "1") then
+ dev_now.ethernet = true
+ end
+
+ net.devices[dev] = dev_now
+
+ -- Notify only once when connection is lost
+ if string.match(dev_now.carrier, "0") and notify == "on" and helpers.get_map(dev) then
+ naughty.notify {
+ title = dev,
+ text = "No carrier",
+ icon = helpers.icons_dir .. "no_net.png",
+ screen = screen
+ }
+ helpers.set_map(dev, false)
+ elseif string.match(dev_now.carrier, "1") then
+ helpers.set_map(dev, true)
+ end
+
+ net_now.carrier = dev_now.carrier
+ net_now.state = dev_now.state
+ net_now.devices[dev] = dev_now
+ -- net_now.sent and net_now.received will be
+ -- the totals across all specified devices
+ end
+
+ net_now.sent = string.format("%.1f", net_now.sent)
+ net_now.received = string.format("%.1f", net_now.received)
+
+ widget = net.widget
+ settings()
+ end
+
+ helpers.newtimer("network", timeout, net.update)
+
+ return net
+end
+
+return factory
diff --git a/Phranque/awesome/lain/widget/pulse.lua b/Phranque/awesome/lain/widget/pulse.lua
new file mode 100644
index 00000000..f63fe55c
--- /dev/null
+++ b/Phranque/awesome/lain/widget/pulse.lua
@@ -0,0 +1,57 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2016, Luca CPZ
+
+--]]
+
+local helpers = require("lain.helpers")
+local shell = require("awful.util").shell
+local wibox = require("wibox")
+local string = string
+local type = type
+
+-- PulseAudio volume
+-- lain.widget.pulse
+
+local function factory(args)
+ local pulse = { widget = wibox.widget.textbox(), device = "N/A" }
+ local args = args or {}
+ local timeout = args.timeout or 5
+ local settings = args.settings or function() end
+
+ pulse.devicetype = args.devicetype or "sink"
+ pulse.cmd = args.cmd or "pacmd list-" .. pulse.devicetype .. "s | sed -n -e '/*/,$!d' -e '/index/p' -e '/base volume/d' -e '/volume:/p' -e '/muted:/p' -e '/device\\.string/p'"
+
+ function pulse.update()
+ helpers.async({ shell, "-c", type(pulse.cmd) == "string" and pulse.cmd or pulse.cmd() },
+ function(s)
+ volume_now = {
+ index = string.match(s, "index: (%S+)") or "N/A",
+ device = string.match(s, "device.string = \"(%S+)\"") or "N/A",
+ muted = string.match(s, "muted: (%S+)") or "N/A"
+ }
+
+ pulse.device = volume_now.index
+
+ 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"
+
+ widget = pulse.widget
+ settings()
+ end)
+ end
+
+ helpers.newtimer("pulse", timeout, pulse.update)
+
+ return pulse
+end
+
+return factory
diff --git a/Phranque/awesome/lain/widget/pulsebar.lua b/Phranque/awesome/lain/widget/pulsebar.lua
new file mode 100644
index 00000000..317468f4
--- /dev/null
+++ b/Phranque/awesome/lain/widget/pulsebar.lua
@@ -0,0 +1,162 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2013, Luca CPZ
+ * (c) 2013, Rman
+
+--]]
+
+local helpers = require("lain.helpers")
+local awful = require("awful")
+local naughty = require("naughty")
+local wibox = require("wibox")
+local math = math
+local string = string
+local type = type
+local tonumber = tonumber
+
+-- PulseAudio volume bar
+-- lain.widget.pulsebar
+
+local function factory(args)
+ local pulsebar = {
+ colors = {
+ background = "#000000",
+ mute = "#EB8F8F",
+ unmute = "#A4CE8A"
+ },
+
+ _current_level = 0,
+ _mute = "no",
+ device = "N/A"
+ }
+
+ local args = args or {}
+ local timeout = args.timeout or 5
+ local settings = args.settings or function() end
+ local width = args.width or 63
+ local height = args.height or 1
+ local margins = args.margins or 1
+ local paddings = args.paddings or 1
+ local ticks = args.ticks or false
+ local ticks_size = args.ticks_size or 7
+
+ pulsebar.colors = args.colors or pulsebar.colors
+ pulsebar.followtag = args.followtag or false
+ pulsebar.notification_preset = args.notification_preset
+ 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'"
+
+ if not pulsebar.notification_preset then
+ pulsebar.notification_preset = {
+ font = "Monospace 10"
+ }
+ end
+
+ pulsebar.bar = wibox.widget {
+ color = pulsebar.colors.unmute,
+ background_color = pulsebar.colors.background,
+ forced_height = height,
+ forced_width = width,
+ margins = margins,
+ paddings = paddings,
+ ticks = ticks,
+ ticks_size = ticks_size,
+ widget = wibox.widget.progressbar,
+ }
+
+ pulsebar.tooltip = awful.tooltip({ objects = { pulsebar.bar } })
+
+ function pulsebar.update(callback)
+ helpers.async({ awful.util.shell, "-c", type(pulsebar.cmd) == "string" and pulsebar.cmd or pulsebar.cmd() },
+ function(s)
+ volume_now = {
+ index = string.match(s, "index: (%S+)") or "N/A",
+ device = string.match(s, "device.string = \"(%S+)\"") or "N/A",
+ muted = string.match(s, "muted: (%S+)") or "N/A"
+ }
+
+ pulsebar.device = volume_now.index
+
+ 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
+
+ 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)
+ pulsebar.bar:set_value(pulsebar._current_level / 100)
+ if pulsebar._current_level == 0 or mute == "yes" then
+ pulsebar._mute = mute
+ pulsebar.tooltip:set_text ("[muted]")
+ pulsebar.bar.color = pulsebar.colors.mute
+ else
+ pulsebar._mute = "no"
+ pulsebar.tooltip:set_text(string.format("%s %s: %s", pulsebar.devicetype, pulsebar.device, volu))
+ pulsebar.bar.color = pulsebar.colors.unmute
+ end
+
+ settings()
+
+ if type(callback) == "function" then callback() end
+ end
+ end)
+ end
+
+ function pulsebar.notify()
+ pulsebar.update(function()
+ local preset = pulsebar.notification_preset
+
+ preset.title = string.format("%s %s - %s%%", pulsebar.devicetype, pulsebar.device, pulsebar._current_level)
+
+ if pulsebar._mute == "yes" then
+ preset.title = preset.title .. " muted"
+ end
+
+ -- 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
+ end
+
+ int = math.modf((pulsebar._current_level / 100) * tot)
+ preset.text = string.format("[%s%s]", string.rep("|", int),
+ string.rep(" ", tot - int))
+
+ if pulsebar.followtag then preset.screen = awful.screen.focused() end
+
+ 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
+ end)
+ end
+
+ helpers.newtimer(string.format("pulsebar-%s", pulsebar.sink), timeout, pulsebar.update)
+
+ return pulsebar
+end
+
+return factory
diff --git a/Phranque/awesome/lain/widget/sysload.lua b/Phranque/awesome/lain/widget/sysload.lua
new file mode 100644
index 00000000..adf3e035
--- /dev/null
+++ b/Phranque/awesome/lain/widget/sysload.lua
@@ -0,0 +1,38 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2013, Luca CPZ
+ * (c) 2010-2012, Peter Hofmann
+
+--]]
+
+local helpers = require("lain.helpers")
+local wibox = require("wibox")
+local open, match = io.open, string.match
+
+-- System load
+-- lain.widget.sysload
+
+local function factory(args)
+ local sysload = { widget = wibox.widget.textbox() }
+ local args = args or {}
+ local timeout = args.timeout or 2
+ local settings = args.settings or function() end
+
+ function sysload.update()
+ local f = open("/proc/loadavg")
+ local ret = f:read("*all")
+ f:close()
+
+ load_1, load_5, load_15 = match(ret, "([^%s]+) ([^%s]+) ([^%s]+)")
+
+ widget = sysload.widget
+ settings()
+ end
+
+ helpers.newtimer("sysload", timeout, sysload.update)
+
+ return sysload
+end
+
+return factory
diff --git a/Phranque/awesome/lain/widget/temp.lua b/Phranque/awesome/lain/widget/temp.lua
new file mode 100644
index 00000000..3c93252b
--- /dev/null
+++ b/Phranque/awesome/lain/widget/temp.lua
@@ -0,0 +1,41 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2013, Luca CPZ
+
+--]]
+
+local helpers = require("lain.helpers")
+local wibox = require("wibox")
+local open = io.open
+local tonumber = tonumber
+
+-- coretemp
+-- lain.widget.temp
+
+local function factory(args)
+ local temp = { widget = wibox.widget.textbox() }
+ local args = args or {}
+ local timeout = args.timeout or 2
+ local tempfile = args.tempfile or "/sys/class/thermal/thermal_zone0/temp"
+ local settings = args.settings or function() end
+
+ function temp.update()
+ local f = open(tempfile)
+ if f then
+ coretemp_now = tonumber(f:read("*all")) / 1000
+ f:close()
+ else
+ coretemp_now = "N/A"
+ end
+
+ widget = temp.widget
+ settings()
+ end
+
+ helpers.newtimer("coretemp", timeout, temp.update)
+
+ return temp
+end
+
+return factory
diff --git a/Phranque/awesome/lain/widget/weather.lua b/Phranque/awesome/lain/widget/weather.lua
new file mode 100644
index 00000000..f35ca68b
--- /dev/null
+++ b/Phranque/awesome/lain/widget/weather.lua
@@ -0,0 +1,160 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2015, Luca CPZ
+
+--]]
+
+local helpers = require("lain.helpers")
+local json = require("lain.util").dkjson
+local focused = require("awful.screen").focused
+local naughty = require("naughty")
+local wibox = require("wibox")
+local math = math
+local os = os
+local string = string
+local tonumber = tonumber
+
+-- OpenWeatherMap
+-- current weather and X-days forecast
+-- lain.widget.weather
+
+local function factory(args)
+ local weather = { widget = wibox.widget.textbox() }
+ local args = args or {}
+ local APPID = args.APPID or "3e321f9414eaedbfab34983bda77a66e" -- lain's default
+ local timeout = args.timeout or 60 * 15 -- 15 min
+ local timeout_forecast = args.timeout or 60 * 60 * 24 -- 24 hrs
+ local current_call = args.current_call or "curl -s 'http://api.openweathermap.org/data/2.5/weather?id=%s&units=%s&lang=%s&APPID=%s'"
+ local forecast_call = args.forecast_call or "curl -s 'http://api.openweathermap.org/data/2.5/forecast/daily?id=%s&units=%s&lang=%s&cnt=%s&APPID=%s'"
+ local city_id = args.city_id or 0 -- placeholder
+ local units = args.units or "metric"
+ local lang = args.lang or "en"
+ local cnt = args.cnt or 5
+ local date_cmd = args.date_cmd or "date -u -d @%d +'%%a %%d'"
+ local icons_path = args.icons_path or helpers.icons_dir .. "openweathermap/"
+ local notification_preset = args.notification_preset or {}
+ local notification_text_fun = args.notification_text_fun or
+ function (wn)
+ local day = os.date("%a %d", wn["dt"])
+ local tmin = math.floor(wn["temp"]["min"])
+ local tmax = math.floor(wn["temp"]["max"])
+ local desc = wn["weather"][1]["description"]
+ return string.format("%s: %s, %d - %d ", day, desc, tmin, tmax)
+ end
+ local weather_na_markup = args.weather_na_markup or " N/A "
+ local followtag = args.followtag or false
+ local showpopup = args.showpopup or "on"
+ local settings = args.settings or function() end
+
+ weather.widget:set_markup(weather_na_markup)
+ weather.icon_path = icons_path .. "na.png"
+ weather.icon = wibox.widget.imagebox(weather.icon_path)
+
+ function weather.show(t_out)
+ weather.hide()
+
+ if followtag then
+ notification_preset.screen = focused()
+ end
+
+ if not weather.notification_text then
+ weather.update()
+ weather.forecast_update()
+ end
+
+ weather.notification = naughty.notify({
+ text = weather.notification_text,
+ icon = weather.icon_path,
+ timeout = t_out,
+ preset = notification_preset
+ })
+ end
+
+ function weather.hide()
+ if weather.notification then
+ naughty.destroy(weather.notification)
+ weather.notification = nil
+ end
+ end
+
+ function weather.attach(obj)
+ obj:connect_signal("mouse::enter", function()
+ weather.show(0)
+ end)
+ obj:connect_signal("mouse::leave", function()
+ weather.hide()
+ end)
+ end
+
+ function weather.forecast_update()
+ local cmd = string.format(forecast_call, city_id, units, lang, cnt, APPID)
+ helpers.async(cmd, function(f)
+ local pos, err
+ weather_now, pos, err = json.decode(f, 1, nil)
+
+ if not err and type(weather_now) == "table" and tonumber(weather_now["cod"]) == 200 then
+ weather.notification_text = ""
+ for i = 1, weather_now["cnt"] do
+ weather.notification_text = weather.notification_text ..
+ notification_text_fun(weather_now["list"][i])
+ if i < weather_now["cnt"] then
+ weather.notification_text = weather.notification_text .. "\n"
+ end
+ end
+ end
+ end)
+ end
+
+ function weather.update()
+ local cmd = string.format(current_call, city_id, units, lang, APPID)
+ helpers.async(cmd, function(f)
+ local pos, err, icon
+ weather_now, pos, err = json.decode(f, 1, nil)
+
+ if not err and type(weather_now) == "table" and tonumber(weather_now["cod"]) == 200 then
+ local sunrise = tonumber(weather_now["sys"]["sunrise"])
+ local sunset = tonumber(weather_now["sys"]["sunset"])
+ local icon = weather_now["weather"][1]["icon"]
+ local loc_now = os.time() -- local time
+ local loc_m = os.time { year = os.date("%Y"), month = os.date("%m"), day = os.date("%d"), hour = 0 } -- local time from midnight
+ local loc_d = os.date("*t", loc_now) -- table YMDHMS for current local time (for TZ calculation)
+ local utc_d = os.date("!*t", loc_now) -- table YMDHMS for current UTC time
+ local utc_now = os.time(utc_d) -- UTC time now
+ local offdt = (loc_d.isdst and 1 or 0) * 3600 + 100 * (loc_d.min - utc_d.min) / 60 -- DST offset
+ local offset = os.difftime(loc_now, utc_now) + (loc_d.isdst and 1 or 0) * 3600 + 100 * (loc_d.min - utc_d.min) / 60 -- TZ offset (including DST)
+ local offday = (offset < 0 and -86400) or 86400 -- 24 hour correction value (+86400 or -86400)
+
+ -- if current UTC time is earlier then local midnight -> positive offset (negative otherwise)
+ if offset * (loc_m - utc_now + offdt) > 0 then
+ sunrise = sunrise + offday -- Shift sunset and sunrise times by 24 hours
+ sunset = sunset + offday
+ end
+
+ if sunrise <= loc_now and loc_now <= sunset then
+ icon = string.gsub(icon, "n", "d")
+ else
+ icon = string.gsub(icon, "d", "n")
+ end
+
+ weather.icon_path = icons_path .. icon .. ".png"
+ widget = weather.widget
+ settings()
+ else
+ weather.icon_path = icons_path .. "na.png"
+ weather.widget:set_markup(weather_na_markup)
+ end
+
+ weather.icon:set_image(weather.icon_path)
+ end)
+ end
+
+ if showpopup == "on" then weather.attach(weather.widget) end
+
+ weather.timer = helpers.newtimer("weather-" .. city_id, timeout, weather.update, false, true)
+ weather.timer_forecast = helpers.newtimer("weather_forecast-" .. city_id, timeout, weather.forecast_update, false, true)
+
+ return weather
+end
+
+return factory
diff --git a/Phranque/awesome/lain/wiki/Home.md b/Phranque/awesome/lain/wiki/Home.md
new file mode 100644
index 00000000..d098efe3
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/Home.md
@@ -0,0 +1,43 @@
+Welcome to the Lain wiki!
+
+If you spot a typo or have a suggestion to improve these pages, please notify me opening an [issue](https://github.com/lcpz/lain/issues) format. Thank you.
+
+Dependencies
+------------
+
+Package | Requested by | Reasons of choice
+--- | --- | ---
+[curl](https://curl.haxx.se) | `imap`, `mpd`, and `weather` widgets | 1. faster and simpler to use than [LuaSocket](https://github.com/diegonehab/luasocket); 2. it's in the core of almost every distro; 3. can be called [asynchronously](https://awesomewm.org/doc/api/libraries/awful.spawn.html#easy_async)
+GLib >= 2.54 | `fs` widget | Pure Awesome/Lua implementation.
+
+The second dependency will be removed once all major distros update their Gio/Glib versions.
+
+Installation
+------------
+
+### Arch Linux
+
+[AUR package](https://aur.archlinux.org/packages/lain-git/)
+
+### Other distributions
+
+```shell
+git clone https://github.com/lcpz/lain.git ~/.config/awesome/lain
+```
+
+Also available via [LuaRocks](https://luarocks.org/modules/aajjbb/lain).
+
+Usage
+--------
+
+First, include it into your `rc.lua`:
+
+```lua
+local lain = require("lain")
+```
+
+Then check out the submodules you want:
+
+- [Layouts](https://github.com/lcpz/lain/wiki/Layouts)
+- [Widgets](https://github.com/lcpz/lain/wiki/Widgets)
+- [Utilities](https://github.com/lcpz/lain/wiki/Utilities)
diff --git a/Phranque/awesome/lain/wiki/Layouts.md b/Phranque/awesome/lain/wiki/Layouts.md
new file mode 100644
index 00000000..0286d4b5
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/Layouts.md
@@ -0,0 +1,255 @@
+
+ lain/layout
+ .
+ |-- termfair
+ |-- termfair.center
+ |-- cascade
+ |-- cascade.tile
+ |-- centerwork
+ |-- centerwork.horizontal
+
+Usage
+=====
+
+As usual, specify your favourites in `awful.layout.layouts`, or set them on specific tags with [`awful.layout.set`](https://awesomewm.org/doc/api/libraries/awful.layout.html#set).
+
+```lua
+awful.layout.set(lain.layout.termfair, tag)
+```
+
+How do layouts work?
+====================
+
+`termfair`
+--------
+
+This layout restricts the size of each window. Each window will have the
+same width but is variable in height. Furthermore, windows are
+left-aligned. The basic workflow is as follows (the number above the
+screen is the number of open windows, the number in a cell is the fixed
+number of a client):
+
+ (1) (2) (3)
+ +---+---+---+ +---+---+---+ +---+---+---+
+ | | | | | | | | | | | |
+ | 1 | | | -> | 2 | 1 | | -> | 3 | 2 | 1 | ->
+ | | | | | | | | | | | |
+ +---+---+---+ +---+---+---+ +---+---+---+
+
+ (4) (5) (6)
+ +---+---+---+ +---+---+---+ +---+---+---+
+ | 4 | | | | 5 | 4 | | | 6 | 5 | 4 |
+ +---+---+---+ -> +---+---+---+ -> +---+---+---+
+ | 3 | 2 | 1 | | 3 | 2 | 1 | | 3 | 2 | 1 |
+ +---+---+---+ +---+---+---+ +---+---+---+
+
+The first client will be located in the left column. When opening
+another window, this new window will be placed in the left column while
+moving the first window into the middle column. Once a row is full,
+another row above it will be created.
+
+Default number of columns and rows are respectively taken from `nmaster`
+and `ncol` values in `awful.tag`, but you can set your own.
+
+For example, this sets `termfair` to 3 columns and at least 1 row:
+
+```lua
+lain.layout.termfair.nmaster = 3
+lain.layout.termfair.ncol = 1
+```
+
+`termfair.center`
+----------
+
+Similar to `termfair`, but with fixed number of vertical columns. Cols are centerded until there are `nmaster` columns, then windows are stacked as slaves, with possibly `ncol` clients per column at most.
+
+ (1) (2) (3)
+ +---+---+---+ +-+---+---+-+ +---+---+---+
+ | | | | | | | | | | | | |
+ | | 1 | | -> | | 1 | 2 | | -> | 1 | 2 | 3 | ->
+ | | | | | | | | | | | | |
+ +---+---+---+ +-+---+---+-+ +---+---+---+
+
+ (4) (5)
+ +---+---+---+ +---+---+---+
+ | | | 3 | | | 2 | 4 |
+ + 1 + 2 +---+ -> + 1 +---+---+
+ | | | 4 | | | 3 | 5 |
+ +---+---+---+ +---+---+---+
+
+Like `termfair`, default number of columns and rows are respectively taken from `nmaster`
+and `ncol` values in `awful.tag`, but you can set your own.
+
+For example, this sets `termfair.center` to 3 columns and at least 1 row:
+
+```lua
+lain.layout.termfair.center.nmaster = 3
+lain.layout.termfair.center.ncol = 1
+```
+
+`cascade`
+-------
+
+Cascade all windows of a tag.
+
+You can control the offsets by setting these two variables:
+
+```lua
+lain.layout.cascade.offset_x = 64
+lain.layout.cascade.offset_y = 16
+```
+
+The following reserves space for 5 windows:
+
+```lua
+lain.layout.cascade.nmaster = 5
+```
+
+That is, no window will get resized upon the creation of a new window,
+unless there's more than 5 windows.
+
+`cascade.tile`
+-----------
+
+Similar to `awful.layout.suit.tile` layout, however, clients in the slave
+column are cascaded instead of tiled.
+
+Left column size can be set, otherwise is controlled by `mwfact` of the
+tag. Additional windows will be opened in another column on the right.
+New windows are placed above old windows.
+
+Whether the slave column is placed on top of the master window or not is
+controlled by the value of `ncol`. A value of 1 means "overlapping slave column"
+and anything else means "don't overlap windows".
+
+Usage example:
+
+```lua
+lain.layout.cascade.tile.offset_x = 2
+lain.layout.cascade.tile.offset_y = 32
+lain.layout.cascade.tile.extra_padding = 5
+lain.layout.cascade.tile.nmaster = 5
+lain.layout.cascade.tile.ncol = 2
+```
+
+`extra_padding` reduces the size of the master window if "overlapping
+slave column" is activated. This allows you to see if there are any
+windows in your slave column.
+
+Setting `offset_x` to a very small value or even 0 is recommended to avoid wasting space.
+
+`centerwork`
+----------
+
+You start with one window, centered horizontally:
+
+ +--------------------------+
+ | +----------+ |
+ | | | |
+ | | | |
+ | | | |
+ | | MAIN | |
+ | | | |
+ | | | |
+ | | | |
+ | | | |
+ | +----------+ |
+ +--------------------------+
+
+This is your main working window. You do most of the work right here.
+Sometimes, you may want to open up additional windows. They're put on left and right, alternately.
+
+ +--------------------------+
+ | +---+ +----------+ +---+ |
+ | | | | | | | |
+ | | | | | | | |
+ | | | | | | | |
+ | +---+ | MAIN | +---+ |
+ | +---+ | | +---+ |
+ | | | | | | | |
+ | | | | | | | |
+ | | | | | | | |
+ | +---+ +----------+ +---+ |
+ +--------------------------+
+
+*Please note:* If you use Awesome's default configuration, navigation in
+this layout may be very confusing. How do you get from the main window
+to satellite ones depends on the order in which the windows are opened.
+Thus, use of `awful.client.focus.bydirection()` is suggested.
+Here's an example:
+
+```lua
+globalkeys = awful.util.table.join(
+ -- [...]
+ awful.key({ modkey }, "j",
+ function()
+ awful.client.focus.bydirection("down")
+ if client.focus then client.focus:raise() end
+ end),
+ awful.key({ modkey }, "k",
+ function()
+ awful.client.focus.bydirection("up")
+ if client.focus then client.focus:raise() end
+ end),
+ awful.key({ modkey }, "h",
+ function()
+ awful.client.focus.bydirection("left")
+ if client.focus then client.focus:raise() end
+ end),
+ awful.key({ modkey }, "l",
+ function()
+ awful.client.focus.bydirection("right")
+ if client.focus then client.focus:raise() end
+ end),
+ -- [...]
+)
+```
+
+`centerwork.horizontal`
+-----------
+
+Same as `centerwork`, except that the main
+window expands horizontally, and the additional windows
+are put ontop/below it. Useful if you have a screen turned 90°.
+
+Pre 4.0 `uselesstile` patches
+=============================
+
+In branch 3.5, this module provided useless gaps layouts. Since useless gaps have been implemented in Awesome 4.0, those layouts have been removed.
+
+Following are a couple of `uselesstile` variants that were not part of lain. They are kept only for reference and are not supported.
+
+Xmonad-like
+-----------
+
+If you want to have `awful.layout.suit.tile` behave like xmonad, with internal gaps two times wider than external ones, download [this](https://gist.github.com/lcpz/9e56dcfbe66bfe14967c) as `lain/layout/uselesstile`.
+
+Inverted master
+---------------
+
+Want to invert master window position? Use [this](https://gist.github.com/lcpz/c59dc59c9f99d98218eb) version. You can set `single_gap` with `width` and `height` in your `theme.lua`, in order to define the window geometry when there's only one client, otherwise it goes maximized. An example:
+
+ theme.single_gap = { width = 600, height = 100 }
+
+What about layout icons?
+========================
+
+They are located in ``lain/icons/layout``.
+
+To use them, define new `layout_*` variables in your ``theme.lua``. For instance:
+
+```lua
+theme.lain_icons = os.getenv("HOME") ..
+ "/.config/awesome/lain/icons/layout/default/"
+theme.layout_termfair = theme.lain_icons .. "termfair.png"
+theme.layout_centerfair = theme.lain_icons .. "centerfair.png" -- termfair.center
+theme.layout_cascade = theme.lain_icons .. "cascade.png"
+theme.layout_cascadetile = theme.lain_icons .. "cascadetile.png" -- cascade.tile
+theme.layout_centerwork = theme.lain_icons .. "centerwork.png"
+theme.layout_centerworkh = theme.lain_icons .. "centerworkh.png" -- centerwork.horizontal
+```
+
+Credit goes to [Nicolas Estibals](https://github.com/nestibal) for creating
+layout icons for default theme.
+
+You can use them as a template for your custom versions.
\ No newline at end of file
diff --git a/Phranque/awesome/lain/wiki/Utilities.md b/Phranque/awesome/lain/wiki/Utilities.md
new file mode 100644
index 00000000..dd54056f
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/Utilities.md
@@ -0,0 +1,338 @@
+Quake
+-----
+
+A Quake-like dropdown container for your favourite application.
+
+**Usage**
+
+Define it globally to have a single instance for all screens:
+
+```lua
+local quake = lain.util.quake()
+```
+
+or define it in `connect_for_each_screen` to have one instance for each screen:
+
+```lua
+awful.screen.connect_for_each_screen(function(s)
+ -- Quake application
+ s.quake = lain.util.quake()
+ -- [...]
+```
+
+**Keybinding example**
+
+If using a global instance:
+```lua
+awful.key({ modkey, }, "z", function () quake:toggle() end),
+```
+
+If using a per-screen instance:
+```lua
+awful.key({ modkey, }, "z", function () awful.screen.focused().quake:toggle() end),
+```
+
+**Input table**
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`app` | client to spawn | string | "xterm"
+`name` | client name | string | "QuakeDD"
+`argname` | how to specify client name | string | "-name %s"
+`extra` | extra `app` arguments | string | empty string
+`border` | border width | integer | 1
+`visible` | initially visible | boolean | false
+`followtag` | always spawn on currently focused screen | boolean | false
+`overlap` | Overlap the wibox or not | boolean | false
+`settings` | Additional settings to make on the client | function | `nil`
+`screen` | screen where to spawn the client | integer | `awful.screen.focused()`
+`height` | dropdown client height | float in [0,1] or exact pixels number | 0.25
+`width` | dropdown client width | float in [0,1] or exact pixels number | 1
+`vert` | vertical position | string, possible values: "top", "bottom", "center" | "top"
+`horiz` | horizontal position | string, possible values: "left", "right", "center" | "left"
+
+`height` and `width` express a fraction of the workspace.
+
+`settings` is a function which takes the client as input, and can be used to customize its properties. For instance:
+
+```lua
+-- set the client sticky
+s.quake = lain.util.quake { settings = function(c) c.sticky = true end }
+```
+
+Read [here](https://awesomewm.org/doc/api/classes/client.html#Object_properties) for the complete list of properties.
+
+**Notes**
+
+* [Does not work](https://github.com/lcpz/lain/issues/358) with `gnome-terminal`, `konsole`, or any other terminal which is strictly designed for a Desktop Environment. Just pick a better terminal, [there's plenty](https://wiki.archlinux.org/index.php/List_of_applications#Terminal_emulators).
+* Set `followtag = true` if [experiencing issues with multiple screens](https://github.com/lcpz/lain/issues/346).
+* If you have a `awful.client.setslave` rule for your application, ensure you use an exception for `QuakeDD` (or your defined `name`). Otherwise, you may run into problems with focus.
+* If you are using a VTE-based terminal like `termite`, be sure to set [`argname = "--name %s"`](https://github.com/lcpz/lain/issues/211).
+
+Separators
+----------
+
+Adds Cairo separators.
+
+```lua
+local separators = lain.util.separators
+```
+
+A separator function `separators.separator` takes two color arguments, defined as strings. `"alpha"` argument is allowed. Example:
+
+```lua
+arrl_dl = separators.arrow_left(beautiful.bg_focus, "alpha")
+arrl_ld = separators.arrow_left("alpha", beautiful.bg_focus)
+```
+
+You can customize height and width by setting `separators_height` and `separators_width` in your `theme.lua`. Default values are 0 and 9, respectively.
+
+List of functions:
+
+ +-- separators
+ |
+ |`-- arrow_right() Draw a right arrow.
+ `-- arrow_left() Draw a left arrow.
+
+markup
+------
+
+Mades markup easier.
+
+```lua
+local markup = lain.util.markup
+```
+
+List of functions:
+
+ +-- markup
+ |
+ |`-- bold() Set bold.
+ |`-- italic() Set italicized text.
+ |`-- strike() Set strikethrough text.
+ |`-- underline() Set underlined text.
+ |`-- monospace() Set monospaced text.
+ |`-- big() Set bigger text.
+ |`-- small() Set smaller text.
+ |`-- font() Set the font of the text.
+ |`-- font() Set the font of the text.
+ |`-- color() Set background and foreground color.
+ |`-- fontfg() Set font and foreground color.
+ |`-- fontbg() Set font and background color.
+ `-- fontcolor() Set font, plus background and foreground colors.
+ |
+ |`--+ bg
+ | |
+ | `-- color() Set background color.
+ |
+ `--+ fg
+ |
+ `-- color() Set foreground color.
+
+they all take one argument, which is the text to markup, except the following:
+
+```lua
+markup.font(font, text)
+markup.color(fg, bg, text)
+markup.fontfg(font, fg, text)
+markup.fontbg(font, bg, text)
+markup.fontcolor(font, fg, bg, text)
+markup.fg.color(color, text)
+markup.bg.color(color, text)
+```
+
+Dynamic tagging
+---------------
+
+That is:
+
+- add a new tag;
+- rename current tag;
+- move current tag;
+- delete current tag.
+
+If you delete a tag, any rule set on it shall be broken, so be careful.
+
+Use it with key bindings like these:
+
+```lua
+awful.key({ modkey, "Shift" }, "n", function () lain.util.add_tag(mylayout) end),
+awful.key({ modkey, "Shift" }, "r", function () lain.util.rename_tag() end),
+awful.key({ modkey, "Shift" }, "Left", function () lain.util.move_tag(1) end), -- move to next tag
+awful.key({ modkey, "Shift" }, "Right", function () lain.util.move_tag(-1) end), -- move to previous tag
+awful.key({ modkey, "Shift" }, "d", function () lain.util.delete_tag() end),
+```
+
+The argument in `lain.util.add_tag` represents the tag layout, and is optional: if not present, it will be defaulted to `awful.layout.suit.tile`.
+
+Useless gaps resize
+---------------------
+
+Changes `beautiful.useless_gaps` on the fly.
+
+```lua
+lain.util.useless_gap_resize(thatmuch, s, t)
+```
+
+The argument `thatmuch` is the number of pixel to add to/substract from gaps (integer).
+
+The arguments `s` and `t` are the `awful.screen` and `awful.tag` in which you want to change the gap. They are optional.
+
+Following are example keybindings for changing client gaps on current screen and tag.
+
+Example 1:
+
+```lua
+-- On the fly useless gaps change
+awful.key({ altkey, "Control" }, "+", function () lain.util.useless_gaps_resize(1) end),
+awful.key({ altkey, "Control" }, "-", function () lain.util.useless_gaps_resize(-1) end),
+```
+
+where `altkey = Mod1`. Example 2:
+
+```lua
+mywidget:buttons(awful.util.table.join (
+ awful.button({}, 4, function() lain.util.useless_gaps_resize(-1) end),
+ awful.button({}, 5, function() lain.util.useless_gaps_resize(1) end)
+ end)
+))
+```
+
+so when hovering the mouse over `mywidget`, you can adjust useless gaps size by scrolling with the mouse wheel.
+
+tag\_view\_nonempty
+-------------------
+
+This function lets you jump to the next/previous non-empty tag.
+It takes two arguments:
+
+* `direction`: `1` for next non-empty tag, `-1` for previous.
+* `sc`: Screen which the taglist is in. Default is `mouse.screen` or `1`. This
+ argument is optional.
+
+You can use it with key bindings like these:
+
+```lua
+-- Non-empty tag browsing
+awful.key({ altkey }, "Left", function () lain.util.tag_view_nonempty(-1) end),
+awful.key({ altkey }, "Right", function () lain.util.tag_view_nonempty(1) end),
+```
+
+where `altkey = "Mod1"`.
+
+magnify\_client
+---------------
+
+Set a client to floating and resize it in the same way the "magnifier"
+layout does it. Place it on the "current" screen (derived from the mouse
+position). This allows you to magnify any client you wish, regardless of
+the currently used layout. Use it with a client keybinding like this:
+
+```lua
+clientkeys = awful.util.table.join(
+ -- [...]
+ awful.key({ modkey, "Control" }, "m", lain.util.magnify_client),
+ -- [...]
+)
+```
+
+If you want to "de-magnify" it, just retype the keybinding.
+
+If you want magnified client to respond to `incmwfact`, read [here](https://github.com/lcpz/lain/issues/195).
+
+menu\_clients\_current\_tags
+----------------------------
+
+Similar to `awful.menu.clients`, but this menu only shows the clients
+of currently visible tags. Use it with a key binding like this:
+
+```lua
+awful.key({ "Mod1" }, "Tab", function()
+ lain.util.menu_clients_current_tags({ width = 350 }, { keygrabber = true })
+end),
+```
+
+menu\_iterator
+--------------
+
+A generic menu utility which enables iteration over lists of possible
+actions to execute. The perfect example is a menu for choosing what
+configuration to apply to X with `xrandr`, as suggested on the [Awesome wiki page](https://awesomewm.org/recipes/xrandr).
+
+
+
+
An example Synergy menu, courtesy of sim590
+
+
+You can either manually create a menu by defining a table in this format:
+
+```lua
+{ { "choice description 1", callbackFuction1 }, { "choice description 2", callbackFunction2 }, ... }
+```
+
+or use `util.menu_iterator.menu`. Once you have your menu, use it with `lain.menu_iterator.iterate`.
+
+### Input tables
+
+**lain.menu_iterator.iterate**
+
+| Argument | Description | Type
+|---|---| ---
+| `menu` | the menu to iterate on | table
+| `timeout` | time (in seconds) to wait on a choice before the choice is accepted | integer (default: 4)
+| `icon` | path to the icon to display in `naughty.notify` window | string
+
+**lain.menu_iterator.menu**
+
+| Argument | Description | Type
+|---|---| ---
+`choices` | list of choices (e.g., `{ "choice1", "choice2", ... }`) | array of strings
+`name` | name of the program related to this menu | string
+`selected_cb` | callback to execute for each selected choice, it takes one choice (string) as argument; can be `nil` (no action to execute) | function
+`rejected_cb` | callback to execute for all rejected choices (the remaining choices, once one is selected), it takes one choice (string) as argument; can be `nil` (no action to execute) | function
+`extra_choices` | more choices to be added to the menu; unlike `choices`, these ones won't trigger `rejected_cb` | array of `{ choice, callback }` pairs, where `choice` is a string and `callback` is a function
+`combination` | how choices have to be combined in the menu; possible values are: "single" (default), the set of possible choices will simply be the input set ; "powerset", the set of possible choices will be the [power set](https://en.wikipedia.org/wiki/Power_set) of the input set | string
+
+### Examples
+
+A simple example is:
+
+```lua
+local mymenu_iterable = lain.util.menu_iterator.menu {
+ choices = {"My first choice", "My second choice"},
+ name = "My awesome program",
+ selected_cb = function(choice)
+ -- do something with selected choice
+ end,
+ rejected_cb = function(choice)
+ -- do something with every rejected choice
+ end
+}
+```
+
+The variable `mymenu_iterable` is a menu compatible with the function `lain.util.menu_iterator.iterate`, which will iterate over it and displays notification with `naughty.notify` every time it is called. You can use it like this:
+
+```lua
+local confirm_timeout = 5 -- time to wait before confirming the menu selection
+local my_notify_icon = "/path/to/icon" -- the icon to display in the notification
+lain.util.menu_iterator.iterate(mymenu_iterable, confirm_timeout, my_notify_icon)
+```
+
+Once `confirm_timeout` has passed without anymore calls to `iterate`, the choice is made and the associated callbacks (both for selected and rejected choices) are spawned.
+
+A useful practice is to add a `Cancel` option as an extra choice for canceling a menu selection. Extending the above example:
+
+```lua
+local mymenu_iterable = lain.util.menu_iterator.menu {
+ choices = {"My first choice", "My second choice"},
+ name = "My awesome program",
+ selected_cb = function(choice)
+ -- do something with selected choice
+ end,
+ rejected_cb = function(choice)
+ -- do something with every rejected choice
+ end
+ -- nil means no action to do
+ extra_choices = { {"Cancel"}, nil }
+}
+```
diff --git a/Phranque/awesome/lain/wiki/Widgets.md b/Phranque/awesome/lain/wiki/Widgets.md
new file mode 100644
index 00000000..34f518df
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/Widgets.md
@@ -0,0 +1,55 @@
+# Usage
+
+Every lain widget is a table.
+
+A lain widget is generated by a `function`.
+
+The `function` signature, input and output arguments can be found in the [related wiki entry](https://github.com/lcpz/lain/wiki/Widgets#index).
+
+Every lain widget contains a `wibox.widget`, which is updated by a timed function. To access the widget, use the field `widget`, while to access the timed function, use the field `update`. Some lain widgets may also have an `icon` field, which is a `wibox.widget.imagebox`, and/or a `timer` field, which is the `gears.timer` on `update`.
+
+Every `function` may take either a table or a list of variables as input.
+
+If the input is a table, you must define a function variable called `settings` in it. There you will be able to define `widget` appearance.
+
+For instance, if `widget` is a textbox, to markup it call `widget:set_markup(...)` within `settings`.
+
+In the scope of `settings` you can use predefined arguments, which are specified in the wiki entries.
+
+Example of a lain widget:
+
+```lua
+local cpu = lain.widget.cpu {
+ settings = function()
+ widget:set_markup("Cpu " .. cpu_now.usage)
+ end
+}
+-- to access the widget: cpu.widget
+```
+
+If you want to see some applications, check [awesome-copycats](https://github.com/lcpz/awesome-copycats).
+
+# Index
+
+- [alsa](https://github.com/lcpz/lain/wiki/alsa)
+- [alsabar](https://github.com/lcpz/lain/wiki/alsabar)
+- [bat](https://github.com/lcpz/lain/wiki/bat)
+- [cal](https://github.com/lcpz/lain/wiki/cal)
+- [cpu](https://github.com/lcpz/lain/wiki/cpu)
+- [fs](https://github.com/lcpz/lain/wiki/fs)
+- [imap](https://github.com/lcpz/lain/wiki/imap)
+- [mem](https://github.com/lcpz/lain/wiki/mem)
+- [mpd](https://github.com/lcpz/lain/wiki/mpd)
+- [net](https://github.com/lcpz/lain/wiki/net)
+- [pulse](https://github.com/lcpz/lain/wiki/pulse)
+- [pulsebar](https://github.com/lcpz/lain/wiki/pulsebar)
+- [sysload](https://github.com/lcpz/lain/wiki/sysload)
+- [temp](https://github.com/lcpz/lain/wiki/temp)
+- [weather](https://github.com/lcpz/lain/wiki/weather)
+
+## Users contributed
+
+- [moc](https://github.com/lcpz/lain/wiki/moc)
+- [redshift](https://github.com/lcpz/lain/wiki/redshift)
+- [task](https://github.com/lcpz/lain/wiki/task)
+- [tp_smapi](https://github.com/lcpz/lain/wiki/tp_smapi)
diff --git a/Phranque/awesome/lain/wiki/_Footer.md b/Phranque/awesome/lain/wiki/_Footer.md
new file mode 100644
index 00000000..b64f13bf
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/_Footer.md
@@ -0,0 +1 @@
+[Home](https://github.com/lcpz/lain/wiki) • [Layouts](https://github.com/lcpz/lain/wiki/Layouts) • [Widgets](https://github.com/lcpz/lain/wiki/Widgets) • [Utilities](https://github.com/lcpz/lain/wiki/Utilities)
diff --git a/Phranque/awesome/lain/wiki/_Sidebar.md b/Phranque/awesome/lain/wiki/_Sidebar.md
new file mode 100644
index 00000000..0289783a
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/_Sidebar.md
@@ -0,0 +1,26 @@
+* [Home](https://github.com/lcpz/lain/wiki/Home)
+* [Layouts](https://github.com/lcpz/lain/wiki/Layouts)
+ * [Usage](https://github.com/lcpz/lain/wiki/Layouts#Usage)
+ * [How do layouts work?](https://github.com/lcpz/lain/wiki/Layouts#how-do-layouts-work)
+ * [termfair](https://github.com/lcpz/lain/wiki/Layouts#termfair)
+ * [centerfair](https://github.com/lcpz/lain/wiki/Layouts#termfaircenter)
+ * [cascade](https://github.com/lcpz/lain/wiki/Layouts#cascade)
+ * [cascadetile](https://github.com/lcpz/lain/wiki/Layouts#cascadetile)
+ * [centerwork](https://github.com/lcpz/lain/wiki/Layouts#centerwork)
+ * [centerworkh](https://github.com/lcpz/lain/wiki/Layouts#centerworkhorizontal)
+ * [Pre 4.0 uselesstile patches](https://github.com/lcpz/lain/wiki/Layouts#pre-40-uselesstile-patches)
+ * [What about layout icons?](https://github.com/lcpz/lain/wiki/Layouts#what-about-layout-icons)
+* [Widgets](https://github.com/lcpz/lain/wiki/Widgets)
+ * [Usage](https://github.com/lcpz/lain/wiki/Widgets#usage)
+ * [Index](https://github.com/lcpz/lain/wiki/Widgets#index)
+ * [Users contributed](https://github.com/lcpz/lain/wiki/Widgets#users-contributed)
+* [Utilities](https://github.com/lcpz/lain/wiki/Utilities)
+ * [quake](https://github.com/lcpz/lain/wiki/Utilities#quake)
+ * [separators](https://github.com/lcpz/lain/wiki/Utilities#separators)
+ * [markup](https://github.com/lcpz/lain/wiki/Utilities#markup)
+ * [dynamic tagging](https://github.com/lcpz/lain/wiki/Utilities#dynamic-tagging)
+ * [useless_gaps_resize](https://github.com/lcpz/lain/wiki/Utilities#useless-gaps-resize)
+ * [tag_view_non_empty](https://github.com/lcpz/lain/wiki/Utilities#tag_view_nonempty)
+ * [magnify_client](https://github.com/lcpz/lain/wiki/Utilities#magnify_client)
+ * [menu_clients_current_tags](https://github.com/lcpz/lain/wiki/Utilities#menu_clients_current_tags)
+ * [menu_iterator](https://github.com/lcpz/lain/wiki/Utilities#menu_iterator)
diff --git a/Phranque/awesome/lain/wiki/alsa.md b/Phranque/awesome/lain/wiki/alsa.md
new file mode 100644
index 00000000..6882fcb1
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/alsa.md
@@ -0,0 +1,136 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Shows ALSA volume.
+
+```lua
+local volume = lain.widget.alsa()
+```
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`timeout` | Refresh timeout (in seconds) | integer | 5
+`cmd` | Alsa mixer command | string | "amixer"
+`channel` | Mixer channel | string | "Master"
+`togglechannel` | Toggle channel | string | `nil`
+`settings` | User settings | function | empty function
+
+`cmd` is useful if you need to pass additional arguments to amixer. For instance, you may want to define `cmd = "amixer -c X"` in order to set amixer with card `X`.
+
+`settings` can use the following variables:
+
+Variable | Meaning | Type | Values
+--- | --- | --- | ---
+`volume_now.level` | Volume level | integer | 0-100
+`volume_now.status` | Device status | string | "on", "off"
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`widget` | The widget | `wibox.widget.textbox`
+`channel` | ALSA channel | string
+`update` | Update `widget` | function
+
+## Toggle channel
+
+In case mute toggling can't be mapped to master channel (this happens, for instance, when you are using an HDMI output), define togglechannel as your S/PDIF device. You can get the device ID with `scontents` command.
+
+For instance, if card number is 1 and S/PDIF number is 3:
+
+```shell
+$ amixer -c 1 scontents
+Simple mixer control 'Master',0
+ Capabilities: volume
+ Playback channels: Front Left - Front Right
+ Capture channels: Front Left - Front Right
+ Limits: 0 - 255
+ Front Left: 255 [100%]
+ Front Right: 255 [100%]
+Simple mixer control 'IEC958',0
+ Capabilities: pswitch pswitch-joined
+ Playback channels: Mono
+ Mono: Playback [on]
+Simple mixer control 'IEC958',1
+ Capabilities: pswitch pswitch-joined
+ Playback channels: Mono
+ Mono: Playback [on]
+Simple mixer control 'IEC958',2
+ Capabilities: pswitch pswitch-joined
+ Playback channels: Mono
+ Mono: Playback [on]
+Simple mixer control 'IEC958',3
+ Capabilities: pswitch pswitch-joined
+ Playback channels: Mono
+ Mono: Playback [on]
+```
+
+you have to set `togglechannel = "IEC958,3"`.
+
+## Buttons
+
+If you want buttons, just add the following after your widget in `rc.lua`.
+
+```lua
+volume.widget:buttons(awful.util.table.join(
+ awful.button({}, 1, function() -- left click
+ awful.spawn(string.format("%s -e alsamixer", terminal))
+ end),
+ awful.button({}, 2, function() -- middle click
+ os.execute(string.format("%s set %s 100%%", volume.cmd, volume.channel))
+ volume.update()
+ end),
+ awful.button({}, 3, function() -- right click
+ os.execute(string.format("%s set %s toggle", volume.cmd, volume.togglechannel or volume.channel))
+ volume.update()
+ end),
+ awful.button({}, 4, function() -- scroll up
+ os.execute(string.format("%s set %s 1%%+", volume.cmd, volume.channel))
+ volume.update()
+ end),
+ awful.button({}, 5, function() -- scroll down
+ os.execute(string.format("%s set %s 1%%-", volume.cmd, volume.channel))
+ volume.update()
+ end)
+))
+```
+
+## Keybindings
+
+You can control the widget with keybindings like these:
+
+```lua
+-- ALSA volume control
+awful.key({ altkey }, "Up",
+ function ()
+ os.execute(string.format("amixer set %s 1%%+", volume.channel))
+ volume.update()
+ end),
+awful.key({ altkey }, "Down",
+ function ()
+ os.execute(string.format("amixer set %s 1%%-", volume.channel))
+ volume.update()
+ end),
+awful.key({ altkey }, "m",
+ function ()
+ os.execute(string.format("amixer set %s toggle", volume.togglechannel or volume.channel))
+ volume.update()
+ end),
+awful.key({ altkey, "Control" }, "m",
+ function ()
+ os.execute(string.format("amixer set %s 100%%", volume.channel))
+ volume.update()
+ end),
+awful.key({ altkey, "Control" }, "0",
+ function ()
+ os.execute(string.format("amixer set %s 0%%", volume.channel))
+ volume.update()
+ end),
+```
+
+where `altkey = "Mod1"`.
diff --git a/Phranque/awesome/lain/wiki/alsabar.md b/Phranque/awesome/lain/wiki/alsabar.md
new file mode 100644
index 00000000..f16b14ff
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/alsabar.md
@@ -0,0 +1,102 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Shows ALSA volume with a progressbar; provides tooltips and notifications.
+
+```lua
+local volume = lain.widget.alsabar()
+```
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`timeout` | Refresh timeout (in seconds) | integer | 5
+`settings` | User settings | function | empty function
+`width` | Bar width | number | 63
+`height` | Bar height | number | 1
+`margins` | Bar margins | number | 1
+`paddings` | Bar paddings | number | 1
+`ticks` | Set bar ticks on | boolean | false
+`ticks_size` | Ticks size | integer | 7
+`cmd` | ALSA mixer command | string | "amixer"
+`channel` | Mixer channel | string | "Master"
+`togglechannel` | Toggle channel | string | `nil`
+`colors` | Bar colors | table | see [Default colors](https://github.com/lcpz/lain/wiki/alsabar#default-colors)
+`notification_preset` | Notification preset | table | See [default `notification_preset`](https://github.com/lcpz/lain/wiki/alsabar#default-notification_preset)
+`followtag` | Display the notification on currently focused screen | boolean | false
+
+`cmd` is useful if you need to pass additional arguments to `amixer`. For instance, you may want to define `cmd = "amixer -c X"` in order to set amixer with card `X`.
+
+In case mute toggling can't be mapped to master channel (this happens, for instance, when you are using an HDMI output), define `togglechannel` as your S/PDIF device. Read [`alsa`](https://github.com/lcpz/lain/wiki/alsa#toggle-channel) page to know how.
+
+`settings` can use the following variables:
+
+Variable | Meaning | Type | Values
+--- | --- | --- | ---
+`volume_now.level` | Volume level | integer | 0-100
+`volume_now.status` | Device status | string | "on", "off"
+
+With multiple screens, the default behaviour is to show a visual notification pop-up window on the first screen. By setting `followtag` to `true` it will be shown on the currently focused tag screen.
+
+### Default colors
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`background` | Bar backgrund color | string | "#000000"
+`mute` | Bar mute color | string | "#EB8F8F"
+`unmute` | Bar unmute color | string | "#A4CE8A"
+
+### Default `notification_preset`
+
+```lua
+notification_preset = {
+ font = "Monospace 10"
+}
+```
+
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`bar` | The widget | `wibox.widget.progressbar`
+`channel` | ALSA channel | string
+`notify` | The notification | function
+`update` | Update `bar` | function
+`tooltip` | The tooltip | `awful.tooltip`
+
+## Buttons
+
+If you want buttons, just add the following after your widget in `rc.lua`.
+
+```lua
+volume.bar:buttons(awful.util.table.join(
+ awful.button({}, 1, function() -- left click
+ awful.spawn(string.format("%s -e alsamixer", terminal))
+ end),
+ awful.button({}, 2, function() -- middle click
+ os.execute(string.format("%s set %s 100%%", volume.cmd, volume.channel))
+ volume.update()
+ end),
+ awful.button({}, 3, function() -- right click
+ os.execute(string.format("%s set %s toggle", volume.cmd, volume.togglechannel or volume.channel))
+ volume.update()
+ end),
+ awful.button({}, 4, function() -- scroll up
+ os.execute(string.format("%s set %s 1%%+", volume.cmd, volume.channel))
+ volume.update()
+ end),
+ awful.button({}, 5, function() -- scroll down
+ os.execute(string.format("%s set %s 1%%-", volume.cmd, volume.channel))
+ volume.update()
+ end)
+))
+```
+
+## Keybindings
+
+Read [here](https://github.com/lcpz/lain/wiki/alsa#keybindings). If you want notifications, use `volume.notify()` instead of `volume.update()`.
diff --git a/Phranque/awesome/lain/wiki/bat.md b/Phranque/awesome/lain/wiki/bat.md
new file mode 100644
index 00000000..b7d6cd1f
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/bat.md
@@ -0,0 +1,99 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Shows the remaining time and percentage capacity of your laptop battery, as well
+as the current wattage. Multiple batteries are supported.
+
+Displays a notification when battery is fully charged, low, or critical.
+
+```lua
+local mybattery = lain.widget.bat()
+```
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`timeout` | Refresh timeout (in seconds) | integer | 30
+`pspath` | Power supply directory path | string | "/sys/class/power_supply/"
+`battery` | Single battery id | string | autodetected
+`batteries` | Multiple batteries id table | table of strings | autodetected
+`ac` | AC | string | autodetected
+`notify` | Show notification popups | string | "on"
+`n_perc` | Percentages assumed for critical and low battery levels | table of integers | `{5, 15}`
+`settings` | User settings | function | empty function
+
+The widget will try to autodetect `battery`, `batteries` and `ac`. If something
+goes wrong, you will have to define them manually. In that case, you only have
+to define one between `battery` and `batteries`. If you have one battery, you
+can either use `args.battery = "BAT*"` or `args.batteries = {"BAT*"}`, where `BAT*`
+is the identifier of your battery in `pspath` (do not use it as a wildcard).
+Of course, if you have multiple batteries, you need to use the latter option.
+
+To disable notifications, set `notify` to `"off"`.
+
+If you define `pspath`, **be sure** to not forget the final slash (/).
+
+`settings` can use the `bat_now` table, which contains the following strings:
+
+- `status`, general status ("N/A", "Discharging", "Charging", "Full");
+- `n_status[i]`, i-th battery status (like above);
+- `ac_status`, AC-plug flag (0 if cable is unplugged, 1 if plugged, "N/A" otherwise);
+- `perc`, total charge percentage (integer between 0 and 100 or "N/A");
+- `n_perc[i]`, i-th battery charge percentage (like above);
+- `time`, time remaining until charge if charging, until discharge if discharging (HH:MM string or "N/A");
+- `watt`, battery watts (float with 2 decimals).
+
+and can modify the following three tables, which will be the preset for the naughty notifications:
+* `bat_notification_charged_preset` (used if battery is fully charged and connected to AC)
+* `bat_notification_low_preset` (used if battery charge level <= 15)
+* `bat_notification_critical_preset` (used if battery charge level <= 5)
+
+Check [here](https://awesomewm.org/doc/api/libraries/naughty.html#notify) for
+the list of variables they can contain. Default definitions:
+
+```lua
+bat_notification_charged_preset = {
+ title = "Battery full",
+ text = "You can unplug the cable",
+ timeout = 15,
+ fg = "#202020",
+ bg = "#CDCDCD"
+ }
+
+```
+
+```lua
+bat_notification_low_preset = {
+ title = "Battery low",
+ text = "Plug the cable!",
+ timeout = 15,
+ fg = "#202020",
+ bg = "#CDCDCD"
+}
+```
+```lua
+bat_notification_critical_preset = {
+ title = "Battery exhausted",
+ text = "Shutdown imminent",
+ timeout = 15,
+ fg = "#000000",
+ bg = "#FFFFFF"
+}
+```
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`widget` | The widget | `wibox.widget.textbox`
+`update` | Update `widget` | function
+
+The `update` function can be used to refresh the widget before `timeout` expires.
+
+## Note
+
+Alternatively, you can try the [`upower` recipe](https://awesomewm.org/recipes/watch).
diff --git a/Phranque/awesome/lain/wiki/cal.md b/Phranque/awesome/lain/wiki/cal.md
new file mode 100644
index 00000000..336b9297
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/cal.md
@@ -0,0 +1,81 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Creates a calendar notification that can be attached to widgets.
+
+This is a simpler but [faster](https://github.com/awesomeWM/awesome/issues/1861)
+alternative to [`awful.widget.calendar_popup`](https://awesomewm.org/doc/api/classes/awful.widget.calendar_popup.html), which emulates UNIX's `cal`.
+
+```lua
+local mycal = lain.widget.cal()
+```
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`attach_to` | List of widgets | table | empty table
+`week_start` | First day of the week | integer | 2 (Monday)
+`three` | Display three months spanning the date | boolean | false
+`followtag` | Display the notification on currently focused screen | boolean | false
+`icons` | Path to calendar icons | string | [icons/cal/white/](https://github.com/lcpz/lain/tree/master/icons/cal/white)
+`notification_preset` | Notification preset | table | See [default `notification_preset`](https://github.com/lcpz/lain/wiki/calendar#default-notification_preset)
+
+Set `attach_to` as the list of widgets to which you want to attach the calendar, like this:
+
+```lua
+local mycal = lain.widget.cal {
+ attach_to = { mywidget1, mywidget2, ... },
+ -- [...]
+}
+```
+
+For every widget in `attach_to`:
+
+- Left click / scroll down: switch to previous month.
+- Middle click show current month.
+- Right click / scroll up: switch to next month.
+
+With multiple screens, the default behaviour is to show a visual notification pop-up window on the first screen. By setting `followtag` to `true` it will be shown on the currently focused tag screen.
+
+### Default `notification_preset`
+
+```lua
+notification_preset = {
+ font = "Monospace 10",
+ fg = "#FFFFFF",
+ bg = "#000000"
+}
+```
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`attach` | Attach the calendar to an input widget | function
+`show` | Show calendar | function
+`hide` | Hide calendar | function
+
+`attach` takes as argument any widget you want to attach the calendar to, while
+`show` takes as optional argument an integer to indicate the seconds to timeout.
+
+## Keybinding
+
+```lua
+awful.key({ altkey }, "c", function () mycal.show(7) end)
+```
+
+Where `altkey = "Mod1"`.
+
+## Notes
+
+* Naughty notifications require `notification_preset.font` to be **monospaced**, in order to correctly display the output.
+* If you want to [disable notification icon](https://github.com/lcpz/lain/pull/351), set `icons = ""` in the input table.
+* If you want to localise the calendar, put `os.setlocale(os.getenv("LANG"))` in your `rc.lua`.
+* If you want to get notifications [only with mouse clicks](https://github.com/lcpz/lain/issues/320) on a given widget, use this code:
+ ```lua
+ yourwidget:disconnect_signal("mouse::enter", mycal.hover_on)
+ ```
diff --git a/Phranque/awesome/lain/wiki/cpu.md b/Phranque/awesome/lain/wiki/cpu.md
new file mode 100644
index 00000000..c13d05aa
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/cpu.md
@@ -0,0 +1,30 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Shows the current CPU usage, both in general and per core.
+
+```lua
+local mycpu = lain.widget.cpu()
+```
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`timeout` | Refresh timeout (in seconds) | integer | 2
+`settings` | User settings | function | empty function
+
+`settings` can use these strings:
+
+* `cpu_now.usage`, the general use percentage;
+* `cpu_now[i].usage`, the i-th core use percentage, with `i` starting from 1.
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`widget` | The widget | `wibox.widget.textbox`
+`update` | Update `widget` | function
\ No newline at end of file
diff --git a/Phranque/awesome/lain/wiki/fs.md b/Phranque/awesome/lain/wiki/fs.md
new file mode 100644
index 00000000..60a78422
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/fs.md
@@ -0,0 +1,81 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Shows file systems informations.
+
+If a partition is given in input, a notification will be displayed when it is almost full.
+
+```lua
+local mypartition = lain.widget.fs()
+```
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`timeout` | Refresh timeout (in seconds) | integer | 600
+`partition` | (Optional) Partition to watch: a notification will be displayed when full | string | `nil`
+`notification_preset` | Notification preset | table | See [default `notification_preset`](https://github.com/lcpz/lain/wiki/fs#default-notification_preset)
+`followtag` | Display the notification on currently focused screen | boolean | false
+`showpopup` | Display popups with mouse hovering | string, possible values: "on", "off" | "on"
+`settings` | User settings | function | empty function
+
+`settings` can use the table `fs_now`, which contains a string entry for each file system path available. For instance, root infos are located in the variable `fs_now["/"]`. Every entry in this table have the following variables:
+
+Variable | Meaning | Type
+--- | --- | ---
+`units` | (multiple of) units used | string ("Kb", "Mb", "Gb", and so on)
+`percentage` | the used percentage | integer
+`size` | size in `units` of the given fs | float
+`used` | amount of space used in the given fs, expressed in `units` | float
+`free` | amount of free space in the given fs, expressed in `units` | float
+
+Usage example:
+
+```lua
+-- shows used (percentage) and remaining space in home partition
+local fsroothome = lain.widget.fs({
+ settings = function()
+ widget:set_text("/home: " .. fs_now["/home"].percentage .. "% (" ..
+ fs_now["/home"].free .. " " .. fs_now["/home"].units .. " left)")
+ end
+})
+-- output example: "/home: 37% (239.4 Gb left)"
+```
+
+With multiple screens, the default behaviour is to show a visual notification pop-up window on the first screen. By setting `followtag` to `true` it will be shown on the currently focused tag screen.
+
+### Default `notification_preset`
+
+```lua
+notification_preset = {
+ font = "Monospace 10",
+ fg = "#FFFFFF",
+ bg = "#000000"
+}
+```
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`widget` | The widget | `wibox.widget.textbox`
+`show` | The notification | function
+
+You can display the notification with a key binding like this:
+
+```lua
+awful.key({ altkey }, "h", function () mypartition.show(seconds, scr) end),
+```
+
+where ``altkey = "Mod1"`` and ``show`` arguments, both optionals, are:
+
+* `seconds`, notification time in seconds
+* `scr`, screen which to display the notification in
+
+## Note
+
+Naughty notifications require `notification_preset.font` to be **monospaced**, in order to correctly display the output.
diff --git a/Phranque/awesome/lain/wiki/imap.md b/Phranque/awesome/lain/wiki/imap.md
new file mode 100644
index 00000000..683777ec
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/imap.md
@@ -0,0 +1,115 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Shows mails count fetching over IMAP.
+
+```lua
+local myimap = lain.widget.imap(args)
+```
+
+New mails are notified like this:
+
+ +--------------------------------------------+
+ | +---+ |
+ | |\ /| donald@disney.org has 3 new messages |
+ | +---+ |
+ +--------------------------------------------+
+
+## Input table
+
+Required parameters are:
+
+Variable | Meaning | Type
+--- | --- | ---
+`server` | Mail server | string
+`mail` | User mail | string
+`password` | User password | string
+
+while the optional are:
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`port` | IMAP port | integer | 993
+`timeout` | Refresh timeout (in seconds) | integer | 60
+`pwdtimeout` | Timeout for password retrieval function (see [here](https://github.com/lcpz/lain/wiki/imap#password-security)) | integer | 10
+`is_plain` | Define whether `password` is a plain password (true) or a command that retrieves it (false) | boolean | false
+`followtag` | Notification behaviour | boolean | false
+`notify` | Show notification popups | string | "on"
+`settings` | User settings | function | empty function
+
+`settings` can use `imap_now` table, which contains the following non negative integers:
+
+- `["MESSAGES"]`
+- `["RECENT"]`
+- `["UNSEEN"]`
+
+example of fetch: `total = imap_now["MESSAGES"]`. For backwards compatibility, `settings` can also use `mailcount`, a pointer to `imap_now["UNSEEN"]`.
+
+Also, `settings` can modify `mail_notification_preset` table, which will be the preset for the naughty notifications. Check [here](https://awesomewm.org/apidoc/libraries/naughty.html#notify) for the list of variables it can contain. Default definition:
+
+```lua
+mail_notification _preset = {
+ icon = "lain/icons/mail.png",
+ position = "top_left"
+}
+```
+
+Note that `mailcount` and `imap_now` elements are equals to 0 either if there are no new mails or credentials are invalid, so make sure that your settings are correct.
+
+With multiple screens, the default behaviour is to show a visual notification pop-up window on the first screen. By setting `followtag` to `true` it will be shown on the currently focused tag screen.
+
+You can have multiple instances of this widget at the same time.
+
+## Password security
+
+The reason why `is_plain` is false by default is to discourage the habit of storing passwords in plain.
+
+In general, when `is_plain == false`, `password` can be either a string, a table or a function: the widget will execute it asynchronously in the first two cases.
+
+### Using plain passwords
+
+You can set your password in plain like this:
+
+```lua
+myimapcheck = lain.widget.imap {
+ is_plain = true,
+ password = "mymailpassword",
+ -- [...]
+}
+```
+
+and you will have the same security provided by `~/.netrc`.
+
+### Using a password manager
+
+I recommend to use [spm](https://notabug.org/kl3/spm) or [pass](https://www.passwordstore.org). In this case, `password` has to be a function. Example stub:
+
+```lua
+myimapcheck = lain.widget.imap {
+ password = function()
+ -- do your retrieval
+ return retrieved_password, try_again
+ end,
+ -- [...]
+}
+```
+
+Where `retrieved_password` is the password retrieved from the manager, and `try_again` supports [DBus Secret Service](https://specifications.freedesktop.org/secret-service).
+
+The process flow is that the first `password()` call spawns the unlock prompt, then the second call retrieves the password. [Here](https://gist.github.com/lcpz/1854fc4320f4701957cd5309c8eed4a6) is an example of `password` function.
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`widget` | The widget | `wibox.widget.textbox`
+`update` | Update `widget` | function
+`timer` | The widget timer | [`gears.timer`](https://awesomewm.org/doc/api/classes/gears.timer.html)
+`pwdtimer` | Password retrieval timer (available only if `password` is a function)| [`gears.timer`](https://awesomewm.org/doc/api/classes/gears.timer.html)
+
+The `update` function can be used to refresh the widget before `timeout` expires.
+
+You can use `timer` to start/stop the widget as you like.
diff --git a/Phranque/awesome/lain/wiki/mem.md b/Phranque/awesome/lain/wiki/mem.md
new file mode 100644
index 00000000..205d535f
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/mem.md
@@ -0,0 +1,33 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Shows memory status in MiB, [like `top` and `free -h`](https://github.com/lcpz/lain/issues/271).
+
+```lua
+local mymem = lain.widget.mem()
+```
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`timeout` | Refresh timeout (in seconds) | integer | 2
+`settings` | User settings | function | empty function
+
+in `settings` you can use the following variables:
+
+Variable | Meaning | Type
+--- | --- | ---
+`mem_now.used` | Memory used (MiB) | string
+`mem_now.swapused` | Swap memory used (MiB) | string
+`mem_now.perc` | Memory percentage | int
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`widget` | The widget | `wibox.widget.textbox`
+`update` | Update `widget` | function
\ No newline at end of file
diff --git a/Phranque/awesome/lain/wiki/moc.md b/Phranque/awesome/lain/wiki/moc.md
new file mode 100644
index 00000000..130bccbb
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/moc.md
@@ -0,0 +1,122 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+A widget for showing the current song track's information from MOC (Music On Console).
+
+```lua
+local mymoc = lain.widget.contrib.moc()
+```
+
+Now playing songs are notified like this:
+
+ +--------------------------------------------------------+
+ | +-------+ |
+ | |/^\_/^\| Now playing |
+ | |\ O O /| Cannibal Corpse (Hammer Smashed Face) - 1993 |
+ | | '.o.' | Hammer Smashed Face (Radio Disney Version) |
+ | +-------+ |
+ +--------------------------------------------------------+
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`timeout` | Refresh timeout (in seconds) | integer | 1
+`music_dir` | Music directory | string | "~/Music"
+`cover_size` | Album art notification size (both height and width) | integer | 100
+`cover_pattern` | Pattern for the album art file | string | `*\\.(jpg|jpeg|png|gif)`*
+`default_art` | Default art | string | ""
+`followtag` | Display the notification on currently focused screen | boolean | false
+`settings` | User settings | function | empty function
+
+\* In Lua, "\\\\" means "\" escaped.
+
+Default `cover_pattern` definition will made the widget set the first jpg, jpeg, png or gif file found in the directory as the album art.
+
+Pay attention to case sensitivity when defining `music_dir`.
+
+`settings` can use `moc_now` table, which contains the following string values:
+
+- state (possible values: "PLAY", "PAUSE", "STOP")
+- file
+- artist
+- title
+- album
+- elapsed (Time elapsed for the current track)
+- total (The current track's total time)
+
+and can modify `moc_notification_preset` table, which will be the preset for the naughty notifications. Check [here](https://awesomewm.org/apidoc/libraries/naughty.html#notify) for the list of variables it can contain. Default definition:
+
+```lua
+moc_notification_preset = {
+ title = "Now playing",
+ timeout = 6,
+ text = string.format("%s (%s) - %s\n%s", moc_now.artist,
+ moc_now.album, moc_now.elapsed, moc_now.title)
+}
+```
+
+With multiple screens, the default behaviour is to show a visual notification pop-up window on the first screen. By setting `followtag` to `true` it will be shown on the currently focused tag screen.
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`widget` | The widget | `wibox.widget.textbox`
+`update` | Update `widget` | function
+`timer` | The widget timer | [`gears.timer`](https://awesomewm.org/doc/api/classes/gears.timer.html)
+
+The `update` function can be used to refresh the widget before `timeout` expires.
+
+You can use `timer` to start/stop the widget as you like.
+
+## Keybindings
+
+You can control the widget with key bindings like these:
+
+```lua
+-- MOC control
+awful.key({ altkey, "Control" }, "Up",
+ function ()
+ os.execute("mocp -G") -- toggle
+ moc.update()
+ end),
+awful.key({ altkey, "Control" }, "Down",
+ function ()
+ os.execute("mocp -s") -- stop
+ moc.update()
+ end),
+awful.key({ altkey, "Control" }, "Left",
+ function ()
+ os.execute("mocp -r") -- previous
+ moc.update()
+ end),
+awful.key({ altkey, "Control" }, "Right",
+ function ()
+ os.execute("mocp -f") -- next
+ moc.update()
+ end),
+```
+
+where `altkey = "Mod1"`.
+
+If you don't use the widget for long periods and wish to spare CPU, you can toggle it with a keybinding like this:
+
+```lua
+-- toggle MOC widget
+awful.key({ altkey }, "0",
+ function ()
+ local common = { text = "MOC widget ", position = "top_middle", timeout = 2 }
+ if moc.timer.started then
+ moc.timer:stop()
+ common.text = common.text .. markup.bold("OFF")
+ else
+ moc.timer:start()
+ common.text = common.text .. markup.bold("ON")
+ end
+ naughty.notify(common)
+ end),
+```
diff --git a/Phranque/awesome/lain/wiki/mpd.md b/Phranque/awesome/lain/wiki/mpd.md
new file mode 100644
index 00000000..3dae0143
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/mpd.md
@@ -0,0 +1,176 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Shows MPD status.
+
+```lua
+local mympd = lain.widget.mpd()
+```
+
+Now playing songs are notified like this:
+
+ +--------------------------------------------------------+
+ | +-------+ |
+ | |/^\_/^\| Now playing |
+ | |\ O O /| Cannibal Corpse (Hammer Smashed Face) - 1993 |
+ | | '.o.' | Hammer Smashed Face (Radio Disney Version) |
+ | +-------+ |
+ +--------------------------------------------------------+
+
+**Note:** if MPD is turned off or not set correctly, the widget will constantly display "N/A" values.
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`timeout` | Refresh timeout (in seconds) | integer | 2
+`password` | MPD password | string | ""
+`host` | MPD server | string | "127.0.0.1"
+`port` | MPD port | string | "6600"
+`music_dir` | Music directory | string | "~/Music"
+`cover_size` | Album art notification size (both height and width) | integer | 100
+`cover_pattern` | Pattern for the album art file | string | `*.(jpg\|jpeg\|png\|gif)$`
+`default_art` | Default art | string | `nil`
+`notify` | Show notification popups | string | "on"
+`followtag` | Notification behaviour | boolean | false
+`settings` | User settings | function | empty function
+
+\* In Lua, "\\\\" means "\" escaped.
+
+Default `cover_pattern` definition will made the widget set the first jpg, jpeg, png or gif file found in the directory as the album art.
+
+Pay attention to case sensitivity when defining `music_dir`.
+
+`settings` can use `mpd_now` table, which contains the following values:
+
+(**note:** the first four are boolean [flags](https://github.com/lcpz/lain/pull/205), the remaining are all strings)
+
+- random_mode
+- single_mode
+- repeat_mode
+- consume_mode
+- pls_pos (playlist position)
+- pls_len (playlist length)
+- state (possible values: "play", "pause", "stop")
+- file
+- artist
+- title
+- name
+- album
+- track
+- genre
+- date
+- [time](https://github.com/lcpz/lain/pull/90) (length of current song, in seconds)
+- [elapsed](https://github.com/lcpz/lain/pull/90) (elapsed time of current song, in seconds)
+
+and can modify `mpd_notification_preset` table, which will be the preset for the naughty notifications. Check [here](https://awesomewm.org/doc/api/libraries/naughty.html#notify) for the list of variables it can contain. Default definition:
+
+```lua
+mpd_notification_preset = {
+ title = "Now playing",
+ timeout = 6,
+ text = string.format("%s (%s) - %s\n%s", mpd_now.artist,
+ mpd_now.album, mpd_now.date, mpd_now.title)
+}
+```
+
+With multiple screens, the default behaviour is to show a visual notification pop-up window on the first screen. By setting `followtag` to `true` it will be shown on the currently focused tag screen.
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`widget` | The textbox | `wibox.widget.textbox`
+`update` | Update `widget` | function
+`timer` | The widget timer | [`gears.timer`](https://awesomewm.org/doc/api/classes/gears.timer.html)
+
+The `update` function can be used to refresh the widget before `timeout` expires.
+
+You can use `timer` to start/stop the widget as you like.
+
+## Keybindings
+
+You can control the widget with keybindings like these:
+
+```lua
+-- MPD control
+awful.key({ altkey, "Control" }, "Up",
+ function ()
+ awful.spawn.with_shell("mpc toggle || ncmpc toggle || pms toggle")
+ mympd.update()
+ end),
+awful.key({ altkey, "Control" }, "Down",
+ function ()
+ awful.spawn.with_shell("mpc stop || ncmpc stop || pms stop")
+ mympd.update()
+ end),
+awful.key({ altkey, "Control" }, "Left",
+ function ()
+ awful.spawn.with_shell("mpc prev || ncmpc prev || pms prev")
+ mympd.update()
+ end),
+awful.key({ altkey, "Control" }, "Right",
+ function ()
+ awful.spawn.with_shell("mpc next || ncmpc next || pms next")
+ mympd.update()
+ end),
+```
+
+where `altkey = "Mod1"`.
+
+If you don't use the widget for long periods and wish to spare CPU, you can toggle it with a keybinding like this:
+
+```lua
+-- disable MPD widget
+awful.key({ altkey }, "0",
+ function ()
+ local common = {
+ text = "MPD widget ",
+ position = "top_middle",
+ timeout = 2
+ }
+ if mympd.timer.started then
+ mympd.timer:stop()
+ common.text = common.text .. markup.bold("OFF")
+ else
+ mympd.timer:start()
+ common.text = common.text .. markup.bold("ON")
+ end
+ naughty.notify(common)
+ end),
+```
+
+## Notes
+
+### Cover not showing in notifications
+
+If the cover file is existent but not showed in notifications, [try](https://github.com/lcpz/lain/issues/393) setting `music_dir` to a symlink of your music folder, rather than to a physical path.
+
+### Always use `set_markup`
+
+In `settings`, if you use `widget:set_text`, [it will ignore Pango markup](https://github.com/lcpz/lain/issues/258), so be sure to always use `widget:set_markup`.
+
+### Volume fade in toggling MPD
+
+If you want a fade in/out in toggling MPD, you can put [this script](https://gist.github.com/lcpz/76e315bc27c6cdf7edd5021964b88df1) in your local `bin` directory:
+
+```shell
+$ curl https://gist.githubusercontent.com/lcpz/76e315bc27c6cdf7edd5021964b88df1/raw/97f7ba586418a4e07637cfbc91d2974278dfa623/mpd-fade -o ~/bin/mpc-fade
+$ chmod +x ~/bin/mpc-fade
+```
+
+Set your 1% decrease/increase commands [here](https://gist.github.com/lcpz/76e315bc27c6cdf7edd5021964b88df1#file-mpd-fade-L8-L9), then use a keybinding like this:
+
+```lua
+-- MPD toggle with volume fading
+awful.key({ "Shift" }, "Pause",
+ function()
+ awful.spawn.easy_async("mpc-fade 20 4", -- mpc-fade
+ function(stdout, stderr, reason, exit_code)
+ mympd.update()
+ end)
+ end),
+```
diff --git a/Phranque/awesome/lain/wiki/net.md b/Phranque/awesome/lain/wiki/net.md
new file mode 100644
index 00000000..252399ea
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/net.md
@@ -0,0 +1,115 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Monitors network interfaces and shows current traffic.
+
+```lua
+local mynet = lain.widget.net()
+```
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`timeout` | Refresh timeout (in seconds) | integer | 2
+`iface` | Network device(s) | string (single interface) or table of strings (multiple interfaces) | autodetected
+`units` | Units | integer | 1024 (kilobytes)
+`notify` | Display "no carrier" notifications | string | "on"
+`wifi_state` | Get wifi connection status | string | "off"
+`eth_state` | Get ethernet connection status | string | "off"
+`screen` | Notifications screen | integer | 1
+`settings` | User settings | function | empty function
+
+`iface` can be a string or an table of the form `{ "eth0", "eth1", ... }` containing a list of the devices to collect data on.
+
+If more than one device is included, `net_now.sent` and `net_now.received` will contain cumulative values over all given devices.
+Use `net_now.devices["eth0"]` to access `sent`, `received`, `state` or `carrier` per device.
+
+Possible alternative values for `units` are 1 (byte) or multiple of 1024: 1024^2 (mb), 1024^3 (gb), and so on.
+
+If `notify = "off"` is set, the widget won't display a notification when there's no carrier.
+
+`settings` can use the following `iface` related strings:
+
+- `net_now.carrier` ("0", "1");
+- `net_now.state` ("up", "down");
+- `net_now.sent` and `net_now.received` (numbers) will be the sum across all specified interfaces;
+- `net_now.devices["interface"]` contains the same attributes as the old api for each interface. More on this in the "Multiple devices" section below.
+
+If `wifi_state = "on"` is set, `settings` can use the following extra strings attached to `net_now.devices["wireless interface"]`:
+- `wifi` (true, false) indicates if the interface is connected to a network;
+- `signal` (number) is the connection signal strength in dBm;
+
+If `eth_state = "on"` is set, `settings` can use the following extra string: `net_now.devices["ethernet interface"].ethernet`, which is a boolean indicating if an ethernet connection's active.
+
+For compatibility reasons, if multiple devices are given, `net_now.carrier` and `net_now.state` correspond to the last interface in the iface table and should not be relied upon (deprecated).
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`widget` | The widget | `wibox.widget.textbox`
+`update` | Update `widget` | function
+
+## Notes
+
+### Setting `iface` manually
+
+If the widget [spawns a "no carrier" notification and you are sure to have an active network device](https://github.com/lcpz/lain/issues/102), then autodetection is probably not working. This may due to [your user privileges](https://github.com/lcpz/lain/issues/102#issuecomment-246470526). In this case you can set `iface` manually. You can see which device is **UP,LOWER_UP** with the following command:
+
+```shell
+ip link show
+```
+## Usage examples
+### Two widgets for upload/download rates from the same `iface`
+
+```lua
+local mynetdown = wibox.widget.textbox()
+local mynetup = lain.widget.net {
+ settings = function()
+ widget:set_markup(net_now.sent)
+ netdowninfo:set_markup(net_now.received)
+ end
+}
+```
+### Wifi connection + signal strength indicator and ethernet connection indicator
+```lua
+local wifi_icon = wibox.widget.imagebox()
+local eth_icon = wibox.widget.imagebox()
+local net = lain.widget.net {
+ notify = "off",
+ wifi_state = "on",
+ eth_state = "on",
+ settings = function()
+ local eth0 = net_now.devices.eth0
+ if eth0 then
+ if eth0.ethernet then
+ eth_icon:set_image(ethernet_icon_filename)
+ else
+ eth_icon:set_image()
+ end
+ end
+
+ local wlan0 = net_now.devices.wlan0
+ if wlan0 then
+ if wlan0.wifi then
+ local signal = wlan0.signal
+ if signal < -83 then
+ wifi_icon:set_image(wifi_weak_filename)
+ elseif signal < -70 then
+ wifi_icon:set_image(wifi_mid_filename)
+ elseif signal < -53 then
+ wifi_icon:set_image(wifi_good_filename)
+ elseif signal >= -53 then
+ wifi_icon:set_image(wifi_great_filename)
+ end
+ else
+ wifi_icon:set_image()
+ end
+ end
+ end
+}
+```
diff --git a/Phranque/awesome/lain/wiki/pulse.md b/Phranque/awesome/lain/wiki/pulse.md
new file mode 100644
index 00000000..311b4743
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/pulse.md
@@ -0,0 +1,135 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Shows and controls PulseAudio volume.
+
+```lua
+local volume = lain.widget.pulse()
+```
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`timeout` | Refresh timeout (in seconds) | integer | 5
+`devicetype` | PulseAudio device type | string ("sink", "source") | "sink"
+`cmd` | PulseAudio command | string or function | see [here](https://github.com/lcpz/lain/blob/master/widget/pulse.lua#L26)
+`settings` | User settings | function | empty function
+
+`cmd` is a terminal command to catch infos from current default device. You can redefine it, being sure that the ouput is something like this:
+
+```shell
+* index: 0
+ volume: front-left: 18340 / 28% / -33.18 dB, front-right: 18340 / 28% / -33.18 dB
+ muted: no
+ device.string = "front:1"
+```
+
+If your devices change dynamically, you can define it as a function which returns a command string.
+
+If sed doesn't work, you can try with a grep variant:
+
+```lua
+cmd = "pacmd list-" .. pulse.devicetype .. "s | grep -e $(pactl info | grep -e 'ink' | cut -d' ' -f3) -e 'volume: front' -e 'muted'"
+```
+
+### `settings` variables
+
+`settings` can use the following variables:
+
+Variable | Meaning | Type | Values
+--- | --- | --- | ---
+`volume_now.device` | Device name | string | device name or "N/A"
+`volume_now.index` | Device index | string | >= "0"
+`volume_now.muted` | Device mute status | string | "yes", "no", "N/A"
+`volume_now.channel` | Device channels | table of string integers | `volume_now.channel[i]`, where `i >= 1`
+`volume_now.left` | Front left sink level or first source | string | "0"-"100"
+`volume_now.right` | Front right sink level or second source | string | "0"-"100"
+
+`volume_now.channel` is a table of your PulseAudio devices. Fetch a channel level like this: `volume_now.channel[i]`, where `i >= 1`.
+
+`volume_now.{left,right}` are pointers for `volume_now.{channel[1], channel[2]}` (stereo).
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`widget` | The widget | `wibox.widget.textbox`
+`update` | Update `widget` | function
+
+## Buttons
+
+```lua
+volume.widget:buttons(awful.util.table.join(
+ awful.button({}, 1, function() -- left click
+ awful.spawn("pavucontrol")
+ end),
+ awful.button({}, 2, function() -- middle click
+ os.execute(string.format("pactl set-sink-volume %d 100%%", volume.device))
+ volume.update()
+ end),
+ awful.button({}, 3, function() -- right click
+ os.execute(string.format("pactl set-sink-mute %d toggle", volume.device))
+ volume.update()
+ end),
+ awful.button({}, 4, function() -- scroll up
+ os.execute(string.format("pactl set-sink-volume %d +1%%", volume.device))
+ volume.update()
+ end),
+ awful.button({}, 5, function() -- scroll down
+ os.execute(string.format("pactl set-sink-volume %d -1%%", volume.device))
+ volume.update()
+ end)
+))
+```
+
+## Keybindings
+
+```lua
+-- PulseAudio volume control
+awful.key({ altkey }, "Up",
+ function ()
+ os.execute(string.format("pactl set-sink-volume %d +1%%", volume.device))
+ volume.update()
+ end),
+awful.key({ altkey }, "Down",
+ function ()
+ os.execute(string.format("pactl set-sink-volume %d -1%%", volume.device))
+ volume.update()
+ end),
+awful.key({ altkey }, "m",
+ function ()
+ os.execute(string.format("pactl set-sink-mute %d toggle", volume.device))
+ volume.update()
+ end),
+awful.key({ altkey, "Control" }, "m",
+ function ()
+ os.execute(string.format("pactl set-sink-volume %d 100%%", volume.device))
+ volume.update()
+ end),
+awful.key({ altkey, "Control" }, "0",
+ function ()
+ os.execute(string.format("pactl set-sink-volume %d 0%%", volume.device))
+ volume.update()
+ end),
+```
+
+where `altkey = "Mod1"`.
+
+## Example
+
+```lua
+-- PulseAudio volume (based on multicolor theme)
+local volume = lain.widget.pulse {
+ settings = function()
+ vlevel = volume_now.left .. "-" .. volume_now.right .. "% | " .. volume_now.device
+ if volume_now.muted == "yes" then
+ vlevel = vlevel .. " M"
+ end
+ widget:set_markup(lain.util.markup("#7493d2", vlevel))
+ end
+}
+```
diff --git a/Phranque/awesome/lain/wiki/pulseaudio.md b/Phranque/awesome/lain/wiki/pulseaudio.md
new file mode 100644
index 00000000..6c11cb4e
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/pulseaudio.md
@@ -0,0 +1,150 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Shows and controls PulseAudio volume.
+
+```lua
+local volume = lain.widget.pulseaudio()
+```
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`timeout` | Refresh timeout seconds | number | 5
+`devicetype | PulseAudio device type | string ("sink", "source") | "sink"
+`cmd` | PulseAudio command | string | [link](https://github.com/lcpz/lain/blob/master/widget/pulseaudio.lua#L28)
+`scallback` | PulseAudio sink callback | function | `nil`
+`settings` | User settings | function | empty function
+
+`cmd` catch infos from current default sink. You can redefine it, being sure that the ouput is something like this:
+
+```shell
+* index: 0
+ volume: front-left: 18340 / 28% / -33.18 dB, front-right: 18340 / 28% / -33.18 dB
+ muted: no
+ device.string = "front:1"
+```
+
+**Note:** you can set PulseAudio default sink like this: `pacmd set-default-sink #sink`.
+
+If [`sed`](https://github.com/lcpz/lain/blob/master/widget/pulseaudio.lua#L28) doesn't work, you can try with `grep`:
+
+```shell
+pacmd list-sinks | grep -e $(pactl info | grep -e 'ink' | cut -d' ' -f3) -e 'volume: front' -e 'muted'
+```
+
+`scallback` is a callback function to update `cmd`, in case you switch between audio channels and therefore PulseAudio sink changes. If default `cmd` works for you, you can tell `scallback` to work in the same way:
+
+```lua
+scallback = function()
+ devicetype = "sink"
+ return "pacmd list-" .. devicetype .. "s | sed -n -e '0,/*/d' -e '/base volume/d' -e '/volume:/p' -e '/muted:/p' -e '/device\\.string/p'"
+end
+```
+
+### `settings` variables
+
+`settings` can use the following variables:
+
+Variable | Meaning | Type | Values
+--- | --- | --- | ---
+`volume_now.index` | Sink index | string | >= "0"
+`volume_now.sink` | Sink name | string | sink name or "N/A"
+`volume_now.muted` | Sink mute status | string | "yes", "no", "N/A"
+`volume_now.channel` | Sink channels | table of string integers | `volume_now.channel[i]`, where `i >= 1`
+`volume_now.left` | Front left level | string | "0"-"100"
+`volume_now.right` | Front right level | string | "0"-"100"
+
+`volume_now.channel` is a table of your pulseaudio sink channels. Fetch a channel level like this: `volume_now.channel[i]`, where `i >= 1`.
+
+`volume_now.{left,right}` are pointers for `volume_now.{channel[1], channel[2]}` (stereo).
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`widget` | The widget | `wibox.widget.textbox`
+`update` | Update `widget` | function
+
+## Buttons
+
+If you want buttons, just add the following after your widget in `rc.lua`.
+
+```lua
+volume.widget:buttons(awful.util.table.join(
+ awful.button({}, 1, function() -- left click
+ awful.spawn("pavucontrol")
+ end),
+ awful.button({}, 2, function() -- middle click
+ awful.spawn(string.format("pactl set-sink-volume %d 100%%", volume.sink))
+ volume.update()
+ end),
+ awful.button({}, 3, function() -- right click
+ awful.spawn(string.format("pactl set-sink-mute %d toggle", volume.sink))
+ volume.update()
+ end),
+ awful.button({}, 4, function() -- scroll up
+ awful.spawn(string.format("pactl set-sink-volume %d +1%%", volume.sink))
+ volume.update()
+ end),
+ awful.button({}, 5, function() -- scroll down
+ awful.spawn(string.format("pactl set-sink-volume %d -1%%", volume.sink))
+ volume.update()
+ end)
+))
+```
+
+## Keybindings
+
+You can control the widget with key bindings like these:
+
+```lua
+-- PulseAudio volume control
+awful.key({ altkey }, "Up",
+ function ()
+ os.execute(string.format("pactl set-sink-volume %d +1%%", volumewidget.sink))
+ volume.update()
+ end),
+awful.key({ altkey }, "Down",
+ function ()
+ os.execute(string.format("pactl set-sink-volume %d -1%%", volumewidget.sink))
+ volume.update()
+ end),
+awful.key({ altkey }, "m",
+ function ()
+ os.execute(string.format("pactl set-sink-mute %d toggle", volumewidget.sink))
+ volume.update()
+ end),
+awful.key({ altkey, "Control" }, "m",
+ function ()
+ os.execute(string.format("pactl set-sink-volume %d 100%%", volume.sink))
+ volume.update()
+ end),
+awful.key({ altkey, "Control" }, "0",
+ function ()
+ os.execute(string.format("pactl set-sink-volume %d 0%%", volume.sink))
+ volume.update()
+ end),
+```
+
+where `altkey = "Mod1"`.
+
+## Example
+
+```lua
+-- PulseAudio volume (based on multicolor theme)
+local volume = lain.widget.pulseaudio({
+ settings = function()
+ vlevel = volume_now.left .. "-" .. volume_now.right .. "% | " .. volume_now.sink
+ if volume_now.muted == "yes" then
+ vlevel = vlevel .. " M"
+ end
+
+ widget:set_markup(lain.util.markup("#7493d2", vlevel))
+ end
+})
+```
\ No newline at end of file
diff --git a/Phranque/awesome/lain/wiki/pulsebar.md b/Phranque/awesome/lain/wiki/pulsebar.md
new file mode 100644
index 00000000..61a86cdd
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/pulsebar.md
@@ -0,0 +1,94 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Shows PulseAudio volume with a progressbar; provides tooltips and notifications.
+
+```lua
+local volume = lain.widget.pulsebar()
+```
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`timeout` | Refresh timeout (in seconds) | integer | 5
+`settings` | User settings | function | empty function
+`width` | Bar width | number | 63
+`height` | Bar height | number | 1
+`margins` | Bar margins | number | 1
+`paddings` | Bar paddings | number | 1
+`ticks` | Set bar ticks on | boolean | false
+`ticks_size` | Ticks size | number | 7
+`scallback` | [PulseAudio sink callback](https://github.com/lcpz/lain/wiki/pulseaudio/) | function | `nil`
+`sink` | Mixer sink | number | 0
+`colors` | Bar colors | table | see [Default colors](https://github.com/lcpz/lain/wiki/pulsebar#default-colors)
+`notification_preset` | Notification preset | table | See [default `notification_preset`](https://github.com/lcpz/lain/wiki/pulsebar#default-notification_preset)
+`followtag` | Display the notification on currently focused screen | boolean | false
+`notification_preset` | Notification preset | table | See [default `notification_preset`](https://github.com/lcpz/lain/wiki/pulsebar#default-notification_preset)
+`devicetype` | PulseAudio device type | string ("sink", "source") | "sink"
+`cmd` | PulseAudio command | string or function | see [here](https://github.com/lcpz/lain/blob/master/widget/pulsebar.lua#L48)
+
+Read [pulse](https://github.com/lcpz/lain/wiki/pulse) page for `cmd` settings.
+
+`settings` can use [these variables](https://github.com/lcpz/lain/wiki/pulse#settings-variables).
+
+With multiple screens, the default behaviour is to show a visual notification pop-up window on the first screen. By setting `followtag` to `true` it will be shown on the currently focused tag screen.
+
+### Default colors
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`background` | Bar backgrund color | string | "#000000"
+`mute` | Bar mute color | string | "#EB8F8F"
+`unmute` | Bar unmute color | string | "#A4CE8A"
+
+### Default `notification_preset`
+
+```lua
+notification_preset = {
+ font = "Monospace 10"
+}
+```
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`bar` | The widget | `wibox.widget.progressbar`
+`device` | PulseAudio device | string
+`notify` | The notification | function
+`update` | Update state | function
+`tooltip` | The tooltip | `awful.tooltip`
+
+## Buttons
+
+```lua
+volume.bar:buttons(awful.util.table.join(
+ awful.button({}, 1, function() -- left click
+ awful.spawn("pavucontrol")
+ end),
+ awful.button({}, 2, function() -- middle click
+ os.execute(string.format("pactl set-sink-volume %d 100%%", volume.device))
+ volume.update()
+ end),
+ awful.button({}, 3, function() -- right click
+ os.execute(string.format("pactl set-sink-mute %d toggle", volume.device))
+ volume.update()
+ end),
+ awful.button({}, 4, function() -- scroll up
+ os.execute(string.format("pactl set-sink-volume %d +1%%", volume.device))
+ volume.update()
+ end),
+ awful.button({}, 5, function() -- scroll down
+ os.execute(string.format("pactl set-sink-volume %d -1%%", volume.device))
+ volume.update()
+ end)
+))
+```
+
+## Keybindings
+
+Same as [here](https://github.com/lcpz/lain/wiki/pulse#keybindings). If you want notifications, use `volume.notify()` instead of `volume.update()`.
diff --git a/Phranque/awesome/lain/wiki/redshift.md b/Phranque/awesome/lain/wiki/redshift.md
new file mode 100644
index 00000000..b83ad3a7
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/redshift.md
@@ -0,0 +1,100 @@
+### What is Redshift? #
+
+[**Project homepage**](http://jonls.dk/redshift/)
+
+>**Redshift** is an application that adjusts the computer display's color temperature based upon the Sun's apparent position in relation to the user's location on Earth.
+>
+>The program is free software, inspired by the proprietary f.lux, and can be used to reduce eye strain as well as insomnia and delayed sleep phase syndrome.
+>
+>The computer display's color temperature transitions evenly from night to daytime temperature to allow the user's eyes to slowly adapt. At night, the color temperature is low and is typically 3000–4000 K (default is 3500 K), preferably matching the room's lighting temperature. Typical color temperature during the daytime is 5500–6500 K (default is 5500 K).
+
+**Source:** [Wikipedia](https://en.wikipedia.org/wiki/Redshift_%28software%29)
+
+### Preparations
+
+**Redshift must be installed** on your system if you want to use this widget.
+
+Packages should be available for most distributions. Source code and build instructions can be found on Github [here](https://github.com/jonls/redshift).
+
+You also need a valid config file. Please see the [project homepage](http://jonls.dk/redshift/) for details. An example: [`~/.config/redshift.conf`](https://github.com/jonls/redshift/blob/master/redshift.conf.sample).
+
+You have to match the location settings to your personal situation: you can adjust the `lat` and `lon` variables using a [web service](https://encrypted.google.com/search?q=get+latitude+and+longitude).
+
+You might also want to modify the color temperatures to fit your preferences.
+
+### Using the widget
+
+This widget provides the following functions:
+
+| function | meaning |
+| --- | --- |
+| `redshift:toggle()` | Toggles Redshift adjustments on or off, and also restarts it if terminates. |
+| `redshift:attach(widget, update_function)` | Attach to a widget. Click on the widget to toggle redshift on or off. `update_function` is a callback function which will be triggered each time Redshift changes its status. (See the examples below.) |
+
+### Usage examples
+
+#### Textbox status widget
+
+```lua
+myredshift = wibox.widget.textbox()
+lain.widget.contrib.redshift:attach(
+ myredshift,
+ function (active)
+ if active then
+ myredshift:set_text("RS on")
+ else
+ myredshift:set_text("RS off")
+ end
+ end
+)
+```
+
+Then add `myredshift.widget` to your wibox.
+
+#### Checkbox status widget
+
+```lua
+local markup = lain.util.markup
+
+local myredshift = wibox.widget{
+ checked = false,
+ check_color = "#EB8F8F",
+ border_color = "#EB8F8F",
+ border_width = 1,
+ shape = gears.shape.square,
+ widget = wibox.widget.checkbox
+}
+
+local myredshift_text = wibox.widget{
+ align = "center",
+ widget = wibox.widget.textbox,
+}
+
+local myredshift_stack = wibox.widget{
+ myredshift,
+ myredshift_text,
+ layout = wibox.layout.stack
+}
+
+lain.widget.contrib.redshift:attach(
+ myredshift,
+ function (active)
+ if active then
+ myredshift_text:set_markup(markup(beautiful.bg_normal, "R"))
+ else
+ myredshift_text:set_markup(markup(beautiful.fg_normal, "R"))
+ end
+ myredshift.checked = active
+ end
+)
+```
+
+Then add the `myredshift_stack` widget to your wibox.
+
+#### Keybinding
+
+Add this to the keybindings in your `rc.lua`:
+```lua
+-- Toggle redshift with Mod+Shift+t
+awful.key({ modkey, "Shift" }, "t", function () lain.widget.contrib.redshift:toggle() end),
+```
\ No newline at end of file
diff --git a/Phranque/awesome/lain/wiki/sysload.md b/Phranque/awesome/lain/wiki/sysload.md
new file mode 100644
index 00000000..6fb1c04f
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/sysload.md
@@ -0,0 +1,27 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Shows the current system load.
+
+```lua
+mysysload = lain.widget.sysload()
+```
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`timeout` | Refresh timeout (in seconds) | integer | 2
+`settings` | User settings | function | empty function
+
+`settings` can use strings `load_1`, `load_5` and `load_15`, which are the load averages over 1, 5, and 15 minutes.
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`widget` | The widget | `wibox.widget.textbox`
+`update` | Update `widget` | function
\ No newline at end of file
diff --git a/Phranque/awesome/lain/wiki/task.md b/Phranque/awesome/lain/wiki/task.md
new file mode 100644
index 00000000..af53a6a3
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/task.md
@@ -0,0 +1,51 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Attaches a [taskwarrior](http://taskwarrior.org) notifications to a widget, and lets you execute `task` commands from the promptbox.
+
+```lua
+lain.widget.contrib.task.attach(widget, args)
+```
+
+`args` is an optional table which can contain:
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`show_cmd` | Taskwarrior command to show in the popup | string | "task next"
+`prompt_text` | Prompt text | string | "Enter task command: "
+`followtag` | Display the notification on currently focused screen | boolean | false
+`notification_preset` | Notification preset | table | See [default `notification_preset`](https://github.com/lcpz/lain/wiki/task#default-notification_preset)
+
+The tasks are shown in a notification popup when the mouse is moved over the attached `widget`, and the popup is hidden when the mouse is moved away. By default, the notification will show the output of `task next`. With `show_cmd`, the `task` popup command can be customized, for example if you want to [filter the tasks](https://taskwarrior.org/docs/filter.html) or show a [custom report](https://github.com/lcpz/lain/pull/213).
+
+With multiple screens, the default behaviour is to show a visual notification pop-up window on the first screen. By setting `followtag` to `true` it will be shown on the currently focused tag screen.
+
+You can call the notification with a keybinding like this:
+
+```lua
+awful.key({ modkey, altkey }, "t", function () lain.widget.contrib.task.show(scr) end),
+```
+
+where ``altkey = "Mod1"`` and `scr` (optional) indicates the screen which you want the notification in.
+
+And you can prompt to input a `task` command with a keybinding like this:
+
+```lua
+awful.key({ altkey }, "t", lain.widget.contrib.task.prompt),
+```
+
+### Default `notification_preset`
+
+```lua
+notification_preset = {
+ font = "Monospace 10",
+ icon = helpers.icons_dir .. "/taskwarrior.png"
+}
+```
+
+## Note
+
+* If your widget does not display `task next` output, try changing Taskwarrior verbose, for instance: `show_cmd = 'task rc.verbose:label'` or `show_cmd = 'task rc.verbose:nothing'`.
\ No newline at end of file
diff --git a/Phranque/awesome/lain/wiki/temp.md b/Phranque/awesome/lain/wiki/temp.md
new file mode 100644
index 00000000..3857d927
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/temp.md
@@ -0,0 +1,32 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Shows the current CPU temperature.
+
+```lua
+local mytemp = lain.widget.temp()
+```
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`timeout` | Refresh timeout (in seconds) | integer | 2
+`tempfile` | Path of file which stores core temperature value | string | "/sys/class/thermal/thermal_zone0/temp"
+`settings` | User settings | function | empty function
+
+`settings` can use the string `coretemp_now`, which means current core temperature, expressed in Celsius (linux standard).
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`widget` | The widget | `wibox.widget.textbox`
+`update` | Update `widget` | function
+
+## Note
+
+Depending on the architecture, note that your temp files location [might vary](https://github.com/lcpz/lain/issues/84#issuecomment-72751763).
\ No newline at end of file
diff --git a/Phranque/awesome/lain/wiki/tp_smapi.md b/Phranque/awesome/lain/wiki/tp_smapi.md
new file mode 100644
index 00000000..edb108fa
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/tp_smapi.md
@@ -0,0 +1,103 @@
+# Description
+
+[`tp_smapi`](http://www.thinkwiki.org/wiki/Tp_smapi) interface and widget creator.
+
+```lua
+local tp_smapi = lain.widget.contrib.tp_smapi(apipath)
+```
+
+The argument `apipath` is an optional string which defines the API path. Its default value is `"/sys/devices/platform/smapi"`.
+
+# Functions
+
+## tp_smapi.get(batid, feature)
+
+Gets the `feature` of battery `batid`. Returns a string. The list of available features is available at [this page](https://www.thinkwiki.org/wiki/Tp_smapi#Battery_status_features).
+
+## tp_smapi.installed(batid)
+
+Checks if battery `batid` is installed. Returns a boolean.
+
+## tp_smapi.status(batid)
+
+Gets the status of battery `batid`. Returns a string ("charging", "discharging", or "full").
+
+## tp_smapi.percentage(batid)
+
+Gets the percentage of battery `batid`. Returns a numeric string.
+
+## tp_smapi.time(batid)
+
+Gets the time of battery `batid`. Depending on the current status, it can be either running or charging time. Returns a string of the format `HH:MM`.
+
+## tp_smapi.hide()
+
+Removes any notification spawned by `tp_smapi.show`.
+
+## tp_smapi.show(batid, seconds, scr)
+
+Notifies the current information of battery `batid` for `seconds` seconds on screen `scr`.
+The argument `scr` is optional, and if missing, the notification will be displayed on the currently focused screen.
+
+## tp_smapi.create_widget(args)
+
+Creates a [lain widget](https://github.com/lcpz/lain/wiki/Widgets#usage) of the available ThinkPad batteries.
+
+```lua
+local tpbat = tp_smapi.create_widget()
+```
+
+### Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`widget` | The widget type to use | [`wibox.widget`](https://awesomewm.org/doc/api/classes/wibox.widget.html) | [`wibox.widget.textbox`](https://awesomewm.org/doc/api/classes/wibox.widget.textbox.html)
+`timeout` | Refresh timeout (in seconds) | integer | 30
+`pspath` | Power supply directory path | string | "/sys/class/power_supply/"
+`battery` | Single battery id | string | autodetected
+`batteries` | Multiple batteries id table | table of strings | autodetected
+`settings` | User settings | function | empty function
+
+The widget will try to autodetect `battery` and `batteries`. If something
+goes wrong, you will have to define them manually. In that case, you only have
+to define one between `battery` and `batteries`. If you have one battery, you
+can either use `args.battery = "BAT*"` or `args.batteries = {"BAT*"}`, where `BAT*`
+is the identifier of your battery in `pspath` (do not use it as a wildcard).
+Of course, if you have multiple batteries, you need to use the latter option.
+
+If you define `pspath`, **be sure** to not forget the final slash (/).
+
+`settings` can use the `tpbat_now` table, which contains the following strings:
+
+- `status`, general status ("N/A", "discharging", "charging", "full");
+- `n_status[i]`, i-th battery status (like above);
+- `n_perc[i]`, i-th battery charge percentage (like above);
+- `n_time[i]`, i-th battery running or charging time (HH:MM string or "N/A");
+
+`n_time[i]` is the running time of battery `i` when it is discharging, and the charging time otherwise.
+
+### Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`widget` | The widget | [`wibox.widget`](https://awesomewm.org/doc/api/classes/wibox.widget.html) | [textbox](https://awesomewm.org/doc/api/classes/wibox.widget.textbox.html)
+`batteries` | Battery identifiers | Table of strings
+`update` | Update `widget` | function
+`timer` | The widget timer | [`gears.timer`](https://awesomewm.org/doc/api/classes/gears.timer.html)
+
+The `update` function can be used to refresh the widget before `timeout` expires.
+
+### Usage example
+
+```lua
+local tp_smapi = lain.widget.contrib.tp_smapi()
+local bat = tp_smapi.create_widget {
+ battery = "BAT0",
+ settings = function()
+ widget:set_markup(tpbat_now.n_perc[1] .. "%")
+ end
+}
+
+bat.widget:connect_signal("mouse::enter", function () tp_smapi.show("BAT0") end)
+bat.widget:connect_signal("mouse::leave", function () tp_smapi.hide() end)
+```
diff --git a/Phranque/awesome/lain/wiki/watch.md b/Phranque/awesome/lain/wiki/watch.md
new file mode 100644
index 00000000..ff18a5ca
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/watch.md
@@ -0,0 +1,222 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Template for asynchronous watcher widgets.
+
+Executes an input command and makes the user feed a `wibox.widget` with the output.
+
+```lua
+local mywatch = lain.widget.watch()
+```
+
+This has been implemented in Awesome 4.0 as [`awful.widget.watch`](https://awesomewm.org/doc/api/classes/awful.widget.watch.html). But while Awesome `watch` returns only the widget, Lain one returns a table including its timer and internal update function too.
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`widget` | Widget to feed | `wibox.widget` | `wibox.widget.textbox`
+`timeout` | Refresh timeout seconds | number | 5
+`cmd` | The command to execute | string **or** table | `nil`
+`nostart` | Widget timer doesn't start immediately | boolean | false
+`stoppable` | Widget timer is stoppable | boolean | false
+`settings` | User settings | function | see [Default `settings` function](https://github.com/lcpz/lain/wiki/watch#default-settings-function)
+
+If your command needs a shell, you need to set `cmd` as an array of 3 strings, where the first contains the shell, the second contains `-c`, and the third contains the actual command. Example:
+
+```lua
+cmd = { awful.util.shell, "-c", "myactualcommand" }
+```
+
+`settings` can use the string `output`, which is the output of `cmd`.
+
+### Default `settings` function
+
+```lua
+settings = function() widget:set_text(output) end
+```
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`widget` | The widget | input widget type or `wibox.widget.textbox`
+`update` | Update `widget` | function
+`timer` | The widget timer | [`gears.timer`](https://awesomewm.org/doc/api/classes/gears.timer.html) or `nil`
+
+The `update` function can be used to refresh the widget before `timeout` expires.
+
+If `stoppable == true`, the widget will have an ad-hoc timer, which you can control though `timer` variable.
+
+## Use case examples
+
+### bitcoin
+
+```lua
+-- Bitcoin to USD current price, using Coinbase V1 API
+local bitcoin = lain.widget.watch({
+ timeout = 43200, -- half day
+ stoppable = true,
+ cmd = "curl -m5 -s 'https://coinbase.com/api/v1/prices/buy'",
+ settings = function()
+ local btc, pos, err = require("lain.util").dkjson.decode(output, 1, nil)
+ local btc_price = (not err and btc and btc["subtotal"]["amount"]) or "N/A"
+
+ -- customize here
+ widget:set_text(btc_price)
+ end
+})
+```
+
+### btrfs
+
+```lua
+-- btrfs root df
+local myrootfs = lain.widget.watch({
+ timeout = 600,
+ cmd = "btrfs filesystem df -g /",
+ settings = function()
+ local total, used = string.match(output, "Data.-total=(%d+%.%d+)GiB.-used=(%d+%.%d+)GiB")
+ local percent_used = math.ceil((tonumber(used) / tonumber(total)) * 100)
+
+ -- customize here
+ widget:set_text(" [/: " .. percent_used .. "%] ")
+ end
+})
+```
+
+### cmus
+
+```lua
+-- cmus audio player
+local cmus = lain.widget.watch({
+ timeout = 2,
+ stoppable = true,
+ cmd = "cmus-remote -Q",
+ settings = function()
+ local cmus_now = {
+ state = "N/A",
+ artist = "N/A",
+ title = "N/A",
+ album = "N/A"
+ }
+
+ for w in string.gmatch(output, "(.-)tag") do
+ a, b = w:match("(%w+) (.-)\n")
+ cmus_now[a] = b
+ end
+
+ -- customize here
+ widget:set_text(cmus_now.artist .. " - " .. cmus_now.title)
+ end
+})
+```
+
+### maildir
+
+```lua
+-- checks whether there are files in the "new" directories of a mail dirtree
+local mailpath = "~/Mail"
+local mymaildir = lain.widget.watch({
+ timeout = 60,
+ stoppable = true,
+ cmd = { awful.util.shell, "-c", string.format("ls -1dr %s/*/new/*", mailpath) },
+ settings = function()
+ local inbox_now = { digest = "" }
+
+ for dir in output:gmatch(".-/(%w+)/new") do
+ inbox_now[dir] = 1
+ for _ in output:gmatch(dir) do
+ inbox_now[dir] = inbox_now[dir] + 1
+ end
+ if #inbox_now.digest > 0 then inbox_now.digest = inbox_now.digest .. ", " end
+ inbox_now.digest = inbox_now.digest .. string.format("%s (%d)", dir, inbox_now[dir])
+ end
+
+ -- customize here
+ widget:set_text("mail: " .. inbox_now.digest)
+ end
+})
+```
+
+### mpris
+
+```lua
+-- infos from mpris clients such as spotify and VLC
+-- based on https://github.com/acrisci/playerctl
+local mpris = lain.widget.watch({
+ cmd = "playerctl status && playerctl metadata",
+ timeout = 2,
+ stoppable = true,
+ settings = function()
+ local escape_f = require("awful.util").escape
+ local mpris_now = {
+ state = "N/A",
+ artist = "N/A",
+ title = "N/A",
+ art_url = "N/A",
+ album = "N/A",
+ album_artist = "N/A"
+ }
+
+ mpris_now.state = string.match(output, "Playing") or
+ string.match(output, "Paused") or "N/A"
+
+ for k, v in string.gmatch(output, "'[^:]+:([^']+)':[%s]<%[?'([^']+)'%]?>")
+ do
+ if k == "artUrl" then mpris_now.art_url = v
+ elseif k == "artist" then mpris_now.artist = escape_f(v)
+ elseif k == "title" then mpris_now.title = escape_f(v)
+ elseif k == "album" then mpris_now.album = escape_f(v)
+ elseif k == "albumArtist" then mpris_now.album_artist = escape_f(v)
+ end
+ end
+
+ -- customize here
+ widget:set_text(mpris_now.artist .. " - " .. mpris_now.title)
+ end
+})
+```
+
+### upower
+
+```lua
+-- battery infos from freedesktop upower
+local mybattery = lain.widget.watch({
+ timeout = 30,
+ cmd = { awful.util.shell, "-c", "upower -i /org/freedesktop/UPower/devices/battery_BAT | sed -n '/present/,/icon-name/p'" },
+ settings = function()
+ local bat_now = {
+ present = "N/A",
+ state = "N/A",
+ warninglevel = "N/A",
+ energy = "N/A",
+ energyfull = "N/A",
+ energyrate = "N/A",
+ voltage = "N/A",
+ percentage = "N/A",
+ capacity = "N/A",
+ icon = "N/A"
+ }
+
+ for k, v in string.gmatch(output, '([%a]+[%a|-]+):%s*([%a|%d]+[,|%a|%d]-)') do
+ if k == "present" then bat_now.present = v
+ elseif k == "state" then bat_now.state = v
+ elseif k == "warning-level" then bat_now.warninglevel = v
+ elseif k == "energy" then bat_now.energy = string.gsub(v, ",", ".") -- Wh
+ elseif k == "energy-full" then bat_now.energyfull = string.gsub(v, ",", ".") -- Wh
+ elseif k == "energy-rate" then bat_now.energyrate = string.gsub(v, ",", ".") -- W
+ elseif k == "voltage" then bat_now.voltage = string.gsub(v, ",", ".") -- V
+ elseif k == "percentage" then bat_now.percentage = tonumber(v) -- %
+ elseif k == "capacity" then bat_now.capacity = string.gsub(v, ",", ".") -- %
+ elseif k == "icon-name" then bat_now.icon = v
+ end
+ end
+
+ -- customize here
+ widget:set_text("Bat: " .. bat_now.percentage .. " " .. bat_now.state)
+ end
+})
+```
diff --git a/Phranque/awesome/lain/wiki/weather.md b/Phranque/awesome/lain/wiki/weather.md
new file mode 100644
index 00000000..1875965c
--- /dev/null
+++ b/Phranque/awesome/lain/wiki/weather.md
@@ -0,0 +1,150 @@
+## Usage
+
+[Read here.](https://github.com/lcpz/lain/wiki/Widgets#usage)
+
+### Description
+
+Provides current weather status widgets and X-days forecast popup notifications.
+
+Powered by [OpenWeatherMap](http://openweathermap.org/api) API.
+
+By default, it uses [current](http://openweathermap.org/current) for current weather data and [forecast16](http://openweathermap.org/forecast16) for forecasts.
+
+```lua
+local myweather = lain.widget.weather()
+```
+
+## Input table
+
+Variable | Meaning | Type | Default
+--- | --- | --- | ---
+`timeout` | Refresh timeout seconds for current weather status | number | 900 (15 min)
+`timeout_forecast` | Refresh timeout seconds for forecast notification | number | 86400 (24 hrs)
+`current_call` | Command to fetch weather status data from the API | string | see `default_current_call`
+`forecast_call` | Command to fetch forecast data from the API | string | see `default_forecast_call`
+`city_id` | API city code | number | not set
+`utc_offset` | UTC time offset | function | see [here](https://github.com/lcpz/lain/blob/master/widget/weather.lua#L35-L39)
+`units` | Temperature units system | string | "metric"
+`lang` | API data localization | string | "en"
+`cnt` | Forecast days interval | integer | 5
+`date_cmd` | Forecast notification format style | string | "date -u -d @%d +'%%a %%d'"
+`icons_path` | Icons path | string | `lain/icons/openweathermap`
+`notification_preset` | Preset for notifications | table | empty table
+`notification_text_fun` | Function to format forecast notifications | function | see `notification_text_fun` below
+`weather_na_markup` | Markup to be used when weather textbox is not available | text | " N/A "
+`followtag` | Display the notification on currently focused screen | boolean | false
+`showpopup` | Display popups with mouse hovering | string, possible values: "on", "off" | "on"
+`settings` | User settings | function | empty function
+
+- ``default_current_call``
+
+ `"curl -s 'http://api.openweathermap.org/data/2.5/weather?id=%s&units=%s&lang=%s'"`
+
+ You can rewrite it using any fetcher solution you like, or you can modify it in order to fetch data by city name, instead of ID: just replace `id` with `q`:
+
+ `"curl -s 'http://api.openweathermap.org/data/2.5/weather?q=%s&units=%s&lang=%s'"`
+
+ and set `city_id` with your city name, for instance `city_id = "London,UK"`.
+
+- ``default_forecast_call``
+
+ `"curl -s 'http://api.openweathermap.org/data/2.5/forecast/daily?id=%s&units=%s&lang=%s&cnt=%s'"`
+
+ Like above.
+ If you want to use [forecast5](http://openweathermap.org/forecast5), use this API call string:
+ `http://api.openweathermap.org/data/2.5/forecast?id=%s&units=%s&lang=%s&cnt=%s`
+
+- ``city_id``
+
+ An integer that defines the OpenWeatherMap ID code of your city.
+ To obtain it go to [OpenWeatherMap](http://openweathermap.org/) and query for your city in the top search bar. The link will look like this:
+
+ http://openweathermap.org/city/2643743
+
+ your `city_id` is the number at the end.
+
+- ``units``
+
+ - For temperature in Fahrenheit use `units = "imperial"`
+ - For temperature in Celsius use `units = "metric"` (Lain default)
+ - For temperature in Kelvin use `units = "standard"` (OpenWeatherMap default)
+
+- ``lang``
+
+ See *Multilingual Support* section [here](http://openweathermap.org/current).
+
+- ``cnt``
+
+ Determines how many days to show in the forecast notification. Up to 16 if you use [forecast16](http://openweathermap.org/forecast16) (default), and up to 5 if you use [forecast5](http://openweathermap.org/forecast5).
+
+- ``date_cmd``
+
+ OpenWeatherMap time is in UNIX format, so this variable uses `date` to determine how each line in the forecast notification is formatted. Default looks like this:
+
+ day #daynumber: forecast, temp_min - temp_max
+
+ see `man date` for your customizations.
+
+- ``icons_path``
+
+ You can set your own icons path if you don't wish to use `lain/icons/openweathermap`. Just be sure that your icons are PNGs and named exactly like [OpenWeatherMap ones](http://openweathermap.org/weather-conditions).
+
+- ``notification_preset``
+
+ Notifications preset table. See [here](https://awesomewm.org/doc/api/libraries/naughty.html#notify) for the details.
+
+- ``notification_text_fun``
+ ```lua
+ function (wn)
+ local day = string.gsub(read_pipe(string.format(date_cmd, wn["dt"])), "\n", "")
+ local tmin = math.floor(wn["temp"]["min"])
+ local tmax = math.floor(wn["temp"]["max"])
+ local desc = wn["weather"][1]["description"]
+
+ return string.format("%s: %s, %d - %d ", day, desc, tmin, tmax)
+ end
+ ```
+ See [here](https://github.com/lcpz/lain/issues/186#issuecomment-203400918) for a complete customization example.
+
+- ``followtag``
+
+ With multiple screens, the default behaviour is to show a visual notification pop-up window on the first screen. By setting `followtag` to `true` it will be shown on the currently focused tag screen.
+
+- ``settings``
+
+ In your `settings` function, you can use `widget` variable to refer to the textbox, and the dictionary `weather_now` to refer to data retrieved by `current_call`. The dictionary is built with [dkjson library](http://dkolf.de/src/dkjson-lua.fsl/home), and its structure is defined [here](http://openweathermap.org/weather-data).
+ For instance, you can retrieve current weather status and temperature in this way:
+ ```lua
+ descr = weather_now["weather"][1]["description"]:lower()
+ units = math.floor(weather_now["main"]["temp"])
+ ```
+
+## Output table
+
+Variable | Meaning | Type
+--- | --- | ---
+`widget` | The widget | `wibox.widget.textbox`
+`icon` | The icon | `wibox.widget.imagebox`
+`update` | Update `widget` | function
+`timer` | The widget timer | [`gears.timer`](https://awesomewm.org/doc/api/classes/gears.timer.html)
+`timer_forecast` | The forecast notification timer | [`gears.timer`](https://awesomewm.org/doc/api/classes/gears.timer.html)
+
+## Functions
+
+You can attach the forecast notification to any widget like this:
+
+```lua
+myweather.attach(obj)
+```
+
+Hovering over ``obj`` will display the notification.
+
+## Keybindings
+
+You can create a keybinding for the weather popup like this:
+
+```lua
+awful.key( { "Mod1" }, "w", function () myweather.show(5) end )
+```
+
+where ``show`` argument is an integer defining timeout seconds.
diff --git a/rc.lua b/Phranque/awesome/rc.lua
similarity index 99%
rename from rc.lua
rename to Phranque/awesome/rc.lua
index 2a40eca6..83867f85 100644
--- a/rc.lua
+++ b/Phranque/awesome/rc.lua
@@ -98,7 +98,6 @@ local scrlocker = "slock"
awful.util.terminal = terminal
awful.util.tagnames = { "#1", "#2", "#3", "#4", "#5" }
awful.layout.layouts = {
- awful.layout.suit.floating,
awful.layout.suit.tile,
awful.layout.suit.tile.left,
awful.layout.suit.tile.bottom,
@@ -120,6 +119,7 @@ awful.layout.layouts = {
lain.layout.centerwork.horizontal,
lain.layout.termfair,
lain.layout.termfair.center,
+ awful.layout.suit.floating,
}
awful.util.taglist_buttons = my_table.join(
@@ -186,6 +186,7 @@ lain.layout.cascade.tile.nmaster = 5
lain.layout.cascade.tile.ncol = 2
beautiful.init(string.format("%s/.config/awesome/themes/%s/theme.lua", os.getenv("HOME"), chosen_theme))
+beautiful.useless_gap=5
-- }}}
-- {{{ Menu
@@ -639,12 +640,6 @@ awful.rules.rules = {
{ rule_any = { type = { "dialog", "normal" } },
properties = { titlebars_enabled = false } },
- -- Set Firefox to always map on the first tag on screen 1.
- { rule = { class = "Firefox" },
- properties = { screen = 1, tag = awful.util.tagnames[1] } },
-
- { rule = { class = "Gimp", role = "gimp-image-window" },
- properties = { maximized = true } },
}
-- }}}
diff --git a/Phranque/awesome/rc.lua.template b/Phranque/awesome/rc.lua.template
new file mode 100644
index 00000000..069add85
--- /dev/null
+++ b/Phranque/awesome/rc.lua.template
@@ -0,0 +1,745 @@
+--[[
+
+ Awesome WM configuration template
+ github.com/lcpz
+
+--]]
+
+-- {{{ Required libraries
+local awesome, client, mouse, screen, tag = awesome, client, mouse, screen, tag
+local ipairs, string, os, table, tostring, tonumber, type = ipairs, string, os, table, tostring, tonumber, type
+
+local gears = require("gears")
+local awful = require("awful")
+ require("awful.autofocus")
+local wibox = require("wibox")
+local beautiful = require("beautiful")
+local naughty = require("naughty")
+local lain = require("lain")
+--local menubar = require("menubar")
+local freedesktop = require("freedesktop")
+local hotkeys_popup = require("awful.hotkeys_popup").widget
+ require("awful.hotkeys_popup.keys")
+local my_table = awful.util.table or gears.table -- 4.{0,1} compatibility
+-- }}}
+
+-- {{{ 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)
+ 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
+-- }}}
+
+-- {{{ Autostart windowless processes
+
+-- This function will run once every time Awesome is started
+local function run_once(cmd_arr)
+ for _, cmd in ipairs(cmd_arr) do
+ awful.spawn.with_shell(string.format("pgrep -u $USER -fx '%s' > /dev/null || (%s)", cmd, cmd))
+ end
+end
+
+run_once({ "urxvtd", "unclutter -root" }) -- entries must be separated by commas
+
+-- This function implements the XDG autostart specification
+--[[
+awful.spawn.with_shell(
+ 'if (xrdb -query | grep -q "^awesome\\.started:\\s*true$"); then exit; fi;' ..
+ 'xrdb -merge <<< "awesome.started:true";' ..
+ -- list each of your autostart commands, followed by ; inside single quotes, followed by ..
+ 'dex --environment Awesome --autostart --search-paths "$XDG_CONFIG_DIRS/autostart:$XDG_CONFIG_HOME/autostart"' -- https://github.com/jceb/dex
+)
+--]]
+
+-- }}}
+
+-- {{{ Variable definitions
+
+local themes = {
+ "blackburn", -- 1
+ "copland", -- 2
+ "dremora", -- 3
+ "holo", -- 4
+ "multicolor", -- 5
+ "powerarrow", -- 6
+ "powerarrow-dark", -- 7
+ "rainbow", -- 8
+ "steamburn", -- 9
+ "vertex", -- 10
+}
+
+local chosen_theme = themes[5]
+local modkey = "Mod4"
+local altkey = "Mod1"
+local terminal = "urxvtc"
+local editor = os.getenv("EDITOR") or "vim"
+local gui_editor = "gvim"
+local browser = "firefox"
+local guieditor = "atom"
+local scrlocker = "slock"
+
+awful.util.terminal = terminal
+awful.util.tagnames = { "1", "2", "3", "4", "5" }
+awful.layout.layouts = {
+ awful.layout.suit.floating,
+ awful.layout.suit.tile,
+ awful.layout.suit.tile.left,
+ awful.layout.suit.tile.bottom,
+ awful.layout.suit.tile.top,
+ --awful.layout.suit.fair,
+ --awful.layout.suit.fair.horizontal,
+ --awful.layout.suit.spiral,
+ --awful.layout.suit.spiral.dwindle,
+ --awful.layout.suit.max,
+ --awful.layout.suit.max.fullscreen,
+ --awful.layout.suit.magnifier,
+ --awful.layout.suit.corner.nw,
+ --awful.layout.suit.corner.ne,
+ --awful.layout.suit.corner.sw,
+ --awful.layout.suit.corner.se,
+ --lain.layout.cascade,
+ --lain.layout.cascade.tile,
+ --lain.layout.centerwork,
+ --lain.layout.centerwork.horizontal,
+ --lain.layout.termfair,
+ --lain.layout.termfair.center,
+}
+
+awful.util.taglist_buttons = my_table.join(
+ awful.button({ }, 1, function(t) t:view_only() end),
+ awful.button({ modkey }, 1, function(t)
+ if client.focus then
+ client.focus:move_to_tag(t)
+ end
+ end),
+ awful.button({ }, 3, awful.tag.viewtoggle),
+ awful.button({ modkey }, 3, function(t)
+ if client.focus then
+ client.focus:toggle_tag(t)
+ end
+ end),
+ awful.button({ }, 4, function(t) awful.tag.viewnext(t.screen) end),
+ awful.button({ }, 5, function(t) awful.tag.viewprev(t.screen) end)
+)
+
+awful.util.tasklist_buttons = my_table.join(
+ awful.button({ }, 1, function (c)
+ if c == client.focus then
+ c.minimized = true
+ else
+ --c:emit_signal("request::activate", "tasklist", {raise = true})
+
+ -- Without this, the following
+ -- :isvisible() makes no sense
+ c.minimized = false
+ if not c:isvisible() and c.first_tag then
+ c.first_tag:view_only()
+ end
+ -- This will also un-minimize
+ -- the client, if needed
+ client.focus = c
+ c:raise()
+ end
+ end),
+ awful.button({ }, 2, function (c) c:kill() end),
+ awful.button({ }, 3, function ()
+ local instance = nil
+
+ return function ()
+ if instance and instance.wibox.visible then
+ instance:hide()
+ instance = nil
+ else
+ instance = awful.menu.clients({theme = {width = 250}})
+ end
+ end
+ end),
+ awful.button({ }, 4, function () awful.client.focus.byidx(1) end),
+ awful.button({ }, 5, function () awful.client.focus.byidx(-1) end)
+)
+
+lain.layout.termfair.nmaster = 3
+lain.layout.termfair.ncol = 1
+lain.layout.termfair.center.nmaster = 3
+lain.layout.termfair.center.ncol = 1
+lain.layout.cascade.tile.offset_x = 2
+lain.layout.cascade.tile.offset_y = 32
+lain.layout.cascade.tile.extra_padding = 5
+lain.layout.cascade.tile.nmaster = 5
+lain.layout.cascade.tile.ncol = 2
+
+beautiful.init(string.format("%s/.config/awesome/themes/%s/theme.lua", os.getenv("HOME"), chosen_theme))
+-- }}}
+
+-- {{{ Menu
+local myawesomemenu = {
+ { "hotkeys", function() return false, hotkeys_popup.show_help end },
+ { "manual", terminal .. " -e man awesome" },
+ { "edit config", string.format("%s -e %s %s", terminal, editor, awesome.conffile) },
+ { "restart", awesome.restart },
+ { "quit", function() awesome.quit() end }
+}
+awful.util.mymainmenu = freedesktop.menu.build({
+ icon_size = beautiful.menu_height or 16,
+ before = {
+ { "Awesome", myawesomemenu, beautiful.awesome_icon },
+ -- other triads can be put here
+ },
+ after = {
+ { "Open terminal", terminal },
+ -- other triads can be put here
+ }
+})
+--menubar.utils.terminal = terminal -- Set the Menubar terminal for applications that require it
+-- }}}
+
+-- {{{ Screen
+-- Re-set wallpaper when a screen's geometry changes (e.g. different resolution)
+screen.connect_signal("property::geometry", function(s)
+ -- Wallpaper
+ if beautiful.wallpaper then
+ local wallpaper = beautiful.wallpaper
+ -- If wallpaper is a function, call it with the screen
+ if type(wallpaper) == "function" then
+ wallpaper = wallpaper(s)
+ end
+ gears.wallpaper.maximized(wallpaper, s, true)
+ end
+end)
+-- Create a wibox for each screen and add it
+awful.screen.connect_for_each_screen(function(s) beautiful.at_screen_connect(s) end)
+-- }}}
+
+-- {{{ Mouse bindings
+root.buttons(my_table.join(
+ awful.button({ }, 3, function () awful.util.mymainmenu:toggle() end),
+ awful.button({ }, 4, awful.tag.viewnext),
+ awful.button({ }, 5, awful.tag.viewprev)
+))
+-- }}}
+
+-- {{{ Key bindings
+globalkeys = my_table.join(
+ -- Take a screenshot
+ -- https://github.com/lcpz/dots/blob/master/bin/screenshot
+ awful.key({ altkey }, "p", function() os.execute("screenshot") end,
+ {description = "take a screenshot", group = "hotkeys"}),
+
+ -- X screen locker
+ awful.key({ altkey, "Control" }, "l", function () os.execute(scrlocker) end,
+ {description = "lock screen", group = "hotkeys"}),
+
+ -- Hotkeys
+ awful.key({ modkey, }, "s", hotkeys_popup.show_help,
+ {description = "show help", group="awesome"}),
+ -- Tag browsing
+ awful.key({ modkey, }, "Left", awful.tag.viewprev,
+ {description = "view previous", group = "tag"}),
+ awful.key({ modkey, }, "Right", awful.tag.viewnext,
+ {description = "view next", group = "tag"}),
+ awful.key({ modkey, }, "Escape", awful.tag.history.restore,
+ {description = "go back", group = "tag"}),
+
+ -- Non-empty tag browsing
+ awful.key({ altkey }, "Left", function () lain.util.tag_view_nonempty(-1) end,
+ {description = "view previous nonempty", group = "tag"}),
+ awful.key({ altkey }, "Right", function () lain.util.tag_view_nonempty(1) end,
+ {description = "view previous nonempty", group = "tag"}),
+
+ -- Default client focus
+ awful.key({ altkey, }, "j",
+ function ()
+ awful.client.focus.byidx( 1)
+ end,
+ {description = "focus next by index", group = "client"}
+ ),
+ awful.key({ altkey, }, "k",
+ function ()
+ awful.client.focus.byidx(-1)
+ end,
+ {description = "focus previous by index", group = "client"}
+ ),
+
+ -- By direction client focus
+ awful.key({ modkey }, "j",
+ function()
+ awful.client.focus.global_bydirection("down")
+ if client.focus then client.focus:raise() end
+ end,
+ {description = "focus down", group = "client"}),
+ awful.key({ modkey }, "k",
+ function()
+ awful.client.focus.global_bydirection("up")
+ if client.focus then client.focus:raise() end
+ end,
+ {description = "focus up", group = "client"}),
+ awful.key({ modkey }, "h",
+ function()
+ awful.client.focus.global_bydirection("left")
+ if client.focus then client.focus:raise() end
+ end,
+ {description = "focus left", group = "client"}),
+ awful.key({ modkey }, "l",
+ function()
+ awful.client.focus.global_bydirection("right")
+ if client.focus then client.focus:raise() end
+ end,
+ {description = "focus right", group = "client"}),
+ awful.key({ modkey, }, "w", function () awful.util.mymainmenu:show() end,
+ {description = "show main menu", group = "awesome"}),
+
+ -- Layout manipulation
+ awful.key({ modkey, "Shift" }, "j", function () awful.client.swap.byidx( 1) end,
+ {description = "swap with next client by index", group = "client"}),
+ awful.key({ modkey, "Shift" }, "k", function () awful.client.swap.byidx( -1) end,
+ {description = "swap with previous client by index", group = "client"}),
+ awful.key({ modkey, "Control" }, "j", function () awful.screen.focus_relative( 1) end,
+ {description = "focus the next screen", group = "screen"}),
+ awful.key({ modkey, "Control" }, "k", function () awful.screen.focus_relative(-1) end,
+ {description = "focus the previous screen", group = "screen"}),
+ awful.key({ modkey, }, "u", awful.client.urgent.jumpto,
+ {description = "jump to urgent client", group = "client"}),
+ awful.key({ modkey, }, "Tab",
+ function ()
+ awful.client.focus.history.previous()
+ if client.focus then
+ client.focus:raise()
+ end
+ end,
+ {description = "go back", group = "client"}),
+
+ -- Show/Hide Wibox
+ awful.key({ modkey }, "b", function ()
+ for s in screen do
+ s.mywibox.visible = not s.mywibox.visible
+ if s.mybottomwibox then
+ s.mybottomwibox.visible = not s.mybottomwibox.visible
+ end
+ end
+ end,
+ {description = "toggle wibox", group = "awesome"}),
+
+ -- On the fly useless gaps change
+ awful.key({ altkey, "Control" }, "+", function () lain.util.useless_gaps_resize(1) end,
+ {description = "increment useless gaps", group = "tag"}),
+ awful.key({ altkey, "Control" }, "-", function () lain.util.useless_gaps_resize(-1) end,
+ {description = "decrement useless gaps", group = "tag"}),
+
+ -- Dynamic tagging
+ awful.key({ modkey, "Shift" }, "n", function () lain.util.add_tag() end,
+ {description = "add new tag", group = "tag"}),
+ awful.key({ modkey, "Shift" }, "r", function () lain.util.rename_tag() end,
+ {description = "rename tag", group = "tag"}),
+ awful.key({ modkey, "Shift" }, "Left", function () lain.util.move_tag(-1) end,
+ {description = "move tag to the left", group = "tag"}),
+ awful.key({ modkey, "Shift" }, "Right", function () lain.util.move_tag(1) end,
+ {description = "move tag to the right", group = "tag"}),
+ awful.key({ modkey, "Shift" }, "d", function () lain.util.delete_tag() end,
+ {description = "delete tag", group = "tag"}),
+
+ -- Standard program
+ awful.key({ modkey, }, "Return", function () awful.spawn(terminal) end,
+ {description = "open a terminal", group = "launcher"}),
+ awful.key({ modkey, "Control" }, "r", awesome.restart,
+ {description = "reload awesome", group = "awesome"}),
+ awful.key({ modkey, "Shift" }, "q", awesome.quit,
+ {description = "quit awesome", group = "awesome"}),
+
+ awful.key({ altkey, "Shift" }, "l", function () awful.tag.incmwfact( 0.05) end,
+ {description = "increase master width factor", group = "layout"}),
+ awful.key({ altkey, "Shift" }, "h", function () awful.tag.incmwfact(-0.05) end,
+ {description = "decrease master width factor", group = "layout"}),
+ awful.key({ modkey, "Shift" }, "h", function () awful.tag.incnmaster( 1, nil, true) end,
+ {description = "increase the number of master clients", group = "layout"}),
+ awful.key({ modkey, "Shift" }, "l", function () awful.tag.incnmaster(-1, nil, true) end,
+ {description = "decrease the number of master clients", group = "layout"}),
+ awful.key({ modkey, "Control" }, "h", function () awful.tag.incncol( 1, nil, true) end,
+ {description = "increase the number of columns", group = "layout"}),
+ awful.key({ modkey, "Control" }, "l", function () awful.tag.incncol(-1, nil, true) end,
+ {description = "decrease the number of columns", group = "layout"}),
+ awful.key({ modkey, }, "space", function () awful.layout.inc( 1) end,
+ {description = "select next", group = "layout"}),
+ awful.key({ modkey, "Shift" }, "space", function () awful.layout.inc(-1) end,
+ {description = "select previous", group = "layout"}),
+
+ awful.key({ modkey, "Control" }, "n",
+ function ()
+ local c = awful.client.restore()
+ -- Focus restored client
+ if c then
+ client.focus = c
+ c:raise()
+ end
+ end,
+ {description = "restore minimized", group = "client"}),
+
+ -- Dropdown application
+ awful.key({ modkey, }, "z", function () awful.screen.focused().quake:toggle() end,
+ {description = "dropdown application", group = "launcher"}),
+
+ -- Widgets popups
+ awful.key({ altkey, }, "c", function () if beautiful.cal then beautiful.cal.show(7) end end,
+ {description = "show calendar", group = "widgets"}),
+ awful.key({ altkey, }, "h", function () if beautiful.fs then beautiful.fs.show(7) end end,
+ {description = "show filesystem", group = "widgets"}),
+ awful.key({ altkey, }, "w", function () if beautiful.weather then beautiful.weather.show(7) end end,
+ {description = "show weather", group = "widgets"}),
+
+ -- Brightness
+ awful.key({ }, "XF86MonBrightnessUp", function () os.execute("xbacklight -inc 10") end,
+ {description = "+10%", group = "hotkeys"}),
+ awful.key({ }, "XF86MonBrightnessDown", function () os.execute("xbacklight -dec 10") end,
+ {description = "-10%", group = "hotkeys"}),
+
+ -- ALSA volume control
+ awful.key({ altkey }, "Up",
+ function ()
+ os.execute(string.format("amixer -q set %s 1%%+", beautiful.volume.channel))
+ beautiful.volume.update()
+ end,
+ {description = "volume up", group = "hotkeys"}),
+ awful.key({ altkey }, "Down",
+ function ()
+ os.execute(string.format("amixer -q set %s 1%%-", beautiful.volume.channel))
+ beautiful.volume.update()
+ end,
+ {description = "volume down", group = "hotkeys"}),
+ awful.key({ altkey }, "m",
+ function ()
+ os.execute(string.format("amixer -q set %s toggle", beautiful.volume.togglechannel or beautiful.volume.channel))
+ beautiful.volume.update()
+ end,
+ {description = "toggle mute", group = "hotkeys"}),
+ awful.key({ altkey, "Control" }, "m",
+ function ()
+ os.execute(string.format("amixer -q set %s 100%%", beautiful.volume.channel))
+ beautiful.volume.update()
+ end,
+ {description = "volume 100%", group = "hotkeys"}),
+ awful.key({ altkey, "Control" }, "0",
+ function ()
+ os.execute(string.format("amixer -q set %s 0%%", beautiful.volume.channel))
+ beautiful.volume.update()
+ end,
+ {description = "volume 0%", group = "hotkeys"}),
+
+ -- MPD control
+ awful.key({ altkey, "Control" }, "Up",
+ function ()
+ os.execute("mpc toggle")
+ beautiful.mpd.update()
+ end,
+ {description = "mpc toggle", group = "widgets"}),
+ awful.key({ altkey, "Control" }, "Down",
+ function ()
+ os.execute("mpc stop")
+ beautiful.mpd.update()
+ end,
+ {description = "mpc stop", group = "widgets"}),
+ awful.key({ altkey, "Control" }, "Left",
+ function ()
+ os.execute("mpc prev")
+ beautiful.mpd.update()
+ end,
+ {description = "mpc prev", group = "widgets"}),
+ awful.key({ altkey, "Control" }, "Right",
+ function ()
+ os.execute("mpc next")
+ beautiful.mpd.update()
+ end,
+ {description = "mpc next", group = "widgets"}),
+ awful.key({ altkey }, "0",
+ function ()
+ local common = { text = "MPD widget ", position = "top_middle", timeout = 2 }
+ if beautiful.mpd.timer.started then
+ beautiful.mpd.timer:stop()
+ common.text = common.text .. lain.util.markup.bold("OFF")
+ else
+ beautiful.mpd.timer:start()
+ common.text = common.text .. lain.util.markup.bold("ON")
+ end
+ naughty.notify(common)
+ end,
+ {description = "mpc on/off", group = "widgets"}),
+
+ -- Copy primary to clipboard (terminals to gtk)
+ awful.key({ modkey }, "c", function () awful.spawn.with_shell("xsel | xsel -i -b") end,
+ {description = "copy terminal to gtk", group = "hotkeys"}),
+ -- Copy clipboard to primary (gtk to terminals)
+ awful.key({ modkey }, "v", function () awful.spawn.with_shell("xsel -b | xsel") end,
+ {description = "copy gtk to terminal", group = "hotkeys"}),
+
+ -- User programs
+ awful.key({ modkey }, "q", function () awful.spawn(browser) end,
+ {description = "run browser", group = "launcher"}),
+ awful.key({ modkey }, "a", function () awful.spawn(guieditor) end,
+ {description = "run gui editor", group = "launcher"}),
+
+ -- Default
+ --[[ Menubar
+ awful.key({ modkey }, "p", function() menubar.show() end,
+ {description = "show the menubar", group = "launcher"})
+ --]]
+ --[[ dmenu
+ awful.key({ modkey }, "x", function ()
+ os.execute(string.format("dmenu_run -i -fn 'Monospace' -nb '%s' -nf '%s' -sb '%s' -sf '%s'",
+ beautiful.bg_normal, beautiful.fg_normal, beautiful.bg_focus, beautiful.fg_focus))
+ end,
+ {description = "show dmenu", group = "launcher"})
+ --]]
+ -- Prompt
+ awful.key({ modkey }, "r", function () awful.screen.focused().mypromptbox:run() end,
+ {description = "run prompt", group = "launcher"}),
+
+ awful.key({ modkey }, "x",
+ function ()
+ awful.prompt.run {
+ prompt = "Run Lua code: ",
+ textbox = awful.screen.focused().mypromptbox.widget,
+ exe_callback = awful.util.eval,
+ history_path = awful.util.get_cache_dir() .. "/history_eval"
+ }
+ end,
+ {description = "lua execute prompt", group = "awesome"})
+ --]]
+)
+
+clientkeys = my_table.join(
+ awful.key({ altkey, "Shift" }, "m", lain.util.magnify_client,
+ {description = "magnify client", group = "client"}),
+ awful.key({ modkey, }, "f",
+ function (c)
+ c.fullscreen = not c.fullscreen
+ c:raise()
+ end,
+ {description = "toggle fullscreen", group = "client"}),
+ awful.key({ modkey, "Shift" }, "c", function (c) c:kill() end,
+ {description = "close", group = "client"}),
+ awful.key({ modkey, "Control" }, "space", awful.client.floating.toggle ,
+ {description = "toggle floating", group = "client"}),
+ awful.key({ modkey, "Control" }, "Return", function (c) c:swap(awful.client.getmaster()) end,
+ {description = "move to master", group = "client"}),
+ awful.key({ modkey, }, "o", function (c) c:move_to_screen() end,
+ {description = "move to screen", group = "client"}),
+ awful.key({ modkey, }, "t", function (c) c.ontop = not c.ontop end,
+ {description = "toggle keep on top", group = "client"}),
+ awful.key({ modkey, }, "n",
+ function (c)
+ -- The client currently has the input focus, so it cannot be
+ -- minimized, since minimized clients can't have the focus.
+ c.minimized = true
+ end ,
+ {description = "minimize", group = "client"}),
+ awful.key({ modkey, }, "m",
+ function (c)
+ c.maximized = not c.maximized
+ c:raise()
+ end ,
+ {description = "maximize", group = "client"})
+)
+
+-- Bind all key numbers to tags.
+-- Be careful: we use keycodes to make it works on any keyboard layout.
+-- This should map on the top row of your keyboard, usually 1 to 9.
+for i = 1, 9 do
+ -- Hack to only show tags 1 and 9 in the shortcut window (mod+s)
+ local descr_view, descr_toggle, descr_move, descr_toggle_focus
+ if i == 1 or i == 9 then
+ descr_view = {description = "view tag #", group = "tag"}
+ descr_toggle = {description = "toggle tag #", group = "tag"}
+ descr_move = {description = "move focused client to tag #", group = "tag"}
+ descr_toggle_focus = {description = "toggle focused client on tag #", group = "tag"}
+ end
+ globalkeys = my_table.join(globalkeys,
+ -- View tag only.
+ awful.key({ modkey }, "#" .. i + 9,
+ function ()
+ local screen = awful.screen.focused()
+ local tag = screen.tags[i]
+ if tag then
+ tag:view_only()
+ end
+ end,
+ descr_view),
+ -- Toggle tag display.
+ awful.key({ modkey, "Control" }, "#" .. i + 9,
+ function ()
+ local screen = awful.screen.focused()
+ local tag = screen.tags[i]
+ if tag then
+ awful.tag.viewtoggle(tag)
+ end
+ end,
+ descr_toggle),
+ -- Move client to tag.
+ awful.key({ modkey, "Shift" }, "#" .. i + 9,
+ function ()
+ if client.focus then
+ local tag = client.focus.screen.tags[i]
+ if tag then
+ client.focus:move_to_tag(tag)
+ end
+ end
+ end,
+ descr_move),
+ -- Toggle tag on focused client.
+ awful.key({ modkey, "Control", "Shift" }, "#" .. i + 9,
+ function ()
+ if client.focus then
+ local tag = client.focus.screen.tags[i]
+ if tag then
+ client.focus:toggle_tag(tag)
+ end
+ end
+ end,
+ descr_toggle_focus)
+ )
+end
+
+clientbuttons = gears.table.join(
+ awful.button({ }, 1, function (c)
+ c:emit_signal("request::activate", "mouse_click", {raise = true})
+ end),
+ awful.button({ modkey }, 1, function (c)
+ c:emit_signal("request::activate", "mouse_click", {raise = true})
+ awful.mouse.client.move(c)
+ end),
+ awful.button({ modkey }, 3, function (c)
+ c:emit_signal("request::activate", "mouse_click", {raise = true})
+ awful.mouse.client.resize(c)
+ end)
+)
+
+-- 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 = beautiful.border_width,
+ border_color = beautiful.border_normal,
+ focus = awful.client.focus.filter,
+ raise = true,
+ keys = clientkeys,
+ buttons = clientbuttons,
+ screen = awful.screen.preferred,
+ placement = awful.placement.no_overlap+awful.placement.no_offscreen,
+ size_hints_honor = false
+ }
+ },
+
+ -- Titlebars
+ { rule_any = { type = { "dialog", "normal" } },
+ properties = { titlebars_enabled = true } },
+
+ -- Set Firefox to always map on the first tag on screen 1.
+ { rule = { class = "Firefox" },
+ properties = { screen = 1, tag = awful.util.tagnames[1] } },
+
+ { rule = { class = "Gimp", role = "gimp-image-window" },
+ properties = { maximized = true } },
+}
+-- }}}
+
+-- {{{ Signals
+-- Signal function to execute when a new client appears.
+client.connect_signal("manage", function (c)
+ -- Set the windows at the slave,
+ -- i.e. put it at the end of others instead of setting it master.
+ -- if not awesome.startup then awful.client.setslave(c) end
+
+ if awesome.startup and
+ not c.size_hints.user_position
+ and not c.size_hints.program_position then
+ -- Prevent clients from being unreachable after screen count changes.
+ awful.placement.no_offscreen(c)
+ end
+end)
+
+-- Add a titlebar if titlebars_enabled is set to true in the rules.
+client.connect_signal("request::titlebars", function(c)
+ -- Custom
+ if beautiful.titlebar_fun then
+ beautiful.titlebar_fun(c)
+ return
+ end
+
+ -- Default
+ -- buttons for the titlebar
+ local buttons = my_table.join(
+ awful.button({ }, 1, function()
+ c:emit_signal("request::activate", "titlebar", {raise = true})
+ awful.mouse.client.move(c)
+ end),
+ awful.button({ }, 2, function() c:kill() end),
+ awful.button({ }, 3, function()
+ c:emit_signal("request::activate", "titlebar", {raise = true})
+ awful.mouse.client.resize(c)
+ end)
+ )
+
+ awful.titlebar(c, {size = 16}) : setup {
+ { -- Left
+ awful.titlebar.widget.iconwidget(c),
+ buttons = buttons,
+ layout = wibox.layout.fixed.horizontal
+ },
+ { -- Middle
+ { -- Title
+ align = "center",
+ widget = awful.titlebar.widget.titlewidget(c)
+ },
+ buttons = buttons,
+ layout = wibox.layout.flex.horizontal
+ },
+ { -- Right
+ awful.titlebar.widget.floatingbutton (c),
+ awful.titlebar.widget.maximizedbutton(c),
+ awful.titlebar.widget.stickybutton (c),
+ awful.titlebar.widget.ontopbutton (c),
+ awful.titlebar.widget.closebutton (c),
+ layout = wibox.layout.fixed.horizontal()
+ },
+ layout = wibox.layout.align.horizontal
+ }
+end)
+
+-- Enable sloppy focus, so that focus follows mouse.
+client.connect_signal("mouse::enter", function(c)
+ c:emit_signal("request::activate", "mouse_enter", {raise = true})
+end)
+
+-- No border for maximized clients
+function border_adjust(c)
+ if c.maximized then -- no borders if only 1 client visible
+ c.border_width = 0
+ elseif #awful.screen.focused().clients > 1 then
+ c.border_width = beautiful.border_width
+ c.border_color = beautiful.border_focus
+ end
+end
+
+client.connect_signal("property::maximized", border_adjust)
+client.connect_signal("focus", border_adjust)
+client.connect_signal("unfocus", function(c) c.border_color = beautiful.border_normal end)
+-- }}}
diff --git a/Phranque/awesome/themes/blackburn/icons/awesome.png b/Phranque/awesome/themes/blackburn/icons/awesome.png
new file mode 100644
index 00000000..947a06de
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/awesome.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/dwindle.png b/Phranque/awesome/themes/blackburn/icons/dwindle.png
new file mode 100644
index 00000000..2f4b4ac7
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/dwindle.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/fairh.png b/Phranque/awesome/themes/blackburn/icons/fairh.png
new file mode 100644
index 00000000..b0030e71
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/fairh.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/fairv.png b/Phranque/awesome/themes/blackburn/icons/fairv.png
new file mode 100644
index 00000000..2c0e1579
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/fairv.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/floating.png b/Phranque/awesome/themes/blackburn/icons/floating.png
new file mode 100644
index 00000000..e0bdd4b8
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/floating.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/magnifier.png b/Phranque/awesome/themes/blackburn/icons/magnifier.png
new file mode 100644
index 00000000..3713108a
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/magnifier.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/max.png b/Phranque/awesome/themes/blackburn/icons/max.png
new file mode 100644
index 00000000..8828d122
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/max.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/spiral.png b/Phranque/awesome/themes/blackburn/icons/spiral.png
new file mode 100644
index 00000000..54c2c5e1
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/spiral.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/square_sel.png b/Phranque/awesome/themes/blackburn/icons/square_sel.png
new file mode 100644
index 00000000..6938ccaa
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/square_sel.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/square_unsel.png b/Phranque/awesome/themes/blackburn/icons/square_unsel.png
new file mode 100644
index 00000000..dd8064db
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/square_unsel.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/submenu.png b/Phranque/awesome/themes/blackburn/icons/submenu.png
new file mode 100644
index 00000000..10ca0144
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/submenu.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/tile.png b/Phranque/awesome/themes/blackburn/icons/tile.png
new file mode 100644
index 00000000..409141fa
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/tile.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/tilebottom.png b/Phranque/awesome/themes/blackburn/icons/tilebottom.png
new file mode 100644
index 00000000..de68e9c8
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/tilebottom.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/tileleft.png b/Phranque/awesome/themes/blackburn/icons/tileleft.png
new file mode 100644
index 00000000..c47e4eec
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/tileleft.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/tiletop.png b/Phranque/awesome/themes/blackburn/icons/tiletop.png
new file mode 100644
index 00000000..c50abdf7
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/tiletop.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/close_focus.png b/Phranque/awesome/themes/blackburn/icons/titlebar/close_focus.png
new file mode 100644
index 00000000..d0bc2e84
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/close_focus.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/close_normal.png b/Phranque/awesome/themes/blackburn/icons/titlebar/close_normal.png
new file mode 100644
index 00000000..f99db1ad
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/close_normal.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/floating_focus_active.png b/Phranque/awesome/themes/blackburn/icons/titlebar/floating_focus_active.png
new file mode 100644
index 00000000..1a0649fb
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/floating_focus_active.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/floating_focus_inactive.png b/Phranque/awesome/themes/blackburn/icons/titlebar/floating_focus_inactive.png
new file mode 100644
index 00000000..47f19f6d
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/floating_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/floating_normal_active.png b/Phranque/awesome/themes/blackburn/icons/titlebar/floating_normal_active.png
new file mode 100644
index 00000000..44b8dac7
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/floating_normal_active.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/floating_normal_inactive.png b/Phranque/awesome/themes/blackburn/icons/titlebar/floating_normal_inactive.png
new file mode 100644
index 00000000..98f7f1d7
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/floating_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/maximized_focus_active.png b/Phranque/awesome/themes/blackburn/icons/titlebar/maximized_focus_active.png
new file mode 100644
index 00000000..eab9b37e
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/maximized_focus_active.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/maximized_focus_inactive.png b/Phranque/awesome/themes/blackburn/icons/titlebar/maximized_focus_inactive.png
new file mode 100644
index 00000000..bce1d000
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/maximized_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/maximized_normal_active.png b/Phranque/awesome/themes/blackburn/icons/titlebar/maximized_normal_active.png
new file mode 100644
index 00000000..4f03984e
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/maximized_normal_active.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/maximized_normal_inactive.png b/Phranque/awesome/themes/blackburn/icons/titlebar/maximized_normal_inactive.png
new file mode 100644
index 00000000..2e56d327
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/maximized_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/ontop_focus_active.png b/Phranque/awesome/themes/blackburn/icons/titlebar/ontop_focus_active.png
new file mode 100644
index 00000000..42a9770a
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/ontop_focus_active.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/ontop_focus_inactive.png b/Phranque/awesome/themes/blackburn/icons/titlebar/ontop_focus_inactive.png
new file mode 100644
index 00000000..2f3a2be2
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/ontop_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/ontop_normal_active.png b/Phranque/awesome/themes/blackburn/icons/titlebar/ontop_normal_active.png
new file mode 100644
index 00000000..499dfbfa
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/ontop_normal_active.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/ontop_normal_inactive.png b/Phranque/awesome/themes/blackburn/icons/titlebar/ontop_normal_inactive.png
new file mode 100644
index 00000000..a9a32068
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/ontop_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/sticky_focus_active.png b/Phranque/awesome/themes/blackburn/icons/titlebar/sticky_focus_active.png
new file mode 100644
index 00000000..72864acd
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/sticky_focus_active.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/sticky_focus_inactive.png b/Phranque/awesome/themes/blackburn/icons/titlebar/sticky_focus_inactive.png
new file mode 100644
index 00000000..5493d8ec
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/sticky_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/sticky_normal_active.png b/Phranque/awesome/themes/blackburn/icons/titlebar/sticky_normal_active.png
new file mode 100644
index 00000000..a1501544
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/sticky_normal_active.png differ
diff --git a/Phranque/awesome/themes/blackburn/icons/titlebar/sticky_normal_inactive.png b/Phranque/awesome/themes/blackburn/icons/titlebar/sticky_normal_inactive.png
new file mode 100644
index 00000000..7e6c99b6
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/icons/titlebar/sticky_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/blackburn/theme.lua b/Phranque/awesome/themes/blackburn/theme.lua
new file mode 100644
index 00000000..3179653e
--- /dev/null
+++ b/Phranque/awesome/themes/blackburn/theme.lua
@@ -0,0 +1,274 @@
+--[[
+
+ Blackburn Awesome WM theme 3.0
+ github.com/lcpz
+
+--]]
+
+local gears = require("gears")
+local lain = require("lain")
+local awful = require("awful")
+local wibox = require("wibox")
+
+local os = os
+local my_table = awful.util.table or gears.table -- 4.{0,1} compatibility
+
+local theme = {}
+theme.dir = os.getenv("HOME") .. "/.config/awesome/themes/blackburn"
+theme.wallpaper = theme.dir .. "/wall.png"
+theme.font = "Misc Tamsyn 10.5"
+theme.taglist_font = "Icons 10"
+theme.fg_normal = "#D7D7D7"
+theme.fg_focus = "#F6784F"
+theme.bg_normal = "#060606"
+theme.bg_focus = "#060606"
+theme.fg_urgent = "#CC9393"
+theme.bg_urgent = "#2A1F1E"
+theme.border_width = 1
+theme.border_normal = "#0E0E0E"
+theme.border_focus = "#F79372"
+theme.taglist_fg_focus = "#F6784F"
+theme.taglist_bg_focus = "#060606"
+theme.tasklist_fg_focus = "#F6784F"
+theme.tasklist_bg_focus = "#060606"
+theme.menu_height = 16
+theme.menu_width = 130
+theme.menu_submenu_icon = theme.dir .. "/icons/submenu.png"
+theme.awesome_icon = theme.dir .."/icons/awesome.png"
+theme.taglist_squares_sel = theme.dir .. "/icons/square_sel.png"
+theme.taglist_squares_unsel = theme.dir .. "/icons/square_unsel.png"
+theme.layout_tile = theme.dir .. "/icons/tile.png"
+theme.layout_tileleft = theme.dir .. "/icons/tileleft.png"
+theme.layout_tilebottom = theme.dir .. "/icons/tilebottom.png"
+theme.layout_tiletop = theme.dir .. "/icons/tiletop.png"
+theme.layout_fairv = theme.dir .. "/icons/fairv.png"
+theme.layout_fairh = theme.dir .. "/icons/fairh.png"
+theme.layout_spiral = theme.dir .. "/icons/spiral.png"
+theme.layout_dwindle = theme.dir .. "/icons/dwindle.png"
+theme.layout_max = theme.dir .. "/icons/max.png"
+theme.layout_fullscreen = theme.dir .. "/icons/fullscreen.png"
+theme.layout_magnifier = theme.dir .. "/icons/magnifier.png"
+theme.layout_floating = theme.dir .. "/icons/floating.png"
+theme.tasklist_plain_task_name = true
+theme.tasklist_disable_icon = true
+theme.useless_gap = 0
+theme.titlebar_close_button_focus = theme.dir .. "/icons/titlebar/close_focus.png"
+theme.titlebar_close_button_normal = theme.dir .. "/icons/titlebar/close_normal.png"
+theme.titlebar_ontop_button_focus_active = theme.dir .. "/icons/titlebar/ontop_focus_active.png"
+theme.titlebar_ontop_button_normal_active = theme.dir .. "/icons/titlebar/ontop_normal_active.png"
+theme.titlebar_ontop_button_focus_inactive = theme.dir .. "/icons/titlebar/ontop_focus_inactive.png"
+theme.titlebar_ontop_button_normal_inactive = theme.dir .. "/icons/titlebar/ontop_normal_inactive.png"
+theme.titlebar_sticky_button_focus_active = theme.dir .. "/icons/titlebar/sticky_focus_active.png"
+theme.titlebar_sticky_button_normal_active = theme.dir .. "/icons/titlebar/sticky_normal_active.png"
+theme.titlebar_sticky_button_focus_inactive = theme.dir .. "/icons/titlebar/sticky_focus_inactive.png"
+theme.titlebar_sticky_button_normal_inactive = theme.dir .. "/icons/titlebar/sticky_normal_inactive.png"
+theme.titlebar_floating_button_focus_active = theme.dir .. "/icons/titlebar/floating_focus_active.png"
+theme.titlebar_floating_button_normal_active = theme.dir .. "/icons/titlebar/floating_normal_active.png"
+theme.titlebar_floating_button_focus_inactive = theme.dir .. "/icons/titlebar/floating_focus_inactive.png"
+theme.titlebar_floating_button_normal_inactive = theme.dir .. "/icons/titlebar/floating_normal_inactive.png"
+theme.titlebar_maximized_button_focus_active = theme.dir .. "/icons/titlebar/maximized_focus_active.png"
+theme.titlebar_maximized_button_normal_active = theme.dir .. "/icons/titlebar/maximized_normal_active.png"
+theme.titlebar_maximized_button_focus_inactive = theme.dir .. "/icons/titlebar/maximized_focus_inactive.png"
+theme.titlebar_maximized_button_normal_inactive = theme.dir .. "/icons/titlebar/maximized_normal_inactive.png"
+
+awful.util.tagnames = { "ƀ", "Ƅ", "Ɗ", "ƈ", "ƙ" }
+
+local markup = lain.util.markup
+local separators = lain.util.separators
+local gray = "#9E9C9A"
+
+-- Textclock
+local mytextclock = wibox.widget.textclock(" %H:%M ")
+mytextclock.font = theme.font
+
+-- Calendar
+theme.cal = lain.widget.cal({
+ attach_to = { mytextclock },
+ notification_preset = {
+ font = "Misc Tamsyn 11",
+ fg = theme.fg_normal,
+ bg = theme.bg_normal
+ }
+})
+
+-- Mail IMAP check
+--[[ commented because it needs to be set before use
+theme.mail = lain.widget.imap({
+ timeout = 180,
+ server = "server",
+ mail = "mail",
+ password = "keyring get mail",
+ notification_preset = { fg = white }
+ settings = function()
+ mail = ""
+ count = ""
+
+ if mailcount > 0 then
+ mail = "Mail "
+ count = mailcount .. " "
+ end
+
+ widget:set_markup(markup.font(theme.font, markup(gray, mail) .. count))
+ end
+})
+--]]
+
+-- MPD
+theme.mpd = lain.widget.mpd({
+ settings = function()
+ mpd_notification_preset.fg = white
+ artist = mpd_now.artist .. " "
+ title = mpd_now.title .. " "
+
+ if mpd_now.state == "pause" then
+ artist = "mpd "
+ title = "paused "
+ elseif mpd_now.state == "stop" then
+ artist = ""
+ title = ""
+ end
+
+ widget:set_markup(markup.font(theme.font, markup(gray, artist) .. title .. " "))
+ end
+})
+
+-- /home fs
+--[[ commented because it needs Gio/Glib >= 2.54
+theme.fs = lain.widget.fs({
+ notification_preset = { fg = white, bg = theme.bg_normal, font = "Misc Tamsyn 10.5" },
+ settings = function()
+ fs_header = ""
+ fs_p = ""
+
+ if fs_now["/home"].percentage >= 90 then
+ fs_header = " Hdd "
+ fs_p = fs_now["/home"].percentage
+ end
+
+ widget:set_markup(markup.font(theme.font, markup(gray, fs_header) .. fs_p))
+ end
+})
+--]]
+
+-- Battery
+local bat = lain.widget.bat({
+ settings = function()
+ bat_header = " Bat "
+ bat_p = bat_now.perc .. " "
+ widget:set_markup(markup.font(theme.font, markup(gray, bat_header) .. bat_p))
+ end
+})
+
+-- ALSA volume
+theme.volume = lain.widget.alsa({
+ --togglechannel = "IEC958,3",
+ settings = function()
+ header = " Vol "
+ vlevel = volume_now.level
+
+ if volume_now.status == "off" then
+ vlevel = vlevel .. "M "
+ else
+ vlevel = vlevel .. " "
+ end
+
+ widget:set_markup(markup.font(theme.font, markup(gray, header) .. vlevel))
+ end
+})
+
+-- Weather
+theme.weather = lain.widget.weather({
+ city_id = 2643743, -- placeholder (London)
+ settings = function()
+ units = math.floor(weather_now["main"]["temp"])
+ widget:set_markup(" " .. units .. " ")
+ end
+})
+
+-- Separators
+local first = wibox.widget.textbox(' ')
+local arrl_pre = separators.arrow_right("alpha", "#1A1A1A")
+local arrl_post = separators.arrow_right("#1A1A1A", "alpha")
+
+local barheight = 18
+local barcolor = gears.color({
+ type = "linear",
+ from = { barheight, 0 },
+ to = { barheight, barheight },
+ stops = { {0, theme.bg_focus }, {0.8, theme.border_normal}, {1, "#1A1A1A"} }
+})
+theme.titlebar_bg = barcolor
+
+theme.titlebar_bg_focus = gears.color({
+ type = "linear",
+ from = { barheight, 0 },
+ to = { barheight, barheight },
+ stops = { {0, theme.bg_normal}, {0.5, theme.border_normal}, {1, "#492417"} }
+})
+
+function theme.at_screen_connect(s)
+ -- Quake application
+ s.quake = lain.util.quake({ app = awful.util.terminal })
+
+ -- If wallpaper is a function, call it with the screen
+ local wallpaper = theme.wallpaper
+ if type(wallpaper) == "function" then
+ wallpaper = wallpaper(s)
+ end
+ gears.wallpaper.maximized(wallpaper, s, true)
+
+ -- Tags
+ awful.tag(awful.util.tagnames, s, awful.layout.layouts)
+
+ -- Create a promptbox for each screen
+ s.mypromptbox = awful.widget.prompt()
+ -- Create an imagebox widget which will contains an icon indicating which layout we're using.
+ -- We need one layoutbox per screen.
+ s.mylayoutbox = awful.widget.layoutbox(s)
+ s.mylayoutbox:buttons(my_table.join(
+ awful.button({}, 1, function () awful.layout.inc( 1) end),
+ awful.button({}, 2, function () awful.layout.set( awful.layout.layouts[1] ) end),
+ awful.button({}, 3, function () awful.layout.inc(-1) end),
+ awful.button({}, 4, function () awful.layout.inc( 1) end),
+ awful.button({}, 5, function () awful.layout.inc(-1) end)))
+
+ -- Create a taglist widget
+ s.mytaglist = awful.widget.taglist(s, awful.widget.taglist.filter.all, awful.util.taglist_buttons)
+
+ -- Create a tasklist widget
+ s.mytasklist = awful.widget.tasklist(s, awful.widget.tasklist.filter.currenttags, awful.util.tasklist_buttons, { bg_normal = barcolor, bg_focus = barcolor })
+
+ -- Create the wibox
+ s.mywibox = awful.wibar({ position = "top", screen = s, height = 18, bg = barcolor })
+
+ -- Add widgets to the wibox
+ s.mywibox:setup {
+ layout = wibox.layout.align.horizontal,
+ { -- Left widgets
+ layout = wibox.layout.fixed.horizontal,
+ first,
+ s.mytaglist,
+ arrl_pre,
+ s.mylayoutbox,
+ arrl_post,
+ s.mypromptbox,
+ first,
+ },
+ s.mytasklist, -- Middle widget
+ { -- Right widgets
+ layout = wibox.layout.fixed.horizontal,
+ wibox.widget.systray(),
+ first,
+ theme.mpd.widget,
+ --theme.mail.widget,
+ theme.weather.icon,
+ theme.weather.widget,
+ --theme.fs.widget,
+ bat,
+ theme.volume.widget,
+ mytextclock,
+ },
+ }
+end
+
+return theme
diff --git a/Phranque/awesome/themes/blackburn/wall.png b/Phranque/awesome/themes/blackburn/wall.png
new file mode 100644
index 00000000..6774d91a
Binary files /dev/null and b/Phranque/awesome/themes/blackburn/wall.png differ
diff --git a/Phranque/awesome/themes/copland/icons/ac.png b/Phranque/awesome/themes/copland/icons/ac.png
new file mode 100644
index 00000000..4b00cd14
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/ac.png differ
diff --git a/Phranque/awesome/themes/copland/icons/awesome.png b/Phranque/awesome/themes/copland/icons/awesome.png
new file mode 100644
index 00000000..b24e00d2
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/awesome.png differ
diff --git a/Phranque/awesome/themes/copland/icons/bat.png b/Phranque/awesome/themes/copland/icons/bat.png
new file mode 100644
index 00000000..4ea9b2bd
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/bat.png differ
diff --git a/Phranque/awesome/themes/copland/icons/bat_low.png b/Phranque/awesome/themes/copland/icons/bat_low.png
new file mode 100644
index 00000000..c1986752
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/bat_low.png differ
diff --git a/Phranque/awesome/themes/copland/icons/bat_no.png b/Phranque/awesome/themes/copland/icons/bat_no.png
new file mode 100644
index 00000000..5cf536fa
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/bat_no.png differ
diff --git a/Phranque/awesome/themes/copland/icons/centerfair.png b/Phranque/awesome/themes/copland/icons/centerfair.png
new file mode 100644
index 00000000..185d5e17
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/centerfair.png differ
diff --git a/Phranque/awesome/themes/copland/icons/centerwork.png b/Phranque/awesome/themes/copland/icons/centerwork.png
new file mode 100644
index 00000000..49d5a112
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/centerwork.png differ
diff --git a/Phranque/awesome/themes/copland/icons/disk.png b/Phranque/awesome/themes/copland/icons/disk.png
new file mode 100644
index 00000000..3ccd7288
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/disk.png differ
diff --git a/Phranque/awesome/themes/copland/icons/dwindle.png b/Phranque/awesome/themes/copland/icons/dwindle.png
new file mode 100644
index 00000000..55b61b0c
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/dwindle.png differ
diff --git a/Phranque/awesome/themes/copland/icons/fairh.png b/Phranque/awesome/themes/copland/icons/fairh.png
new file mode 100644
index 00000000..848f5a81
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/fairh.png differ
diff --git a/Phranque/awesome/themes/copland/icons/fairv.png b/Phranque/awesome/themes/copland/icons/fairv.png
new file mode 100644
index 00000000..610eb7bd
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/fairv.png differ
diff --git a/Phranque/awesome/themes/copland/icons/floating.png b/Phranque/awesome/themes/copland/icons/floating.png
new file mode 100644
index 00000000..d8ce00a5
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/floating.png differ
diff --git a/Phranque/awesome/themes/copland/icons/fullscreen.png b/Phranque/awesome/themes/copland/icons/fullscreen.png
new file mode 100644
index 00000000..feb44d10
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/fullscreen.png differ
diff --git a/Phranque/awesome/themes/copland/icons/magnifier.png b/Phranque/awesome/themes/copland/icons/magnifier.png
new file mode 100644
index 00000000..49616bc7
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/magnifier.png differ
diff --git a/Phranque/awesome/themes/copland/icons/max.png b/Phranque/awesome/themes/copland/icons/max.png
new file mode 100644
index 00000000..3f4bdece
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/max.png differ
diff --git a/Phranque/awesome/themes/copland/icons/pause.png b/Phranque/awesome/themes/copland/icons/pause.png
new file mode 100644
index 00000000..381e5325
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/pause.png differ
diff --git a/Phranque/awesome/themes/copland/icons/play.png b/Phranque/awesome/themes/copland/icons/play.png
new file mode 100644
index 00000000..05ad2533
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/play.png differ
diff --git a/Phranque/awesome/themes/copland/icons/spiral.png b/Phranque/awesome/themes/copland/icons/spiral.png
new file mode 100644
index 00000000..664afb45
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/spiral.png differ
diff --git a/Phranque/awesome/themes/copland/icons/square_sel.png b/Phranque/awesome/themes/copland/icons/square_sel.png
new file mode 100644
index 00000000..58b58f26
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/square_sel.png differ
diff --git a/Phranque/awesome/themes/copland/icons/square_unsel.png b/Phranque/awesome/themes/copland/icons/square_unsel.png
new file mode 100644
index 00000000..40411f44
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/square_unsel.png differ
diff --git a/Phranque/awesome/themes/copland/icons/stop.png b/Phranque/awesome/themes/copland/icons/stop.png
new file mode 100644
index 00000000..b01b75e8
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/stop.png differ
diff --git a/Phranque/awesome/themes/copland/icons/submenu.png b/Phranque/awesome/themes/copland/icons/submenu.png
new file mode 100644
index 00000000..10ed7397
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/submenu.png differ
diff --git a/Phranque/awesome/themes/copland/icons/termfair.png b/Phranque/awesome/themes/copland/icons/termfair.png
new file mode 100644
index 00000000..9cfb636e
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/termfair.png differ
diff --git a/Phranque/awesome/themes/copland/icons/tile.png b/Phranque/awesome/themes/copland/icons/tile.png
new file mode 100644
index 00000000..890e354f
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/tile.png differ
diff --git a/Phranque/awesome/themes/copland/icons/tilebottom.png b/Phranque/awesome/themes/copland/icons/tilebottom.png
new file mode 100644
index 00000000..aac4302e
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/tilebottom.png differ
diff --git a/Phranque/awesome/themes/copland/icons/tileleft.png b/Phranque/awesome/themes/copland/icons/tileleft.png
new file mode 100644
index 00000000..c87a32ff
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/tileleft.png differ
diff --git a/Phranque/awesome/themes/copland/icons/tiletop.png b/Phranque/awesome/themes/copland/icons/tiletop.png
new file mode 100644
index 00000000..89f7f8d5
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/tiletop.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/close_focus.png b/Phranque/awesome/themes/copland/icons/titlebar/close_focus.png
new file mode 100644
index 00000000..c1fa3ad1
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/close_focus.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/close_normal.png b/Phranque/awesome/themes/copland/icons/titlebar/close_normal.png
new file mode 100644
index 00000000..77cd8eb7
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/close_normal.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/floating_focus_active.png b/Phranque/awesome/themes/copland/icons/titlebar/floating_focus_active.png
new file mode 100644
index 00000000..8fcf336a
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/floating_focus_active.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/floating_focus_inactive.png b/Phranque/awesome/themes/copland/icons/titlebar/floating_focus_inactive.png
new file mode 100644
index 00000000..47f19f6d
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/floating_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/floating_normal_active.png b/Phranque/awesome/themes/copland/icons/titlebar/floating_normal_active.png
new file mode 100644
index 00000000..df17970e
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/floating_normal_active.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/floating_normal_inactive.png b/Phranque/awesome/themes/copland/icons/titlebar/floating_normal_inactive.png
new file mode 100644
index 00000000..98f7f1d7
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/floating_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/maximized_focus_active.png b/Phranque/awesome/themes/copland/icons/titlebar/maximized_focus_active.png
new file mode 100644
index 00000000..96e8937c
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/maximized_focus_active.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/maximized_focus_inactive.png b/Phranque/awesome/themes/copland/icons/titlebar/maximized_focus_inactive.png
new file mode 100644
index 00000000..bce1d000
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/maximized_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/maximized_normal_active.png b/Phranque/awesome/themes/copland/icons/titlebar/maximized_normal_active.png
new file mode 100644
index 00000000..6ce0b104
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/maximized_normal_active.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/maximized_normal_inactive.png b/Phranque/awesome/themes/copland/icons/titlebar/maximized_normal_inactive.png
new file mode 100644
index 00000000..2e56d327
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/maximized_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/ontop_focus_active.png b/Phranque/awesome/themes/copland/icons/titlebar/ontop_focus_active.png
new file mode 100644
index 00000000..d206ea41
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/ontop_focus_active.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/ontop_focus_inactive.png b/Phranque/awesome/themes/copland/icons/titlebar/ontop_focus_inactive.png
new file mode 100644
index 00000000..2f3a2be2
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/ontop_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/ontop_normal_active.png b/Phranque/awesome/themes/copland/icons/titlebar/ontop_normal_active.png
new file mode 100644
index 00000000..9a26d345
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/ontop_normal_active.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/ontop_normal_inactive.png b/Phranque/awesome/themes/copland/icons/titlebar/ontop_normal_inactive.png
new file mode 100644
index 00000000..a9a32068
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/ontop_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/sticky_focus_active.png b/Phranque/awesome/themes/copland/icons/titlebar/sticky_focus_active.png
new file mode 100644
index 00000000..7e360db0
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/sticky_focus_active.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/sticky_focus_inactive.png b/Phranque/awesome/themes/copland/icons/titlebar/sticky_focus_inactive.png
new file mode 100644
index 00000000..5493d8ec
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/sticky_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/sticky_normal_active.png b/Phranque/awesome/themes/copland/icons/titlebar/sticky_normal_active.png
new file mode 100644
index 00000000..a06138d8
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/sticky_normal_active.png differ
diff --git a/Phranque/awesome/themes/copland/icons/titlebar/sticky_normal_inactive.png b/Phranque/awesome/themes/copland/icons/titlebar/sticky_normal_inactive.png
new file mode 100644
index 00000000..7e6c99b6
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/titlebar/sticky_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/copland/icons/vol.png b/Phranque/awesome/themes/copland/icons/vol.png
new file mode 100644
index 00000000..847d5752
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/vol.png differ
diff --git a/Phranque/awesome/themes/copland/icons/vol_low.png b/Phranque/awesome/themes/copland/icons/vol_low.png
new file mode 100644
index 00000000..54d653e5
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/vol_low.png differ
diff --git a/Phranque/awesome/themes/copland/icons/vol_mute.png b/Phranque/awesome/themes/copland/icons/vol_mute.png
new file mode 100644
index 00000000..361412d7
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/vol_mute.png differ
diff --git a/Phranque/awesome/themes/copland/icons/vol_no.png b/Phranque/awesome/themes/copland/icons/vol_no.png
new file mode 100644
index 00000000..06d1a97c
Binary files /dev/null and b/Phranque/awesome/themes/copland/icons/vol_no.png differ
diff --git a/Phranque/awesome/themes/copland/theme.lua b/Phranque/awesome/themes/copland/theme.lua
new file mode 100644
index 00000000..d3ec23b0
--- /dev/null
+++ b/Phranque/awesome/themes/copland/theme.lua
@@ -0,0 +1,371 @@
+--[[
+
+ Copland Awesome WM theme 2.0
+ github.com/lcpz
+
+--]]
+
+local gears = require("gears")
+local lain = require("lain")
+local awful = require("awful")
+local wibox = require("wibox")
+
+local awesome, client, os = awesome, client, os
+local my_table = awful.util.table or gears.table -- 4.{0,1} compatibility
+
+local theme = {}
+theme.dir = os.getenv("HOME") .. "/.config/awesome/themes/copland"
+theme.wallpaper = theme.dir .. "/wall.png"
+theme.font = "Misc Tamsyn 10.5"
+theme.fg_normal = "#BBBBBB"
+theme.fg_focus = "#78A4FF"
+theme.bg_normal = "#111111"
+theme.bg_focus = "#111111"
+theme.fg_urgent = "#000000"
+theme.bg_urgent = "#FFFFFF"
+theme.border_width = 1
+theme.border_normal = "#141414"
+theme.border_focus = "#93B6FF"
+theme.taglist_fg_focus = "#FFFFFF"
+theme.taglist_bg_focus = "#111111"
+theme.taglist_bg_normal = "#111111"
+theme.titlebar_bg_normal = "#191919"
+theme.titlebar_bg_focus = "#262626"
+theme.menu_height = 16
+theme.menu_width = 130
+theme.tasklist_disable_icon = true
+theme.awesome_icon = theme.dir .."/icons/awesome.png"
+theme.menu_submenu_icon = theme.dir .. "/icons/submenu.png"
+theme.taglist_squares_sel = theme.dir .. "/icons/square_unsel.png"
+theme.taglist_squares_unsel = theme.dir .. "/icons/square_unsel.png"
+theme.vol = theme.dir .. "/icons/vol.png"
+theme.vol_low = theme.dir .. "/icons/vol_low.png"
+theme.vol_no = theme.dir .. "/icons/vol_no.png"
+theme.vol_mute = theme.dir .. "/icons/vol_mute.png"
+theme.disk = theme.dir .. "/icons/disk.png"
+theme.ac = theme.dir .. "/icons/ac.png"
+theme.bat = theme.dir .. "/icons/bat.png"
+theme.bat_low = theme.dir .. "/icons/bat_low.png"
+theme.bat_no = theme.dir .. "/icons/bat_no.png"
+theme.play = theme.dir .. "/icons/play.png"
+theme.pause = theme.dir .. "/icons/pause.png"
+theme.stop = theme.dir .. "/icons/stop.png"
+theme.layout_tile = theme.dir .. "/icons/tile.png"
+theme.layout_tileleft = theme.dir .. "/icons/tileleft.png"
+theme.layout_tilebottom = theme.dir .. "/icons/tilebottom.png"
+theme.layout_tiletop = theme.dir .. "/icons/tiletop.png"
+theme.layout_fairv = theme.dir .. "/icons/fairv.png"
+theme.layout_fairh = theme.dir .. "/icons/fairh.png"
+theme.layout_spiral = theme.dir .. "/icons/spiral.png"
+theme.layout_dwindle = theme.dir .. "/icons/dwindle.png"
+theme.layout_max = theme.dir .. "/icons/max.png"
+theme.layout_fullscreen = theme.dir .. "/icons/fullscreen.png"
+theme.layout_magnifier = theme.dir .. "/icons/magnifier.png"
+theme.layout_floating = theme.dir .. "/icons/floating.png"
+theme.useless_gap = 0
+theme.titlebar_close_button_focus = theme.dir .. "/icons/titlebar/close_focus.png"
+theme.titlebar_close_button_normal = theme.dir .. "/icons/titlebar/close_normal.png"
+theme.titlebar_ontop_button_focus_active = theme.dir .. "/icons/titlebar/ontop_focus_active.png"
+theme.titlebar_ontop_button_normal_active = theme.dir .. "/icons/titlebar/ontop_normal_active.png"
+theme.titlebar_ontop_button_focus_inactive = theme.dir .. "/icons/titlebar/ontop_focus_inactive.png"
+theme.titlebar_ontop_button_normal_inactive = theme.dir .. "/icons/titlebar/ontop_normal_inactive.png"
+theme.titlebar_sticky_button_focus_active = theme.dir .. "/icons/titlebar/sticky_focus_active.png"
+theme.titlebar_sticky_button_normal_active = theme.dir .. "/icons/titlebar/sticky_normal_active.png"
+theme.titlebar_sticky_button_focus_inactive = theme.dir .. "/icons/titlebar/sticky_focus_inactive.png"
+theme.titlebar_sticky_button_normal_inactive = theme.dir .. "/icons/titlebar/sticky_normal_inactive.png"
+theme.titlebar_floating_button_focus_active = theme.dir .. "/icons/titlebar/floating_focus_active.png"
+theme.titlebar_floating_button_normal_active = theme.dir .. "/icons/titlebar/floating_normal_active.png"
+theme.titlebar_floating_button_focus_inactive = theme.dir .. "/icons/titlebar/floating_focus_inactive.png"
+theme.titlebar_floating_button_normal_inactive = theme.dir .. "/icons/titlebar/floating_normal_inactive.png"
+theme.titlebar_maximized_button_focus_active = theme.dir .. "/icons/titlebar/maximized_focus_active.png"
+theme.titlebar_maximized_button_normal_active = theme.dir .. "/icons/titlebar/maximized_normal_active.png"
+theme.titlebar_maximized_button_focus_inactive = theme.dir .. "/icons/titlebar/maximized_focus_inactive.png"
+theme.titlebar_maximized_button_normal_inactive = theme.dir .. "/icons/titlebar/maximized_normal_inactive.png"
+
+-- lain related
+theme.layout_centerfair = theme.dir .. "/icons/centerfair.png"
+theme.layout_termfair = theme.dir .. "/icons/termfair.png"
+theme.layout_centerwork = theme.dir .. "/icons/centerwork.png"
+
+local markup = lain.util.markup
+local blue = theme.fg_focus
+local red = "#EB8F8F"
+local green = "#8FEB8F"
+
+-- Textclock
+--os.setlocale(os.getenv("LANG")) -- to localize the clock
+local mytextclock = wibox.widget.textclock(" %H:%M ")
+mytextclock.font = theme.font
+
+-- Calendar
+theme.cal = lain.widget.cal({
+ attach_to = { mytextclock },
+ notification_preset = {
+ font = "Misc Tamsyn 11",
+ fg = theme.fg_normal,
+ bg = theme.bg_normal
+ }
+})
+
+-- Mail IMAP check
+--[[ commented because it needs to be set before use
+theme.mail = lain.widget.imap({
+ timeout = 180,
+ server = "server",
+ mail = "mail",
+ password = "keyring get mail",
+ settings = function()
+ mail = ""
+ count = ""
+
+ if mailcount > 0 then
+ mail = " Mail "
+ count = mailcount .. " "
+ end
+
+ widget:set_markup(markup(blue, mail) .. count)
+ end
+})
+--]]
+
+-- MPD
+local mpdicon = wibox.widget.imagebox()
+theme.mpd = lain.widget.mpd({
+ settings = function()
+ if mpd_now.state == "play" then
+ title = mpd_now.title
+ artist = " " .. mpd_now.artist .. markup("#777777", " | ")
+ mpdicon:set_image(theme.play)
+ elseif mpd_now.state == "pause" then
+ title = "mpd "
+ artist = "paused" .. markup("#777777", " | ")
+ mpdicon:set_image(theme.pause)
+ else
+ title = ""
+ artist = ""
+ mpdicon._private.image = nil
+ mpdicon:emit_signal("widget::redraw_needed")
+ mpdicon:emit_signal("widget::layout_changed")
+ end
+
+ widget:set_markup(markup.font(theme.font, markup(blue, title) .. artist))
+ end
+})
+
+-- Battery
+local baticon = wibox.widget.imagebox(theme.bat)
+local batbar = wibox.widget {
+ forced_height = 1,
+ forced_width = 59,
+ color = theme.fg_normal,
+ background_color = theme.bg_normal,
+ margins = 1,
+ paddings = 1,
+ ticks = true,
+ ticks_size = 6,
+ widget = wibox.widget.progressbar,
+}
+local batupd = lain.widget.bat({
+ settings = function()
+ if (not bat_now.status) or bat_now.status == "N/A" or type(bat_now.perc) ~= "number" then return end
+
+ if bat_now.status == "Charging" then
+ baticon:set_image(theme.ac)
+ if bat_now.perc >= 98 then
+ batbar:set_color(green)
+ elseif bat_now.perc > 50 then
+ batbar:set_color(theme.fg_normal)
+ elseif bat_now.perc > 15 then
+ batbar:set_color(theme.fg_normal)
+ else
+ batbar:set_color(red)
+ end
+ else
+ if bat_now.perc >= 98 then
+ batbar:set_color(green)
+ elseif bat_now.perc > 50 then
+ batbar:set_color(theme.fg_normal)
+ baticon:set_image(theme.bat)
+ elseif bat_now.perc > 15 then
+ batbar:set_color(theme.fg_normal)
+ baticon:set_image(theme.bat_low)
+ else
+ batbar:set_color(red)
+ baticon:set_image(theme.bat_no)
+ end
+ end
+ batbar:set_value(bat_now.perc / 100)
+ end
+})
+local batbg = wibox.container.background(batbar, "#474747", gears.shape.rectangle)
+local batwidget = wibox.container.margin(batbg, 2, 7, 4, 4)
+
+-- /home fs
+--[[ commented because it needs Gio/Glib >= 2.54
+local fsicon = wibox.widget.imagebox(theme.disk)
+local fsbar = wibox.widget {
+ forced_height = 1,
+ forced_width = 59,
+ color = theme.fg_normal,
+ background_color = theme.bg_normal,
+ margins = 1,
+ paddings = 1,
+ ticks = true,
+ ticks_size = 6,
+ widget = wibox.widget.progressbar,
+}
+theme.fs = lain.widget.fs {
+ notification_preset = { fg = theme.fg_normal, bg = theme.bg_normal, font = "Misc Tamsyn 10.5" },
+ settings = function()
+ if fs_now["/home"].percentage < 90 then
+ fsbar:set_color(theme.fg_normal)
+ else
+ fsbar:set_color("#EB8F8F")
+ end
+ fsbar:set_value(fs_now["/home"].percentage / 100)
+ end
+}
+local fsbg = wibox.container.background(fsbar, "#474747", gears.shape.rectangle)
+local fswidget = wibox.container.margin(fsbg, 2, 7, 4, 4)
+--]]
+
+-- ALSA volume bar
+local volicon = wibox.widget.imagebox(theme.vol)
+theme.volume = lain.widget.alsabar {
+ width = 59, border_width = 0, ticks = true, ticks_size = 6,
+ notification_preset = { font = theme.font },
+ --togglechannel = "IEC958,3",
+ settings = function()
+ if volume_now.status == "off" then
+ volicon:set_image(theme.vol_mute)
+ elseif volume_now.level == 0 then
+ volicon:set_image(theme.vol_no)
+ elseif volume_now.level <= 50 then
+ volicon:set_image(theme.vol_low)
+ else
+ volicon:set_image(theme.vol)
+ end
+ end,
+ colors = {
+ background = theme.bg_normal,
+ mute = red,
+ unmute = theme.fg_normal
+ }
+}
+theme.volume.tooltip.wibox.fg = theme.fg_focus
+theme.volume.bar:buttons(my_table.join (
+ awful.button({}, 1, function()
+ awful.spawn(string.format("%s -e alsamixer", awful.util.terminal))
+ end),
+ awful.button({}, 2, function()
+ os.execute(string.format("%s set %s 100%%", theme.volume.cmd, theme.volume.channel))
+ theme.volume.update()
+ end),
+ awful.button({}, 3, function()
+ os.execute(string.format("%s set %s toggle", theme.volume.cmd, theme.volume.togglechannel or theme.volume.channel))
+ theme.volume.update()
+ end),
+ awful.button({}, 4, function()
+ os.execute(string.format("%s set %s 1%%+", theme.volume.cmd, theme.volume.channel))
+ theme.volume.update()
+ end),
+ awful.button({}, 5, function()
+ os.execute(string.format("%s set %s 1%%-", theme.volume.cmd, theme.volume.channel))
+ theme.volume.update()
+ end)
+))
+local volumebg = wibox.container.background(theme.volume.bar, "#474747", gears.shape.rectangle)
+local volumewidget = wibox.container.margin(volumebg, 2, 7, 4, 4)
+
+-- Weather
+theme.weather = lain.widget.weather({
+ city_id = 2643743, -- placeholder (London)
+})
+
+-- Separators
+local first = wibox.widget.textbox(markup.font("Misc Tamsyn 3", " "))
+local spr = wibox.widget.textbox(' ')
+local small_spr = wibox.widget.textbox(markup.font("Misc Tamsyn 4", " "))
+local bar_spr = wibox.widget.textbox(markup.font("Misc Tamsyn 3", " ") .. markup.fontfg(theme.font, "#777777", "|") .. markup.font("Misc Tamsyn 5", " "))
+
+-- Eminent-like task filtering
+local orig_filter = awful.widget.taglist.filter.all
+
+-- Taglist label functions
+awful.widget.taglist.filter.all = function (t, args)
+ if t.selected or #t:clients() > 0 then
+ return orig_filter(t, args)
+ end
+end
+
+function theme.at_screen_connect(s)
+ -- Quake application
+ s.quake = lain.util.quake({ app = awful.util.terminal })
+
+ -- If wallpaper is a function, call it with the screen
+ local wallpaper = theme.wallpaper
+ if type(wallpaper) == "function" then
+ wallpaper = wallpaper(s)
+ end
+ gears.wallpaper.maximized(wallpaper, s, true)
+
+ -- Tags
+ awful.tag(awful.util.tagnames, s, awful.layout.layouts)
+
+ -- Create a promptbox for each screen
+ s.mypromptbox = awful.widget.prompt()
+ -- Create an imagebox widget which will contains an icon indicating which layout we're using.
+ -- We need one layoutbox per screen.
+ s.mylayoutbox = awful.widget.layoutbox(s)
+ s.mylayoutbox:buttons(my_table.join(
+ awful.button({}, 1, function () awful.layout.inc( 1) end),
+ awful.button({}, 2, function () awful.layout.set( awful.layout.layouts[1] ) end),
+ awful.button({}, 3, function () awful.layout.inc(-1) end),
+ awful.button({}, 4, function () awful.layout.inc( 1) end),
+ awful.button({}, 5, function () awful.layout.inc(-1) end)))
+
+ -- Create a taglist widget
+ s.mytaglist = awful.widget.taglist(s, awful.widget.taglist.filter.all, awful.util.taglist_buttons)
+
+ -- Create a tasklist widget
+ s.mytasklist = awful.widget.tasklist(s, awful.widget.tasklist.filter.currenttags, awful.util.tasklist_buttons)
+
+ -- Create the wibox
+ s.mywibox = awful.wibar({ position = "top", screen = s, height = 18, bg = theme.bg_normal, fg = theme.fg_normal })
+
+ -- Add widgets to the wibox
+ s.mywibox:setup {
+ layout = wibox.layout.align.horizontal,
+ { -- Left widgets
+ layout = wibox.layout.fixed.horizontal,
+ small_spr,
+ s.mylayoutbox,
+ first,
+ bar_spr,
+ s.mytaglist,
+ first,
+ s.mypromptbox,
+ },
+ s.mytasklist, -- Middle widget
+ { -- Right widgets
+ layout = wibox.layout.fixed.horizontal,
+ wibox.widget.systray(),
+ small_spr,
+ --theme.mail.widget,
+ mpdicon,
+ theme.mpd.widget,
+ baticon,
+ batwidget,
+ bar_spr,
+ --fsicon,
+ --fswidget,
+ bar_spr,
+ volicon,
+ volumewidget,
+ bar_spr,
+ mytextclock,
+ },
+ }
+end
+
+return theme
diff --git a/Phranque/awesome/themes/copland/wall.png b/Phranque/awesome/themes/copland/wall.png
new file mode 100644
index 00000000..aa3ce513
Binary files /dev/null and b/Phranque/awesome/themes/copland/wall.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/awesome.png b/Phranque/awesome/themes/dremora/icons/awesome.png
new file mode 100644
index 00000000..ffb105e6
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/awesome.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/dwindle.png b/Phranque/awesome/themes/dremora/icons/dwindle.png
new file mode 100644
index 00000000..04f3918b
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/dwindle.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/fairh.png b/Phranque/awesome/themes/dremora/icons/fairh.png
new file mode 100644
index 00000000..0bb6dab0
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/fairh.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/fairv.png b/Phranque/awesome/themes/dremora/icons/fairv.png
new file mode 100644
index 00000000..aff6c26c
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/fairv.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/floating.png b/Phranque/awesome/themes/dremora/icons/floating.png
new file mode 100644
index 00000000..5d021b16
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/floating.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/magnifier.png b/Phranque/awesome/themes/dremora/icons/magnifier.png
new file mode 100644
index 00000000..dd4dec30
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/magnifier.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/max.png b/Phranque/awesome/themes/dremora/icons/max.png
new file mode 100644
index 00000000..7334f8bd
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/max.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/spiral.png b/Phranque/awesome/themes/dremora/icons/spiral.png
new file mode 100644
index 00000000..31cbbd02
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/spiral.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/square_sel.png b/Phranque/awesome/themes/dremora/icons/square_sel.png
new file mode 100644
index 00000000..85cc605d
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/square_sel.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/square_unsel.png b/Phranque/awesome/themes/dremora/icons/square_unsel.png
new file mode 100644
index 00000000..1c58e4b8
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/square_unsel.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/submenu.png b/Phranque/awesome/themes/dremora/icons/submenu.png
new file mode 100644
index 00000000..988ef9e3
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/submenu.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/tile.png b/Phranque/awesome/themes/dremora/icons/tile.png
new file mode 100644
index 00000000..72cbd6c0
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/tile.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/tilebottom.png b/Phranque/awesome/themes/dremora/icons/tilebottom.png
new file mode 100644
index 00000000..a927128d
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/tilebottom.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/tileleft.png b/Phranque/awesome/themes/dremora/icons/tileleft.png
new file mode 100644
index 00000000..c54d1f59
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/tileleft.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/tiletop.png b/Phranque/awesome/themes/dremora/icons/tiletop.png
new file mode 100644
index 00000000..db74cf30
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/tiletop.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/close_focus.png b/Phranque/awesome/themes/dremora/icons/titlebar/close_focus.png
new file mode 100644
index 00000000..221bca0c
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/close_focus.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/close_normal.png b/Phranque/awesome/themes/dremora/icons/titlebar/close_normal.png
new file mode 100644
index 00000000..f9239837
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/close_normal.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/floating_focus_active.png b/Phranque/awesome/themes/dremora/icons/titlebar/floating_focus_active.png
new file mode 100644
index 00000000..d83bd00a
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/floating_focus_active.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/floating_focus_inactive.png b/Phranque/awesome/themes/dremora/icons/titlebar/floating_focus_inactive.png
new file mode 100644
index 00000000..47f19f6d
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/floating_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/floating_normal_active.png b/Phranque/awesome/themes/dremora/icons/titlebar/floating_normal_active.png
new file mode 100644
index 00000000..b7885525
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/floating_normal_active.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/floating_normal_inactive.png b/Phranque/awesome/themes/dremora/icons/titlebar/floating_normal_inactive.png
new file mode 100644
index 00000000..4adc5e9d
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/floating_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/maximized_focus_active.png b/Phranque/awesome/themes/dremora/icons/titlebar/maximized_focus_active.png
new file mode 100644
index 00000000..2599cd51
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/maximized_focus_active.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/maximized_focus_inactive.png b/Phranque/awesome/themes/dremora/icons/titlebar/maximized_focus_inactive.png
new file mode 100644
index 00000000..bce1d000
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/maximized_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/maximized_normal_active.png b/Phranque/awesome/themes/dremora/icons/titlebar/maximized_normal_active.png
new file mode 100644
index 00000000..a5b1d766
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/maximized_normal_active.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/maximized_normal_inactive.png b/Phranque/awesome/themes/dremora/icons/titlebar/maximized_normal_inactive.png
new file mode 100644
index 00000000..2e56d327
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/maximized_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/ontop_focus_active.png b/Phranque/awesome/themes/dremora/icons/titlebar/ontop_focus_active.png
new file mode 100644
index 00000000..ade12fe3
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/ontop_focus_active.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/ontop_focus_inactive.png b/Phranque/awesome/themes/dremora/icons/titlebar/ontop_focus_inactive.png
new file mode 100644
index 00000000..2f3a2be2
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/ontop_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/ontop_normal_active.png b/Phranque/awesome/themes/dremora/icons/titlebar/ontop_normal_active.png
new file mode 100644
index 00000000..02facc40
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/ontop_normal_active.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/ontop_normal_inactive.png b/Phranque/awesome/themes/dremora/icons/titlebar/ontop_normal_inactive.png
new file mode 100644
index 00000000..a9a32068
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/ontop_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/sticky_focus_active.png b/Phranque/awesome/themes/dremora/icons/titlebar/sticky_focus_active.png
new file mode 100644
index 00000000..ebd45a1f
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/sticky_focus_active.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/sticky_focus_inactive.png b/Phranque/awesome/themes/dremora/icons/titlebar/sticky_focus_inactive.png
new file mode 100644
index 00000000..5493d8ec
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/sticky_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/sticky_normal_active.png b/Phranque/awesome/themes/dremora/icons/titlebar/sticky_normal_active.png
new file mode 100644
index 00000000..15f0bc46
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/sticky_normal_active.png differ
diff --git a/Phranque/awesome/themes/dremora/icons/titlebar/sticky_normal_inactive.png b/Phranque/awesome/themes/dremora/icons/titlebar/sticky_normal_inactive.png
new file mode 100644
index 00000000..7e6c99b6
Binary files /dev/null and b/Phranque/awesome/themes/dremora/icons/titlebar/sticky_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/dremora/theme.lua b/Phranque/awesome/themes/dremora/theme.lua
new file mode 100644
index 00000000..fee78fe3
--- /dev/null
+++ b/Phranque/awesome/themes/dremora/theme.lua
@@ -0,0 +1,252 @@
+--[[
+
+ Dremora Awesome WM theme 2.0
+ github.com/lcpz
+
+--]]
+
+local gears = require("gears")
+local lain = require("lain")
+local awful = require("awful")
+local wibox = require("wibox")
+
+local os = os
+local my_table = awful.util.table or gears.table -- 4.{0,1} compatibility
+
+local theme = {}
+theme.dir = os.getenv("HOME") .. "/.config/awesome/themes/dremora"
+theme.wallpaper = theme.dir .. "/wall.png"
+theme.font = "Misc Tamsyn 10.5"
+theme.taglist_font = "Icons 10"
+theme.fg_normal = "#747474"
+theme.fg_focus = "#DDDCFF"
+theme.bg_normal = "#121212"
+theme.bg_focus = "#121212"
+theme.fg_urgent = "#CC9393"
+theme.bg_urgent = "#2A1F1E"
+theme.border_width = "0"
+theme.border_normal = "#121212"
+theme.border_focus = "#292929"
+theme.titlebar_bg_focus = "#292929"
+theme.taglist_fg_focus = "#dddcff"
+theme.taglist_bg_focus = "#121212"
+theme.menu_height = 16
+theme.menu_width = 130
+theme.menu_submenu_icon = theme.dir .. "/icons/submenu.png"
+theme.awesome_icon = theme.dir .."/icons/awesome.png"
+theme.taglist_squares_sel = theme.dir .. "/icons/square_sel.png"
+theme.taglist_squares_unsel = theme.dir .. "/icons/square_unsel.png"
+theme.layout_tile = theme.dir .. "/icons/tile.png"
+theme.layout_tileleft = theme.dir .. "/icons/tileleft.png"
+theme.layout_tilebottom = theme.dir .. "/icons/tilebottom.png"
+theme.layout_tiletop = theme.dir .. "/icons/tiletop.png"
+theme.layout_fairv = theme.dir .. "/icons/fairv.png"
+theme.layout_fairh = theme.dir .. "/icons/fairh.png"
+theme.layout_spiral = theme.dir .. "/icons/spiral.png"
+theme.layout_dwindle = theme.dir .. "/icons/dwindle.png"
+theme.layout_max = theme.dir .. "/icons/max.png"
+theme.layout_fullscreen = theme.dir .. "/icons/fullscreen.png"
+theme.layout_magnifier = theme.dir .. "/icons/magnifier.png"
+theme.layout_floating = theme.dir .. "/icons/floating.png"
+theme.tasklist_plain_task_name = true
+theme.tasklist_disable_icon = true
+theme.useless_gap = 10
+theme.titlebar_close_button_focus = theme.dir .. "/icons/titlebar/close_focus.png"
+theme.titlebar_close_button_normal = theme.dir .. "/icons/titlebar/close_normal.png"
+theme.titlebar_ontop_button_focus_active = theme.dir .. "/icons/titlebar/ontop_focus_active.png"
+theme.titlebar_ontop_button_normal_active = theme.dir .. "/icons/titlebar/ontop_normal_active.png"
+theme.titlebar_ontop_button_focus_inactive = theme.dir .. "/icons/titlebar/ontop_focus_inactive.png"
+theme.titlebar_ontop_button_normal_inactive = theme.dir .. "/icons/titlebar/ontop_normal_inactive.png"
+theme.titlebar_sticky_button_focus_active = theme.dir .. "/icons/titlebar/sticky_focus_active.png"
+theme.titlebar_sticky_button_normal_active = theme.dir .. "/icons/titlebar/sticky_normal_active.png"
+theme.titlebar_sticky_button_focus_inactive = theme.dir .. "/icons/titlebar/sticky_focus_inactive.png"
+theme.titlebar_sticky_button_normal_inactive = theme.dir .. "/icons/titlebar/sticky_normal_inactive.png"
+theme.titlebar_floating_button_focus_active = theme.dir .. "/icons/titlebar/floating_focus_active.png"
+theme.titlebar_floating_button_normal_active = theme.dir .. "/icons/titlebar/floating_normal_active.png"
+theme.titlebar_floating_button_focus_inactive = theme.dir .. "/icons/titlebar/floating_focus_inactive.png"
+theme.titlebar_floating_button_normal_inactive = theme.dir .. "/icons/titlebar/floating_normal_inactive.png"
+theme.titlebar_maximized_button_focus_active = theme.dir .. "/icons/titlebar/maximized_focus_active.png"
+theme.titlebar_maximized_button_normal_active = theme.dir .. "/icons/titlebar/maximized_normal_active.png"
+theme.titlebar_maximized_button_focus_inactive = theme.dir .. "/icons/titlebar/maximized_focus_inactive.png"
+theme.titlebar_maximized_button_normal_inactive = theme.dir .. "/icons/titlebar/maximized_normal_inactive.png"
+
+awful.util.tagnames = { "ƀ", "Ƅ", "Ɗ", "ƈ", "ƙ" }
+
+local markup = lain.util.markup
+local separators = lain.util.separators
+local white = theme.fg_focus
+local gray = "#858585"
+
+-- Textclock
+local mytextclock = wibox.widget.textclock(markup(gray, " %a")
+.. markup(white, " %d ") .. markup(gray, "%b ") .. markup(white, "%H:%M "))
+mytextclock.font = theme.font
+
+-- Calendar
+theme.cal = lain.widget.cal({
+ attach_to = { mytextclock },
+ notification_preset = {
+ font = "Misc Tamsyn 11",
+ fg = white,
+ bg = theme.bg_normal
+}})
+
+-- Mail IMAP check
+--[[ commented because it needs to be set before use
+theme.mail = lain.widget.imap({
+ timeout = 180,
+ server = "server",
+ mail = "mail",
+ password = "keyring get mail",
+ notification_preset = { fg = white }
+ settings = function()
+ mail = ""
+ count = ""
+
+ if mailcount > 0 then
+ mail = "Mail "
+ count = mailcount .. " "
+ end
+
+ widget:set_markup(markup.font(theme.font, markup(gray, mail) .. markup(white, count)))
+ end
+})
+--]]
+
+-- MPD
+theme.mpd = lain.widget.mpd({
+ settings = function()
+ mpd_notification_preset.fg = white
+ artist = mpd_now.artist .. " "
+ title = mpd_now.title .. " "
+
+ if mpd_now.state == "pause" then
+ artist = "mpd "
+ title = "paused "
+ elseif mpd_now.state == "stop" then
+ artist = ""
+ title = ""
+ end
+
+ widget:set_markup(markup.font(theme.font, markup(gray, artist) .. markup(white, title)))
+ end
+})
+
+-- /home fs
+--[[ commented because it needs Gio/Glib >= 2.54
+theme.fs = lain.widget.fs({
+ notification_preset = { fg = white, bg = theme.bg_normal, font = "Misc Tamsyn 10.5" },
+ settings = function()
+ fs_header = ""
+ fs_p = ""
+
+ if fs_now["/home"].percentage >= 90 then
+ fs_header = " Hdd "
+ fs_p = fs_now["/home"].percentage
+ end
+
+ widget:set_markup(markup.font(theme.font, markup(gray, fs_header) .. markup(white, fs_p)))
+ end
+})
+--]]
+
+-- Battery
+local bat = lain.widget.bat({
+ settings = function()
+ bat_header = " Bat "
+ bat_p = bat_now.perc .. " "
+ widget:set_markup(markup.font(theme.font, markup(gray, bat_header) .. markup(white, bat_p)))
+ end
+})
+
+-- ALSA volume
+theme.volume = lain.widget.alsa({
+ --togglechannel = "IEC958,3",
+ settings = function()
+ header = " Vol "
+ vlevel = volume_now.level
+
+ if volume_now.status == "off" then
+ vlevel = vlevel .. "M "
+ else
+ vlevel = vlevel .. " "
+ end
+
+ widget:set_markup(markup.font(theme.font, markup(gray, header) .. markup(white, vlevel)))
+ end
+})
+
+-- Weather
+theme.weather = lain.widget.weather({
+ city_id = 2643743, -- placeholder (London)
+ notification_preset = { fg = white }
+})
+
+-- Separators
+local first = wibox.widget.textbox(' ')
+local arrl_pre = separators.arrow_right("alpha", "#1A1A1A")
+local arrl_post = separators.arrow_right("#1A1A1A", "alpha")
+
+function theme.at_screen_connect(s)
+ -- Quake application
+ s.quake = lain.util.quake({ app = awful.util.terminal })
+
+ -- If wallpaper is a function, call it with the screen
+ local wallpaper = theme.wallpaper
+ if type(wallpaper) == "function" then
+ wallpaper = wallpaper(s)
+ end
+ gears.wallpaper.maximized(wallpaper, s, true)
+
+ -- Tags
+ awful.tag(awful.util.tagnames, s, awful.layout.layouts)
+
+ -- Create a promptbox for each screen
+ s.mypromptbox = awful.widget.prompt()
+ -- Create an imagebox widget which will contains an icon indicating which layout we're using.
+ -- We need one layoutbox per screen.
+ s.mylayoutbox = awful.widget.layoutbox(s)
+ s.mylayoutbox:buttons(my_table.join(
+ awful.button({}, 1, function () awful.layout.inc( 1) end),
+ awful.button({}, 2, function () awful.layout.set( awful.layout.layouts[1] ) end),
+ awful.button({}, 3, function () awful.layout.inc(-1) end),
+ awful.button({}, 4, function () awful.layout.inc( 1) end),
+ awful.button({}, 5, function () awful.layout.inc(-1) end)))
+ -- Create a taglist widget
+ s.mytaglist = awful.widget.taglist(s, awful.widget.taglist.filter.all, awful.util.taglist_buttons)
+
+ -- Create a tasklist widget
+ s.mytasklist = awful.widget.tasklist(s, awful.widget.tasklist.filter.currenttags, awful.util.tasklist_buttons)
+
+ -- Create the wibox
+ s.mywibox = awful.wibar({ position = "top", screen = s, height = 18, bg = theme.bg_normal, fg = theme.fg_normal })
+
+ -- Add widgets to the wibox
+ s.mywibox:setup {
+ layout = wibox.layout.align.horizontal,
+ { -- Left widgets
+ layout = wibox.layout.fixed.horizontal,
+ first,
+ s.mytaglist,
+ arrl_pre,
+ s.mylayoutbox,
+ arrl_post,
+ s.mypromptbox,
+ first,
+ },
+ s.mytasklist, -- Middle widget
+ { -- Right widgets
+ layout = wibox.layout.fixed.horizontal,
+ wibox.widget.systray(),
+ first,
+ theme.mpd.widget,
+ --theme.mail.widget,
+ --theme.fs.widget,
+ bat.widget,
+ theme.volume.widget,
+ mytextclock,
+ },
+ }
+end
+
+return theme
diff --git a/Phranque/awesome/themes/dremora/wall.png b/Phranque/awesome/themes/dremora/wall.png
new file mode 100644
index 00000000..4501b606
Binary files /dev/null and b/Phranque/awesome/themes/dremora/wall.png differ
diff --git a/Phranque/awesome/themes/holo/icons/awesome_icon.png b/Phranque/awesome/themes/holo/icons/awesome_icon.png
new file mode 100644
index 00000000..ae13162b
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/awesome_icon.png differ
diff --git a/Phranque/awesome/themes/holo/icons/awesome_icon_white.png b/Phranque/awesome/themes/holo/icons/awesome_icon_white.png
new file mode 100644
index 00000000..1b0de6e6
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/awesome_icon_white.png differ
diff --git a/Phranque/awesome/themes/holo/icons/bar.png b/Phranque/awesome/themes/holo/icons/bar.png
new file mode 100644
index 00000000..767885cf
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/bar.png differ
diff --git a/Phranque/awesome/themes/holo/icons/bottom_bar.png b/Phranque/awesome/themes/holo/icons/bottom_bar.png
new file mode 100644
index 00000000..076ad3b1
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/bottom_bar.png differ
diff --git a/Phranque/awesome/themes/holo/icons/cal.png b/Phranque/awesome/themes/holo/icons/cal.png
new file mode 100644
index 00000000..1b023ecc
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/cal.png differ
diff --git a/Phranque/awesome/themes/holo/icons/clock.png b/Phranque/awesome/themes/holo/icons/clock.png
new file mode 100644
index 00000000..6b2c4e54
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/clock.png differ
diff --git a/Phranque/awesome/themes/holo/icons/cpu.png b/Phranque/awesome/themes/holo/icons/cpu.png
new file mode 100644
index 00000000..1ce87eb8
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/cpu.png differ
diff --git a/Phranque/awesome/themes/holo/icons/dwindle.png b/Phranque/awesome/themes/holo/icons/dwindle.png
new file mode 100644
index 00000000..f68b4842
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/dwindle.png differ
diff --git a/Phranque/awesome/themes/holo/icons/fairh.png b/Phranque/awesome/themes/holo/icons/fairh.png
new file mode 100644
index 00000000..c7e5e6d8
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/fairh.png differ
diff --git a/Phranque/awesome/themes/holo/icons/fairv.png b/Phranque/awesome/themes/holo/icons/fairv.png
new file mode 100644
index 00000000..d416b9d9
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/fairv.png differ
diff --git a/Phranque/awesome/themes/holo/icons/floating.png b/Phranque/awesome/themes/holo/icons/floating.png
new file mode 100644
index 00000000..b91cd070
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/floating.png differ
diff --git a/Phranque/awesome/themes/holo/icons/magnifier.png b/Phranque/awesome/themes/holo/icons/magnifier.png
new file mode 100644
index 00000000..e1f744ec
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/magnifier.png differ
diff --git a/Phranque/awesome/themes/holo/icons/max.png b/Phranque/awesome/themes/holo/icons/max.png
new file mode 100644
index 00000000..50d39945
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/max.png differ
diff --git a/Phranque/awesome/themes/holo/icons/mpd.png b/Phranque/awesome/themes/holo/icons/mpd.png
new file mode 100644
index 00000000..efa8ec39
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/mpd.png differ
diff --git a/Phranque/awesome/themes/holo/icons/net_down.png b/Phranque/awesome/themes/holo/icons/net_down.png
new file mode 100644
index 00000000..d5075a1a
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/net_down.png differ
diff --git a/Phranque/awesome/themes/holo/icons/net_up.png b/Phranque/awesome/themes/holo/icons/net_up.png
new file mode 100644
index 00000000..66dfb540
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/net_up.png differ
diff --git a/Phranque/awesome/themes/holo/icons/next.png b/Phranque/awesome/themes/holo/icons/next.png
new file mode 100644
index 00000000..148ce907
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/next.png differ
diff --git a/Phranque/awesome/themes/holo/icons/pause.png b/Phranque/awesome/themes/holo/icons/pause.png
new file mode 100644
index 00000000..9e9bc3f2
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/pause.png differ
diff --git a/Phranque/awesome/themes/holo/icons/play.png b/Phranque/awesome/themes/holo/icons/play.png
new file mode 100644
index 00000000..19499830
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/play.png differ
diff --git a/Phranque/awesome/themes/holo/icons/prev.png b/Phranque/awesome/themes/holo/icons/prev.png
new file mode 100644
index 00000000..d913bbad
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/prev.png differ
diff --git a/Phranque/awesome/themes/holo/icons/spiral.png b/Phranque/awesome/themes/holo/icons/spiral.png
new file mode 100644
index 00000000..ebb7bb30
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/spiral.png differ
diff --git a/Phranque/awesome/themes/holo/icons/spr_bottom_right.png b/Phranque/awesome/themes/holo/icons/spr_bottom_right.png
new file mode 100644
index 00000000..d702f989
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/spr_bottom_right.png differ
diff --git a/Phranque/awesome/themes/holo/icons/spr_left.png b/Phranque/awesome/themes/holo/icons/spr_left.png
new file mode 100644
index 00000000..44822ed9
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/spr_left.png differ
diff --git a/Phranque/awesome/themes/holo/icons/spr_right.png b/Phranque/awesome/themes/holo/icons/spr_right.png
new file mode 100644
index 00000000..5cca7958
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/spr_right.png differ
diff --git a/Phranque/awesome/themes/holo/icons/spr_small.png b/Phranque/awesome/themes/holo/icons/spr_small.png
new file mode 100644
index 00000000..16492a8d
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/spr_small.png differ
diff --git a/Phranque/awesome/themes/holo/icons/spr_very_small.png b/Phranque/awesome/themes/holo/icons/spr_very_small.png
new file mode 100644
index 00000000..8b9a2275
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/spr_very_small.png differ
diff --git a/Phranque/awesome/themes/holo/icons/square_sel.png b/Phranque/awesome/themes/holo/icons/square_sel.png
new file mode 100644
index 00000000..73fb845a
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/square_sel.png differ
diff --git a/Phranque/awesome/themes/holo/icons/square_unsel.png b/Phranque/awesome/themes/holo/icons/square_unsel.png
new file mode 100644
index 00000000..07dd9f04
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/square_unsel.png differ
diff --git a/Phranque/awesome/themes/holo/icons/stop.png b/Phranque/awesome/themes/holo/icons/stop.png
new file mode 100644
index 00000000..bfc819c3
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/stop.png differ
diff --git a/Phranque/awesome/themes/holo/icons/tile.png b/Phranque/awesome/themes/holo/icons/tile.png
new file mode 100644
index 00000000..e2e52bba
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/tile.png differ
diff --git a/Phranque/awesome/themes/holo/icons/tilebottom.png b/Phranque/awesome/themes/holo/icons/tilebottom.png
new file mode 100644
index 00000000..439f528d
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/tilebottom.png differ
diff --git a/Phranque/awesome/themes/holo/icons/tileleft.png b/Phranque/awesome/themes/holo/icons/tileleft.png
new file mode 100644
index 00000000..827e0d17
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/tileleft.png differ
diff --git a/Phranque/awesome/themes/holo/icons/tiletop.png b/Phranque/awesome/themes/holo/icons/tiletop.png
new file mode 100644
index 00000000..7139846a
Binary files /dev/null and b/Phranque/awesome/themes/holo/icons/tiletop.png differ
diff --git a/Phranque/awesome/themes/holo/theme.lua b/Phranque/awesome/themes/holo/theme.lua
new file mode 100644
index 00000000..e93c33d5
--- /dev/null
+++ b/Phranque/awesome/themes/holo/theme.lua
@@ -0,0 +1,389 @@
+--[[
+
+ Holo Awesome WM theme 3.0
+ github.com/lcpz
+
+--]]
+
+local gears = require("gears")
+local lain = require("lain")
+local awful = require("awful")
+local wibox = require("wibox")
+
+local string, os = string, os
+local my_table = awful.util.table or gears.table -- 4.{0,1} compatibility
+
+local theme = {}
+theme.default_dir = require("awful.util").get_themes_dir() .. "default"
+theme.icon_dir = os.getenv("HOME") .. "/.config/awesome/themes/holo/icons"
+theme.wallpaper = os.getenv("HOME") .. "/.config/awesome/themes/holo/wall.png"
+theme.font = "Roboto Bold 10"
+theme.taglist_font = "Roboto Condensed Regular 8"
+theme.fg_normal = "#FFFFFF"
+theme.fg_focus = "#0099CC"
+theme.bg_focus = "#303030"
+theme.bg_normal = "#242424"
+theme.fg_urgent = "#CC9393"
+theme.bg_urgent = "#006B8E"
+theme.border_width = 3
+theme.border_normal = "#252525"
+theme.border_focus = "#0099CC"
+theme.taglist_fg_focus = "#FFFFFF"
+theme.tasklist_bg_normal = "#222222"
+theme.tasklist_fg_focus = "#4CB7DB"
+theme.menu_height = 20
+theme.menu_width = 160
+theme.menu_icon_size = 32
+theme.awesome_icon = theme.icon_dir .. "/awesome_icon_white.png"
+theme.awesome_icon_launcher = theme.icon_dir .. "/awesome_icon.png"
+theme.taglist_squares_sel = theme.icon_dir .. "/square_sel.png"
+theme.taglist_squares_unsel = theme.icon_dir .. "/square_unsel.png"
+theme.spr_small = theme.icon_dir .. "/spr_small.png"
+theme.spr_very_small = theme.icon_dir .. "/spr_very_small.png"
+theme.spr_right = theme.icon_dir .. "/spr_right.png"
+theme.spr_bottom_right = theme.icon_dir .. "/spr_bottom_right.png"
+theme.spr_left = theme.icon_dir .. "/spr_left.png"
+theme.bar = theme.icon_dir .. "/bar.png"
+theme.bottom_bar = theme.icon_dir .. "/bottom_bar.png"
+theme.mpdl = theme.icon_dir .. "/mpd.png"
+theme.mpd_on = theme.icon_dir .. "/mpd_on.png"
+theme.prev = theme.icon_dir .. "/prev.png"
+theme.nex = theme.icon_dir .. "/next.png"
+theme.stop = theme.icon_dir .. "/stop.png"
+theme.pause = theme.icon_dir .. "/pause.png"
+theme.play = theme.icon_dir .. "/play.png"
+theme.clock = theme.icon_dir .. "/clock.png"
+theme.calendar = theme.icon_dir .. "/cal.png"
+theme.cpu = theme.icon_dir .. "/cpu.png"
+theme.net_up = theme.icon_dir .. "/net_up.png"
+theme.net_down = theme.icon_dir .. "/net_down.png"
+theme.layout_tile = theme.icon_dir .. "/tile.png"
+theme.layout_tileleft = theme.icon_dir .. "/tileleft.png"
+theme.layout_tilebottom = theme.icon_dir .. "/tilebottom.png"
+theme.layout_tiletop = theme.icon_dir .. "/tiletop.png"
+theme.layout_fairv = theme.icon_dir .. "/fairv.png"
+theme.layout_fairh = theme.icon_dir .. "/fairh.png"
+theme.layout_spiral = theme.icon_dir .. "/spiral.png"
+theme.layout_dwindle = theme.icon_dir .. "/dwindle.png"
+theme.layout_max = theme.icon_dir .. "/max.png"
+theme.layout_fullscreen = theme.icon_dir .. "/fullscreen.png"
+theme.layout_magnifier = theme.icon_dir .. "/magnifier.png"
+theme.layout_floating = theme.icon_dir .. "/floating.png"
+theme.tasklist_plain_task_name = true
+theme.tasklist_disable_icon = true
+theme.useless_gap = 4
+theme.titlebar_close_button_normal = theme.default_dir.."/titlebar/close_normal.png"
+theme.titlebar_close_button_focus = theme.default_dir.."/titlebar/close_focus.png"
+theme.titlebar_minimize_button_normal = theme.default_dir.."/titlebar/minimize_normal.png"
+theme.titlebar_minimize_button_focus = theme.default_dir.."/titlebar/minimize_focus.png"
+theme.titlebar_ontop_button_normal_inactive = theme.default_dir.."/titlebar/ontop_normal_inactive.png"
+theme.titlebar_ontop_button_focus_inactive = theme.default_dir.."/titlebar/ontop_focus_inactive.png"
+theme.titlebar_ontop_button_normal_active = theme.default_dir.."/titlebar/ontop_normal_active.png"
+theme.titlebar_ontop_button_focus_active = theme.default_dir.."/titlebar/ontop_focus_active.png"
+theme.titlebar_sticky_button_normal_inactive = theme.default_dir.."/titlebar/sticky_normal_inactive.png"
+theme.titlebar_sticky_button_focus_inactive = theme.default_dir.."/titlebar/sticky_focus_inactive.png"
+theme.titlebar_sticky_button_normal_active = theme.default_dir.."/titlebar/sticky_normal_active.png"
+theme.titlebar_sticky_button_focus_active = theme.default_dir.."/titlebar/sticky_focus_active.png"
+theme.titlebar_floating_button_normal_inactive = theme.default_dir.."/titlebar/floating_normal_inactive.png"
+theme.titlebar_floating_button_focus_inactive = theme.default_dir.."/titlebar/floating_focus_inactive.png"
+theme.titlebar_floating_button_normal_active = theme.default_dir.."/titlebar/floating_normal_active.png"
+theme.titlebar_floating_button_focus_active = theme.default_dir.."/titlebar/floating_focus_active.png"
+theme.titlebar_maximized_button_normal_inactive = theme.default_dir.."/titlebar/maximized_normal_inactive.png"
+theme.titlebar_maximized_button_focus_inactive = theme.default_dir.."/titlebar/maximized_focus_inactive.png"
+theme.titlebar_maximized_button_normal_active = theme.default_dir.."/titlebar/maximized_normal_active.png"
+theme.titlebar_maximized_button_focus_active = theme.default_dir.."/titlebar/maximized_focus_active.png"
+
+theme.musicplr = string.format("%s -e ncmpcpp", awful.util.terminal)
+
+local markup = lain.util.markup
+local blue = "#80CCE6"
+local space3 = markup.font("Roboto 3", " ")
+
+-- Clock
+local mytextclock = wibox.widget.textclock(markup("#FFFFFF", space3 .. "%H:%M " .. markup.font("Roboto 4", " ")))
+mytextclock.font = theme.font
+local clock_icon = wibox.widget.imagebox(theme.clock)
+local clockbg = wibox.container.background(mytextclock, theme.bg_focus, gears.shape.rectangle)
+local clockwidget = wibox.container.margin(clockbg, 0, 3, 5, 5)
+
+-- Calendar
+local mytextcalendar = wibox.widget.textclock(markup.fontfg(theme.font, "#FFFFFF", space3 .. "%d %b " .. markup.font("Roboto 5", " ")))
+local calendar_icon = wibox.widget.imagebox(theme.calendar)
+local calbg = wibox.container.background(mytextcalendar, theme.bg_focus, gears.shape.rectangle)
+local calendarwidget = wibox.container.margin(calbg, 0, 0, 5, 5)
+theme.cal = lain.widget.cal({
+ attach_to = { mytextclock, mytextcalendar },
+ notification_preset = {
+ fg = "#FFFFFF",
+ bg = theme.bg_normal,
+ position = "bottom_right",
+ font = "Monospace 10"
+ }
+})
+
+-- Mail IMAP check
+--[[ commented because it needs to be set before use
+theme.mail = lain.widget.imap({
+ timeout = 180,
+ server = "server",
+ mail = "mail",
+ password = "keyring get mail",
+ settings = function()
+ mail_notification_preset.fg = "#FFFFFF"
+ mail = ""
+ count = ""
+
+ if mailcount > 0 then
+ mail = "Mail "
+ count = mailcount .. " "
+ end
+
+ widget:set_markup(markup.font(theme.font, markup(blue, mail) .. markup("#FFFFFF", count)))
+ end
+})
+--]]
+
+-- MPD
+local mpd_icon = awful.widget.launcher({ image = theme.mpdl, command = theme.musicplr })
+local prev_icon = wibox.widget.imagebox(theme.prev)
+local next_icon = wibox.widget.imagebox(theme.nex)
+local stop_icon = wibox.widget.imagebox(theme.stop)
+local pause_icon = wibox.widget.imagebox(theme.pause)
+local play_pause_icon = wibox.widget.imagebox(theme.play)
+theme.mpd = lain.widget.mpd({
+ settings = function ()
+ if mpd_now.state == "play" then
+ mpd_now.artist = mpd_now.artist:upper():gsub("&.-;", string.lower)
+ mpd_now.title = mpd_now.title:upper():gsub("&.-;", string.lower)
+ widget:set_markup(markup.font("Roboto 4", " ")
+ .. markup.font(theme.taglist_font,
+ " " .. mpd_now.artist
+ .. " - " ..
+ mpd_now.title .. " ") .. markup.font("Roboto 5", " "))
+ play_pause_icon:set_image(theme.pause)
+ elseif mpd_now.state == "pause" then
+ widget:set_markup(markup.font("Roboto 4", " ") ..
+ markup.font(theme.taglist_font, " MPD PAUSED ") ..
+ markup.font("Roboto 5", " "))
+ play_pause_icon:set_image(theme.play)
+ else
+ widget:set_markup("")
+ play_pause_icon:set_image(theme.play)
+ end
+ end
+})
+local musicbg = wibox.container.background(theme.mpd.widget, theme.bg_focus, gears.shape.rectangle)
+local musicwidget = wibox.container.margin(musicbg, 0, 0, 5, 5)
+
+musicwidget:buttons(my_table.join(awful.button({ }, 1,
+function () awful.spawn(theme.musicplr) end)))
+prev_icon:buttons(my_table.join(awful.button({}, 1,
+function ()
+ os.execute("mpc prev")
+ theme.mpd.update()
+end)))
+next_icon:buttons(my_table.join(awful.button({}, 1,
+function ()
+ os.execute("mpc next")
+ theme.mpd.update()
+end)))
+stop_icon:buttons(my_table.join(awful.button({}, 1,
+function ()
+ play_pause_icon:set_image(theme.play)
+ os.execute("mpc stop")
+ theme.mpd.update()
+end)))
+play_pause_icon:buttons(my_table.join(awful.button({}, 1,
+function ()
+ os.execute("mpc toggle")
+ theme.mpd.update()
+end)))
+
+-- Battery
+local bat = lain.widget.bat({
+ settings = function()
+ bat_header = " Bat "
+ bat_p = bat_now.perc .. " "
+ if bat_now.ac_status == 1 then
+ bat_p = bat_p .. "Plugged "
+ end
+ widget:set_markup(markup.font(theme.font, markup(blue, bat_header) .. bat_p))
+ end
+})
+
+-- / fs
+--[[ commented because it needs Gio/Glib >= 2.54
+theme.fs = lain.widget.fs({
+ notification_preset = { bg = theme.bg_normal, font = "Monospace 9" },
+})
+--]]
+
+-- ALSA volume bar
+theme.volume = lain.widget.alsabar({
+ notification_preset = { font = "Monospace 9"},
+ --togglechannel = "IEC958,3",
+ width = 80, height = 10, border_width = 0,
+ colors = {
+ background = "#383838",
+ unmute = "#80CCE6",
+ mute = "#FF9F9F"
+ },
+})
+theme.volume.bar.paddings = 0
+theme.volume.bar.margins = 5
+local volumewidget = wibox.container.background(theme.volume.bar, theme.bg_focus, gears.shape.rectangle)
+volumewidget = wibox.container.margin(volumewidget, 0, 0, 5, 5)
+
+-- CPU
+local cpu_icon = wibox.widget.imagebox(theme.cpu)
+local cpu = lain.widget.cpu({
+ settings = function()
+ widget:set_markup(space3 .. markup.font(theme.font, "CPU " .. cpu_now.usage
+ .. "% ") .. markup.font("Roboto 5", " "))
+ end
+})
+local cpubg = wibox.container.background(cpu.widget, theme.bg_focus, gears.shape.rectangle)
+local cpuwidget = wibox.container.margin(cpubg, 0, 0, 5, 5)
+
+-- Net
+local netdown_icon = wibox.widget.imagebox(theme.net_down)
+local netup_icon = wibox.widget.imagebox(theme.net_up)
+local net = lain.widget.net({
+ settings = function()
+ widget:set_markup(markup.font("Roboto 1", " ") .. markup.font(theme.font, net_now.received .. " - "
+ .. net_now.sent) .. markup.font("Roboto 2", " "))
+ end
+})
+local netbg = wibox.container.background(net.widget, theme.bg_focus, gears.shape.rectangle)
+local networkwidget = wibox.container.margin(netbg, 0, 0, 5, 5)
+
+-- Weather
+theme.weather = lain.widget.weather({
+ city_id = 2643743, -- placeholder (London)
+ notification_preset = { font = "Monospace 9", position = "bottom_right" },
+})
+
+-- Launcher
+local mylauncher = awful.widget.button({ image = theme.awesome_icon_launcher })
+mylauncher:connect_signal("button::press", function() awful.util.mymainmenu:toggle() end)
+
+-- Separators
+local first = wibox.widget.textbox(' ')
+local spr_small = wibox.widget.imagebox(theme.spr_small)
+local spr_very_small = wibox.widget.imagebox(theme.spr_very_small)
+local spr_right = wibox.widget.imagebox(theme.spr_right)
+local spr_bottom_right = wibox.widget.imagebox(theme.spr_bottom_right)
+local spr_left = wibox.widget.imagebox(theme.spr_left)
+local bar = wibox.widget.imagebox(theme.bar)
+local bottom_bar = wibox.widget.imagebox(theme.bottom_bar)
+
+local barcolor = gears.color({
+ type = "linear",
+ from = { 32, 0 },
+ to = { 32, 32 },
+ stops = { {0, theme.bg_focus}, {0.25, "#505050"}, {1, theme.bg_focus} }
+})
+
+function theme.at_screen_connect(s)
+ -- Quake application
+ s.quake = lain.util.quake({ app = awful.util.terminal })
+
+ -- If wallpaper is a function, call it with the screen
+ local wallpaper = theme.wallpaper
+ if type(wallpaper) == "function" then
+ wallpaper = wallpaper(s)
+ end
+ gears.wallpaper.maximized(wallpaper, s, true)
+
+ -- Tags
+ awful.tag(awful.util.tagnames, s, awful.layout.layouts)
+
+ -- Create a promptbox for each screen
+ s.mypromptbox = awful.widget.prompt()
+ -- Create an imagebox widget which will contains an icon indicating which layout we're using.
+ -- We need one layoutbox per screen.
+ s.mylayoutbox = awful.widget.layoutbox(s)
+ s.mylayoutbox:buttons(my_table.join(
+ awful.button({}, 1, function () awful.layout.inc( 1) end),
+ awful.button({}, 2, function () awful.layout.set( awful.layout.layouts[1] ) end),
+ awful.button({}, 3, function () awful.layout.inc(-1) end),
+ awful.button({}, 4, function () awful.layout.inc( 1) end),
+ awful.button({}, 5, function () awful.layout.inc(-1) end)))
+ -- Create a taglist widget
+ s.mytaglist = awful.widget.taglist(s, awful.widget.taglist.filter.all, awful.util.taglist_buttons, { bg_focus = barcolor })
+
+ mytaglistcont = wibox.container.background(s.mytaglist, theme.bg_focus, gears.shape.rectangle)
+ s.mytag = wibox.container.margin(mytaglistcont, 0, 0, 5, 5)
+
+ -- Create a tasklist widget
+ s.mytasklist = awful.widget.tasklist(s, awful.widget.tasklist.filter.currenttags, awful.util.tasklist_buttons, { bg_focus = theme.bg_focus, shape = gears.shape.rectangle, shape_border_width = 5, shape_border_color = theme.tasklist_bg_normal, align = "center" })
+
+ -- Create the wibox
+ s.mywibox = awful.wibar({ position = "top", screen = s, height = 32 })
+
+ -- Add widgets to the wibox
+ s.mywibox:setup {
+ layout = wibox.layout.align.horizontal,
+ { -- Left widgets
+ layout = wibox.layout.fixed.horizontal,
+ first,
+ s.mytag,
+ spr_small,
+ s.mylayoutbox,
+ spr_small,
+ s.mypromptbox,
+ },
+ nil, -- Middle widget
+ { -- Right widgets
+ layout = wibox.layout.fixed.horizontal,
+ wibox.widget.systray(),
+ --theme.mail.widget,
+ --bat.widget,
+ spr_right,
+ musicwidget,
+ bar,
+ prev_icon,
+ next_icon,
+ stop_icon,
+ play_pause_icon,
+ bar,
+ mpd_icon,
+ bar,
+ spr_very_small,
+ volumewidget,
+ spr_left,
+ },
+ }
+
+ -- Create the bottom wibox
+ s.mybottomwibox = awful.wibar({ position = "bottom", screen = s, border_width = 0, height = 32 })
+ s.borderwibox = awful.wibar({ position = "bottom", screen = s, height = 1, bg = theme.fg_focus, x = 0, y = 33})
+
+ -- Add widgets to the bottom wibox
+ s.mybottomwibox:setup {
+ layout = wibox.layout.align.horizontal,
+ { -- Left widgets
+ layout = wibox.layout.fixed.horizontal,
+ mylauncher,
+ },
+ s.mytasklist, -- Middle widget
+ { -- Right widgets
+ layout = wibox.layout.fixed.horizontal,
+ spr_bottom_right,
+ netdown_icon,
+ networkwidget,
+ netup_icon,
+ bottom_bar,
+ cpu_icon,
+ cpuwidget,
+ bottom_bar,
+ calendar_icon,
+ calendarwidget,
+ bottom_bar,
+ clock_icon,
+ clockwidget,
+ },
+ }
+end
+
+return theme
diff --git a/Phranque/awesome/themes/holo/wall.png b/Phranque/awesome/themes/holo/wall.png
new file mode 100644
index 00000000..7ec80096
Binary files /dev/null and b/Phranque/awesome/themes/holo/wall.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/ac.png b/Phranque/awesome/themes/multicolor/icons/ac.png
new file mode 100644
index 00000000..272b8327
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/ac.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/bat.png b/Phranque/awesome/themes/multicolor/icons/bat.png
new file mode 100644
index 00000000..bd639279
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/bat.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/clock.png b/Phranque/awesome/themes/multicolor/icons/clock.png
new file mode 100644
index 00000000..22aef71d
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/clock.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/cpu.png b/Phranque/awesome/themes/multicolor/icons/cpu.png
new file mode 100644
index 00000000..e3794ca3
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/cpu.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/dish.png b/Phranque/awesome/themes/multicolor/icons/dish.png
new file mode 100644
index 00000000..c71499be
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/dish.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/dwindle.png b/Phranque/awesome/themes/multicolor/icons/dwindle.png
new file mode 100644
index 00000000..94d8382b
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/dwindle.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/fairh.png b/Phranque/awesome/themes/multicolor/icons/fairh.png
new file mode 100644
index 00000000..4468efcd
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/fairh.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/fairv.png b/Phranque/awesome/themes/multicolor/icons/fairv.png
new file mode 100644
index 00000000..7c8728e8
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/fairv.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/floating.png b/Phranque/awesome/themes/multicolor/icons/floating.png
new file mode 100644
index 00000000..7593a2ab
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/floating.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/fs.png b/Phranque/awesome/themes/multicolor/icons/fs.png
new file mode 100644
index 00000000..1c1b13e1
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/fs.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/fullscreen.png b/Phranque/awesome/themes/multicolor/icons/fullscreen.png
new file mode 100644
index 00000000..f63a69c5
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/fullscreen.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/magnifier.png b/Phranque/awesome/themes/multicolor/icons/magnifier.png
new file mode 100644
index 00000000..88195547
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/magnifier.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/mail.png b/Phranque/awesome/themes/multicolor/icons/mail.png
new file mode 100644
index 00000000..25021f5a
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/mail.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/max.png b/Phranque/awesome/themes/multicolor/icons/max.png
new file mode 100644
index 00000000..4b6a8c3d
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/max.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/mem.png b/Phranque/awesome/themes/multicolor/icons/mem.png
new file mode 100644
index 00000000..d04ee11f
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/mem.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/net_down.png b/Phranque/awesome/themes/multicolor/icons/net_down.png
new file mode 100644
index 00000000..09433b1f
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/net_down.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/net_up.png b/Phranque/awesome/themes/multicolor/icons/net_up.png
new file mode 100644
index 00000000..395d0146
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/net_up.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/note.png b/Phranque/awesome/themes/multicolor/icons/note.png
new file mode 100644
index 00000000..2f41eb3e
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/note.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/note_on.png b/Phranque/awesome/themes/multicolor/icons/note_on.png
new file mode 100755
index 00000000..264d2c8c
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/note_on.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/spiral.png b/Phranque/awesome/themes/multicolor/icons/spiral.png
new file mode 100644
index 00000000..6a1ebf17
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/spiral.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/spkr.png b/Phranque/awesome/themes/multicolor/icons/spkr.png
new file mode 100644
index 00000000..80c20b0b
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/spkr.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/square_a.png b/Phranque/awesome/themes/multicolor/icons/square_a.png
new file mode 100755
index 00000000..1774d955
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/square_a.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/square_b.png b/Phranque/awesome/themes/multicolor/icons/square_b.png
new file mode 100755
index 00000000..b47e9f39
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/square_b.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/submenu.png b/Phranque/awesome/themes/multicolor/icons/submenu.png
new file mode 100644
index 00000000..7c81cab5
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/submenu.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/temp.png b/Phranque/awesome/themes/multicolor/icons/temp.png
new file mode 100755
index 00000000..6c8f8bdc
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/temp.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/tile.png b/Phranque/awesome/themes/multicolor/icons/tile.png
new file mode 100644
index 00000000..64103d9f
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/tile.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/tilebottom.png b/Phranque/awesome/themes/multicolor/icons/tilebottom.png
new file mode 100644
index 00000000..93636366
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/tilebottom.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/tileleft.png b/Phranque/awesome/themes/multicolor/icons/tileleft.png
new file mode 100644
index 00000000..9331898a
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/tileleft.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/tiletop.png b/Phranque/awesome/themes/multicolor/icons/tiletop.png
new file mode 100644
index 00000000..1cef886b
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/tiletop.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/close_focus.png b/Phranque/awesome/themes/multicolor/icons/titlebar/close_focus.png
new file mode 100644
index 00000000..cd297dbd
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/close_focus.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/close_normal.png b/Phranque/awesome/themes/multicolor/icons/titlebar/close_normal.png
new file mode 100644
index 00000000..5448ed8f
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/close_normal.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/floating_focus_active.png b/Phranque/awesome/themes/multicolor/icons/titlebar/floating_focus_active.png
new file mode 100644
index 00000000..ae5a8e23
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/floating_focus_active.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/floating_focus_inactive.png b/Phranque/awesome/themes/multicolor/icons/titlebar/floating_focus_inactive.png
new file mode 100644
index 00000000..c54eb0cf
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/floating_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/floating_normal_active.png b/Phranque/awesome/themes/multicolor/icons/titlebar/floating_normal_active.png
new file mode 100644
index 00000000..62342d19
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/floating_normal_active.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/floating_normal_inactive.png b/Phranque/awesome/themes/multicolor/icons/titlebar/floating_normal_inactive.png
new file mode 100644
index 00000000..e2bbdfa1
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/floating_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/maximized_focus_active.png b/Phranque/awesome/themes/multicolor/icons/titlebar/maximized_focus_active.png
new file mode 100644
index 00000000..957298bc
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/maximized_focus_active.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/maximized_focus_inactive.png b/Phranque/awesome/themes/multicolor/icons/titlebar/maximized_focus_inactive.png
new file mode 100644
index 00000000..11872293
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/maximized_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/maximized_normal_active.png b/Phranque/awesome/themes/multicolor/icons/titlebar/maximized_normal_active.png
new file mode 100644
index 00000000..a705f816
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/maximized_normal_active.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/maximized_normal_inactive.png b/Phranque/awesome/themes/multicolor/icons/titlebar/maximized_normal_inactive.png
new file mode 100644
index 00000000..4c1ab1f9
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/maximized_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/minimize_focus.png b/Phranque/awesome/themes/multicolor/icons/titlebar/minimize_focus.png
new file mode 100644
index 00000000..bbe8cff2
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/minimize_focus.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/minimize_normal.png b/Phranque/awesome/themes/multicolor/icons/titlebar/minimize_normal.png
new file mode 100644
index 00000000..36621d0f
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/minimize_normal.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/ontop_focus_active.png b/Phranque/awesome/themes/multicolor/icons/titlebar/ontop_focus_active.png
new file mode 100644
index 00000000..25a30804
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/ontop_focus_active.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/ontop_focus_inactive.png b/Phranque/awesome/themes/multicolor/icons/titlebar/ontop_focus_inactive.png
new file mode 100644
index 00000000..d3c077ae
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/ontop_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/ontop_normal_active.png b/Phranque/awesome/themes/multicolor/icons/titlebar/ontop_normal_active.png
new file mode 100644
index 00000000..117a203c
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/ontop_normal_active.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/ontop_normal_inactive.png b/Phranque/awesome/themes/multicolor/icons/titlebar/ontop_normal_inactive.png
new file mode 100644
index 00000000..d3a10c8a
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/ontop_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/sticky_focus_active.png b/Phranque/awesome/themes/multicolor/icons/titlebar/sticky_focus_active.png
new file mode 100644
index 00000000..147ce6a2
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/sticky_focus_active.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/sticky_focus_inactive.png b/Phranque/awesome/themes/multicolor/icons/titlebar/sticky_focus_inactive.png
new file mode 100644
index 00000000..396628a2
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/sticky_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/sticky_normal_active.png b/Phranque/awesome/themes/multicolor/icons/titlebar/sticky_normal_active.png
new file mode 100644
index 00000000..bdb5595d
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/sticky_normal_active.png differ
diff --git a/Phranque/awesome/themes/multicolor/icons/titlebar/sticky_normal_inactive.png b/Phranque/awesome/themes/multicolor/icons/titlebar/sticky_normal_inactive.png
new file mode 100644
index 00000000..a96b9b19
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/icons/titlebar/sticky_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/multicolor/theme.lua b/Phranque/awesome/themes/multicolor/theme.lua
new file mode 100644
index 00000000..565b841c
--- /dev/null
+++ b/Phranque/awesome/themes/multicolor/theme.lua
@@ -0,0 +1,345 @@
+--[[
+
+ Multicolor Awesome WM theme 2.0
+ github.com/lcpz
+
+--]]
+
+local gears = require("gears")
+local lain = require("lain")
+local awful = require("awful")
+local wibox = require("wibox")
+
+local os = os
+local my_table = awful.util.table or gears.table -- 4.{0,1} compatibility
+
+local theme = {}
+theme.confdir = os.getenv("HOME") .. "/.config/awesome/themes/multicolor"
+theme.wallpaper = theme.confdir .. "/wall.png"
+theme.font = "xos4 Terminus 8"
+theme.menu_bg_normal = "#000000"
+theme.menu_bg_focus = "#000000"
+theme.bg_normal = "#000000"
+theme.bg_focus = "#000000"
+theme.bg_urgent = "#000000"
+theme.fg_normal = "#aaaaaa"
+theme.fg_focus = "#ff8c00"
+theme.fg_urgent = "#af1d18"
+theme.fg_minimize = "#ffffff"
+theme.border_width = 1
+theme.border_normal = "#1c2022"
+theme.border_focus = "#606060"
+theme.border_marked = "#3ca4d8"
+theme.menu_border_width = 0
+theme.menu_width = 130
+theme.menu_submenu_icon = theme.confdir .. "/icons/submenu.png"
+theme.menu_fg_normal = "#aaaaaa"
+theme.menu_fg_focus = "#ff8c00"
+theme.menu_bg_normal = "#050505dd"
+theme.menu_bg_focus = "#050505dd"
+theme.widget_temp = theme.confdir .. "/icons/temp.png"
+theme.widget_uptime = theme.confdir .. "/icons/ac.png"
+theme.widget_cpu = theme.confdir .. "/icons/cpu.png"
+theme.widget_weather = theme.confdir .. "/icons/dish.png"
+theme.widget_fs = theme.confdir .. "/icons/fs.png"
+theme.widget_mem = theme.confdir .. "/icons/mem.png"
+theme.widget_note = theme.confdir .. "/icons/note.png"
+theme.widget_note_on = theme.confdir .. "/icons/note_on.png"
+theme.widget_netdown = theme.confdir .. "/icons/net_down.png"
+theme.widget_netup = theme.confdir .. "/icons/net_up.png"
+theme.widget_mail = theme.confdir .. "/icons/mail.png"
+theme.widget_batt = theme.confdir .. "/icons/bat.png"
+theme.widget_clock = theme.confdir .. "/icons/clock.png"
+theme.widget_vol = theme.confdir .. "/icons/spkr.png"
+theme.taglist_squares_sel = theme.confdir .. "/icons/square_a.png"
+theme.taglist_squares_unsel = theme.confdir .. "/icons/square_b.png"
+theme.tasklist_plain_task_name = true
+theme.tasklist_disable_icon = true
+theme.useless_gap = 0
+theme.layout_tile = theme.confdir .. "/icons/tile.png"
+theme.layout_tilegaps = theme.confdir .. "/icons/tilegaps.png"
+theme.layout_tileleft = theme.confdir .. "/icons/tileleft.png"
+theme.layout_tilebottom = theme.confdir .. "/icons/tilebottom.png"
+theme.layout_tiletop = theme.confdir .. "/icons/tiletop.png"
+theme.layout_fairv = theme.confdir .. "/icons/fairv.png"
+theme.layout_fairh = theme.confdir .. "/icons/fairh.png"
+theme.layout_spiral = theme.confdir .. "/icons/spiral.png"
+theme.layout_dwindle = theme.confdir .. "/icons/dwindle.png"
+theme.layout_max = theme.confdir .. "/icons/max.png"
+theme.layout_fullscreen = theme.confdir .. "/icons/fullscreen.png"
+theme.layout_magnifier = theme.confdir .. "/icons/magnifier.png"
+theme.layout_floating = theme.confdir .. "/icons/floating.png"
+theme.titlebar_close_button_normal = theme.confdir .. "/icons/titlebar/close_normal.png"
+theme.titlebar_close_button_focus = theme.confdir .. "/icons/titlebar/close_focus.png"
+theme.titlebar_minimize_button_normal = theme.confdir .. "/icons/titlebar/minimize_normal.png"
+theme.titlebar_minimize_button_focus = theme.confdir .. "/icons/titlebar/minimize_focus.png"
+theme.titlebar_ontop_button_normal_inactive = theme.confdir .. "/icons/titlebar/ontop_normal_inactive.png"
+theme.titlebar_ontop_button_focus_inactive = theme.confdir .. "/icons/titlebar/ontop_focus_inactive.png"
+theme.titlebar_ontop_button_normal_active = theme.confdir .. "/icons/titlebar/ontop_normal_active.png"
+theme.titlebar_ontop_button_focus_active = theme.confdir .. "/icons/titlebar/ontop_focus_active.png"
+theme.titlebar_sticky_button_normal_inactive = theme.confdir .. "/icons/titlebar/sticky_normal_inactive.png"
+theme.titlebar_sticky_button_focus_inactive = theme.confdir .. "/icons/titlebar/sticky_focus_inactive.png"
+theme.titlebar_sticky_button_normal_active = theme.confdir .. "/icons/titlebar/sticky_normal_active.png"
+theme.titlebar_sticky_button_focus_active = theme.confdir .. "/icons/titlebar/sticky_focus_active.png"
+theme.titlebar_floating_button_normal_inactive = theme.confdir .. "/icons/titlebar/floating_normal_inactive.png"
+theme.titlebar_floating_button_focus_inactive = theme.confdir .. "/icons/titlebar/floating_focus_inactive.png"
+theme.titlebar_floating_button_normal_active = theme.confdir .. "/icons/titlebar/floating_normal_active.png"
+theme.titlebar_floating_button_focus_active = theme.confdir .. "/icons/titlebar/floating_focus_active.png"
+theme.titlebar_maximized_button_normal_inactive = theme.confdir .. "/icons/titlebar/maximized_normal_inactive.png"
+theme.titlebar_maximized_button_focus_inactive = theme.confdir .. "/icons/titlebar/maximized_focus_inactive.png"
+theme.titlebar_maximized_button_normal_active = theme.confdir .. "/icons/titlebar/maximized_normal_active.png"
+theme.titlebar_maximized_button_focus_active = theme.confdir .. "/icons/titlebar/maximized_focus_active.png"
+
+local markup = lain.util.markup
+
+-- Textclock
+os.setlocale(os.getenv("LANG")) -- to localize the clock
+local clockicon = wibox.widget.imagebox(theme.widget_clock)
+local mytextclock = wibox.widget.textclock(markup("#7788af", "%A %d %B ") .. markup("#ab7367", ">") .. markup("#de5e1e", " %H:%M "))
+mytextclock.font = theme.font
+
+-- Calendar
+theme.cal = lain.widget.cal({
+ attach_to = { mytextclock },
+ notification_preset = {
+ font = "xos4 Terminus 10",
+ fg = theme.fg_normal,
+ bg = theme.bg_normal
+ }
+})
+
+-- Weather
+local weathericon = wibox.widget.imagebox(theme.widget_weather)
+theme.weather = lain.widget.weather({
+ city_id = 2643743, -- placeholder (London)
+ notification_preset = { font = "xos4 Terminus 10", fg = theme.fg_normal },
+ weather_na_markup = markup.fontfg(theme.font, "#eca4c4", "N/A "),
+ settings = function()
+ descr = weather_now["weather"][1]["description"]:lower()
+ units = math.floor(weather_now["main"]["temp"])
+ widget:set_markup(markup.fontfg(theme.font, "#eca4c4", descr .. " @ " .. units .. "°C "))
+ end
+})
+
+-- / fs
+--[[ commented because it needs Gio/Glib >= 2.54
+local fsicon = wibox.widget.imagebox(theme.widget_fs)
+theme.fs = lain.widget.fs({
+ notification_preset = { font = "xos4 Terminus 10", fg = theme.fg_normal },
+ settings = function()
+ widget:set_markup(markup.fontfg(theme.font, "#80d9d8", string.format("%.1f", fs_now["/"].used) .. "% "))
+ end
+})
+--]]
+
+-- Mail IMAP check
+--[[ commented because it needs to be set before use
+local mailicon = wibox.widget.imagebox()
+theme.mail = lain.widget.imap({
+ timeout = 180,
+ server = "server",
+ mail = "mail",
+ password = "keyring get mail",
+ settings = function()
+ if mailcount > 0 then
+ mailicon:set_image(theme.widget_mail)
+ widget:set_markup(markup.fontfg(theme.font, "#cccccc", mailcount .. " "))
+ else
+ widget:set_text("")
+ --mailicon:set_image() -- not working in 4.0
+ mailicon._private.image = nil
+ mailicon:emit_signal("widget::redraw_needed")
+ mailicon:emit_signal("widget::layout_changed")
+ end
+ end
+})
+--]]
+
+-- CPU
+local cpuicon = wibox.widget.imagebox(theme.widget_cpu)
+local cpu = lain.widget.cpu({
+ settings = function()
+ widget:set_markup(markup.fontfg(theme.font, "#e33a6e", cpu_now.usage .. "% "))
+ end
+})
+
+-- Coretemp
+local tempicon = wibox.widget.imagebox(theme.widget_temp)
+local temp = lain.widget.temp({
+ settings = function()
+ widget:set_markup(markup.fontfg(theme.font, "#f1af5f", coretemp_now .. "°C "))
+ end
+})
+
+-- Battery
+local baticon = wibox.widget.imagebox(theme.widget_batt)
+local bat = lain.widget.bat({
+ settings = function()
+ local perc = bat_now.perc ~= "N/A" and bat_now.perc .. "%" or bat_now.perc
+
+ if bat_now.ac_status == 1 then
+ perc = perc .. " plug"
+ end
+
+ widget:set_markup(markup.fontfg(theme.font, theme.fg_normal, perc .. " "))
+ end
+})
+
+-- ALSA volume
+local volicon = wibox.widget.imagebox(theme.widget_vol)
+theme.volume = lain.widget.alsa({
+ settings = function()
+ if volume_now.status == "off" then
+ volume_now.level = volume_now.level .. "M"
+ end
+
+ widget:set_markup(markup.fontfg(theme.font, "#7493d2", volume_now.level .. "% "))
+ end
+})
+
+-- Net
+local netdownicon = wibox.widget.imagebox(theme.widget_netdown)
+local netdowninfo = wibox.widget.textbox()
+local netupicon = wibox.widget.imagebox(theme.widget_netup)
+local netupinfo = lain.widget.net({
+ settings = function()
+ if iface ~= "network off" and
+ string.match(theme.weather.widget.text, "N/A")
+ then
+ theme.weather.update()
+ end
+
+ widget:set_markup(markup.fontfg(theme.font, "#e54c62", net_now.sent .. " "))
+ netdowninfo:set_markup(markup.fontfg(theme.font, "#87af5f", net_now.received .. " "))
+ end
+})
+
+-- MEM
+local memicon = wibox.widget.imagebox(theme.widget_mem)
+local memory = lain.widget.mem({
+ settings = function()
+ widget:set_markup(markup.fontfg(theme.font, "#e0da37", mem_now.used .. "M "))
+ end
+})
+
+-- MPD
+local mpdicon = wibox.widget.imagebox()
+theme.mpd = lain.widget.mpd({
+ settings = function()
+ mpd_notification_preset = {
+ text = string.format("%s [%s] - %s\n%s", mpd_now.artist,
+ mpd_now.album, mpd_now.date, mpd_now.title)
+ }
+
+ if mpd_now.state == "play" then
+ artist = mpd_now.artist .. " > "
+ title = mpd_now.title .. " "
+ mpdicon:set_image(theme.widget_note_on)
+ elseif mpd_now.state == "pause" then
+ artist = "mpd "
+ title = "paused "
+ else
+ artist = ""
+ title = ""
+ --mpdicon:set_image() -- not working in 4.0
+ mpdicon._private.image = nil
+ mpdicon:emit_signal("widget::redraw_needed")
+ mpdicon:emit_signal("widget::layout_changed")
+ end
+ widget:set_markup(markup.fontfg(theme.font, "#e54c62", artist) .. markup.fontfg(theme.font, "#b2b2b2", title))
+ end
+})
+
+function theme.at_screen_connect(s)
+ -- Quake application
+ s.quake = lain.util.quake({ app = awful.util.terminal })
+
+ -- If wallpaper is a function, call it with the screen
+ local wallpaper = theme.wallpaper
+ if type(wallpaper) == "function" then
+ wallpaper = wallpaper(s)
+ end
+ gears.wallpaper.maximized(wallpaper, s, true)
+
+ -- Tags
+ awful.tag(awful.util.tagnames, s, awful.layout.layouts)
+
+ -- Create a promptbox for each screen
+ s.mypromptbox = awful.widget.prompt()
+ -- Create an imagebox widget which will contains an icon indicating which layout we're using.
+ -- We need one layoutbox per screen.
+ s.mylayoutbox = awful.widget.layoutbox(s)
+ s.mylayoutbox:buttons(my_table.join(
+ awful.button({}, 1, function () awful.layout.inc( 1) end),
+ awful.button({}, 2, function () awful.layout.set( awful.layout.layouts[1] ) end),
+ awful.button({}, 3, function () awful.layout.inc(-1) end),
+ awful.button({}, 4, function () awful.layout.inc( 1) end),
+ awful.button({}, 5, function () awful.layout.inc(-1) end)))
+ -- Create a taglist widget
+ s.mytaglist = awful.widget.taglist(s, awful.widget.taglist.filter.all, awful.util.taglist_buttons)
+
+ -- Create a tasklist widget
+ s.mytasklist = awful.widget.tasklist(s, awful.widget.tasklist.filter.currenttags, awful.util.tasklist_buttons)
+
+ -- Create the wibox
+ s.mywibox = awful.wibar({ position = "top", screen = s, height = 20, bg = theme.bg_normal, fg = theme.fg_normal })
+
+ -- Add widgets to the wibox
+ s.mywibox:setup {
+ layout = wibox.layout.align.horizontal,
+ { -- Left widgets
+ layout = wibox.layout.fixed.horizontal,
+ --s.mylayoutbox,
+ s.mytaglist,
+ s.mypromptbox,
+ mpdicon,
+ theme.mpd.widget,
+ },
+ --s.mytasklist, -- Middle widget
+ nil,
+ { -- Right widgets
+ layout = wibox.layout.fixed.horizontal,
+ wibox.widget.systray(),
+ --mailicon,
+ --theme.mail.widget,
+ netdownicon,
+ netdowninfo,
+ netupicon,
+ netupinfo.widget,
+ volicon,
+ theme.volume.widget,
+ memicon,
+ memory.widget,
+ cpuicon,
+ cpu.widget,
+ --fsicon,
+ --theme.fs.widget,
+ weathericon,
+ theme.weather.widget,
+ tempicon,
+ temp.widget,
+ baticon,
+ bat.widget,
+ clockicon,
+ mytextclock,
+ },
+ }
+
+ -- Create the bottom wibox
+ s.mybottomwibox = awful.wibar({ position = "bottom", screen = s, border_width = 0, height = 20, bg = theme.bg_normal, fg = theme.fg_normal })
+
+ -- Add widgets to the bottom wibox
+ s.mybottomwibox:setup {
+ layout = wibox.layout.align.horizontal,
+ { -- Left widgets
+ layout = wibox.layout.fixed.horizontal,
+ },
+ s.mytasklist, -- Middle widget
+ { -- Right widgets
+ layout = wibox.layout.fixed.horizontal,
+ s.mylayoutbox,
+ },
+ }
+end
+
+return theme
diff --git a/Phranque/awesome/themes/multicolor/wall.png b/Phranque/awesome/themes/multicolor/wall.png
new file mode 100644
index 00000000..a3188d04
Binary files /dev/null and b/Phranque/awesome/themes/multicolor/wall.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/ac.png b/Phranque/awesome/themes/powerarrow-dark/icons/ac.png
new file mode 100644
index 00000000..96efcb4e
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/ac.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/battery.png b/Phranque/awesome/themes/powerarrow-dark/icons/battery.png
new file mode 100644
index 00000000..8c85596d
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/battery.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/battery_empty.png b/Phranque/awesome/themes/powerarrow-dark/icons/battery_empty.png
new file mode 100644
index 00000000..00821c91
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/battery_empty.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/battery_low.png b/Phranque/awesome/themes/powerarrow-dark/icons/battery_low.png
new file mode 100644
index 00000000..29f3fc58
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/battery_low.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/centerfair.png b/Phranque/awesome/themes/powerarrow-dark/icons/centerfair.png
new file mode 100644
index 00000000..c4f64b0f
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/centerfair.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/centerwork.png b/Phranque/awesome/themes/powerarrow-dark/icons/centerwork.png
new file mode 100644
index 00000000..90715169
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/centerwork.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/centerworkh.png b/Phranque/awesome/themes/powerarrow-dark/icons/centerworkh.png
new file mode 100644
index 00000000..b1d13793
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/centerworkh.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/cpu.png b/Phranque/awesome/themes/powerarrow-dark/icons/cpu.png
new file mode 100644
index 00000000..35af1c39
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/cpu.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/dwindle.png b/Phranque/awesome/themes/powerarrow-dark/icons/dwindle.png
new file mode 100644
index 00000000..649ea993
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/dwindle.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/fairh.png b/Phranque/awesome/themes/powerarrow-dark/icons/fairh.png
new file mode 100644
index 00000000..62d3d99e
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/fairh.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/fairv.png b/Phranque/awesome/themes/powerarrow-dark/icons/fairv.png
new file mode 100644
index 00000000..131dd0f9
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/fairv.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/floating.png b/Phranque/awesome/themes/powerarrow-dark/icons/floating.png
new file mode 100644
index 00000000..d25c47b9
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/floating.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/hdd.png b/Phranque/awesome/themes/powerarrow-dark/icons/hdd.png
new file mode 100644
index 00000000..0fb68339
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/hdd.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/magnifier.png b/Phranque/awesome/themes/powerarrow-dark/icons/magnifier.png
new file mode 100644
index 00000000..60d3e0d8
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/magnifier.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/mail.png b/Phranque/awesome/themes/powerarrow-dark/icons/mail.png
new file mode 100644
index 00000000..474e6020
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/mail.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/mail_on.png b/Phranque/awesome/themes/powerarrow-dark/icons/mail_on.png
new file mode 100644
index 00000000..19106d7b
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/mail_on.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/max.png b/Phranque/awesome/themes/powerarrow-dark/icons/max.png
new file mode 100644
index 00000000..5b0a5b71
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/max.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/mem.png b/Phranque/awesome/themes/powerarrow-dark/icons/mem.png
new file mode 100644
index 00000000..7860a1a0
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/mem.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/net.png b/Phranque/awesome/themes/powerarrow-dark/icons/net.png
new file mode 100644
index 00000000..bc42fdc3
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/net.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/net_wired.png b/Phranque/awesome/themes/powerarrow-dark/icons/net_wired.png
new file mode 100644
index 00000000..e8cc2bde
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/net_wired.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/note.png b/Phranque/awesome/themes/powerarrow-dark/icons/note.png
new file mode 100644
index 00000000..baa29c32
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/note.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/note_on.png b/Phranque/awesome/themes/powerarrow-dark/icons/note_on.png
new file mode 100644
index 00000000..1a7ab947
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/note_on.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/scissors.png b/Phranque/awesome/themes/powerarrow-dark/icons/scissors.png
new file mode 100644
index 00000000..f8c700de
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/scissors.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/spiral.png b/Phranque/awesome/themes/powerarrow-dark/icons/spiral.png
new file mode 100644
index 00000000..d9ee0f6b
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/spiral.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/square_sel.png b/Phranque/awesome/themes/powerarrow-dark/icons/square_sel.png
new file mode 100644
index 00000000..1102a9f1
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/square_sel.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/square_unsel.png b/Phranque/awesome/themes/powerarrow-dark/icons/square_unsel.png
new file mode 100644
index 00000000..7386b85f
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/square_unsel.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/submenu.png b/Phranque/awesome/themes/powerarrow-dark/icons/submenu.png
new file mode 100644
index 00000000..b55ebced
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/submenu.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/task.png b/Phranque/awesome/themes/powerarrow-dark/icons/task.png
new file mode 100644
index 00000000..9701b684
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/task.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/temp.png b/Phranque/awesome/themes/powerarrow-dark/icons/temp.png
new file mode 100644
index 00000000..6793a9fb
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/temp.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/termfair.png b/Phranque/awesome/themes/powerarrow-dark/icons/termfair.png
new file mode 100644
index 00000000..3e060236
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/termfair.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/tile.png b/Phranque/awesome/themes/powerarrow-dark/icons/tile.png
new file mode 100644
index 00000000..922c05ca
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/tile.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/tilebottom.png b/Phranque/awesome/themes/powerarrow-dark/icons/tilebottom.png
new file mode 100644
index 00000000..6ec9cb87
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/tilebottom.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/tileleft.png b/Phranque/awesome/themes/powerarrow-dark/icons/tileleft.png
new file mode 100644
index 00000000..8e3d2d62
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/tileleft.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/tiletop.png b/Phranque/awesome/themes/powerarrow-dark/icons/tiletop.png
new file mode 100644
index 00000000..3da75a19
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/tiletop.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/close_focus.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/close_focus.png
new file mode 100644
index 00000000..b2051b02
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/close_focus.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/close_normal.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/close_normal.png
new file mode 100644
index 00000000..da6028c1
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/close_normal.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/floating_focus_active.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/floating_focus_active.png
new file mode 100644
index 00000000..5fe84c0d
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/floating_focus_active.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/floating_focus_inactive.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/floating_focus_inactive.png
new file mode 100644
index 00000000..47f19f6d
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/floating_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/floating_normal_active.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/floating_normal_active.png
new file mode 100644
index 00000000..576fa367
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/floating_normal_active.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/floating_normal_inactive.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/floating_normal_inactive.png
new file mode 100644
index 00000000..4adc5e9d
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/floating_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/maximized_focus_active.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/maximized_focus_active.png
new file mode 100644
index 00000000..7d9a11a1
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/maximized_focus_active.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/maximized_focus_inactive.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/maximized_focus_inactive.png
new file mode 100644
index 00000000..bce1d000
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/maximized_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/maximized_normal_active.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/maximized_normal_active.png
new file mode 100644
index 00000000..9f24945b
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/maximized_normal_active.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/maximized_normal_inactive.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/maximized_normal_inactive.png
new file mode 100644
index 00000000..2e56d327
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/maximized_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/ontop_focus_active.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/ontop_focus_active.png
new file mode 100644
index 00000000..41a69e27
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/ontop_focus_active.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/ontop_focus_inactive.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/ontop_focus_inactive.png
new file mode 100644
index 00000000..2f3a2be2
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/ontop_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/ontop_normal_active.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/ontop_normal_active.png
new file mode 100644
index 00000000..0f937b7c
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/ontop_normal_active.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/ontop_normal_inactive.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/ontop_normal_inactive.png
new file mode 100644
index 00000000..a9a32068
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/ontop_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/sticky_focus_active.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/sticky_focus_active.png
new file mode 100644
index 00000000..a9bc8a21
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/sticky_focus_active.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/sticky_focus_inactive.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/sticky_focus_inactive.png
new file mode 100644
index 00000000..5493d8ec
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/sticky_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/sticky_normal_active.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/sticky_normal_active.png
new file mode 100644
index 00000000..1e150f58
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/sticky_normal_active.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/sticky_normal_inactive.png b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/sticky_normal_inactive.png
new file mode 100644
index 00000000..7e6c99b6
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/titlebar/sticky_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/vol.png b/Phranque/awesome/themes/powerarrow-dark/icons/vol.png
new file mode 100644
index 00000000..bbf33d42
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/vol.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/vol_low.png b/Phranque/awesome/themes/powerarrow-dark/icons/vol_low.png
new file mode 100644
index 00000000..aa3ce4dc
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/vol_low.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/vol_mute.png b/Phranque/awesome/themes/powerarrow-dark/icons/vol_mute.png
new file mode 100644
index 00000000..e855fd24
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/vol_mute.png differ
diff --git a/Phranque/awesome/themes/powerarrow-dark/icons/vol_no.png b/Phranque/awesome/themes/powerarrow-dark/icons/vol_no.png
new file mode 100644
index 00000000..bbe917bd
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow-dark/icons/vol_no.png differ
diff --git a/theme.lua b/Phranque/awesome/themes/powerarrow-dark/theme.lua
similarity index 100%
rename from theme.lua
rename to Phranque/awesome/themes/powerarrow-dark/theme.lua
diff --git a/Phranque/awesome/themes/powerarrow/binclock.lua b/Phranque/awesome/themes/powerarrow/binclock.lua
new file mode 100644
index 00000000..22018a72
--- /dev/null
+++ b/Phranque/awesome/themes/powerarrow/binclock.lua
@@ -0,0 +1,95 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2017, Luca CPZ
+ * (c) 2013, romockee
+
+--]]
+
+local gears = require("gears")
+local wibox = require("wibox")
+local date = os.date
+local ipairs = ipairs
+local math = math
+local select = select
+local string = string
+
+local binclock = {}
+
+function binclock.dec2bin(num, bits)
+ local bits, t = bits or select(2, math.frexp(num)), {}
+ for b = bits, 1, -1 do
+ t[b] = math.fmod(num, 2)
+ num = (num - t[b]) / 2
+ end
+ return t
+end
+
+function binclock.paintdot(cr, val, shift)
+ local height = 0
+ for _, bit in ipairs(binclock.dec2bin(val, 4)) do
+ if bit >= 1 then
+ cr:set_source(gears.color(binclock.color_active))
+ else
+ cr:set_source(gears.color(binclock.color_inactive))
+ end
+ cr:rectangle(shift, height, binclock.dotsize, binclock.dotsize)
+ cr:fill()
+ height = height + binclock.dotsize + binclock.step
+ end
+end
+
+local function factory(args)
+ local args = args or {}
+
+ binclock.width = args.width or 42
+ binclock.height = args.height or 18
+ binclock.show_seconds = args.show_seconds or false
+ binclock.color_active = args.color_active or "#CCCCCC"
+ binclock.color_inactive = args.color_inactive or "#444444"
+ binclock.dotsize = math.floor(binclock.height / 5)
+ binclock.step = math.floor(binclock.dotsize / 3)
+
+ binclock.widget = wibox.widget {
+ fit = function(self, context, width, height)
+ return binclock.width, binclock.height
+ end,
+ draw = function(self, context, cr, width, height)
+ local t = date("*t")
+
+ local hour = string.format("%02d", t.hour)
+ local min = string.format("%02d", t.min)
+ local sec = string.format("%02d", t.sec)
+
+ local col_count = 4
+ if binclock.show_seconds then
+ col_count = 6
+ end
+ local step = math.floor((binclock.width - col_count * binclock.dotsize) / 8)
+
+ binclock.paintdot(cr, string.sub(hour, 1, 1), step, 2)
+ binclock.paintdot(cr, string.sub(hour, 2, 2), binclock.dotsize + 2 * step)
+
+ binclock.paintdot(cr, string.sub(min, 1, 1), binclock.dotsize * 2 + 4 * step)
+ binclock.paintdot(cr, string.sub(min, 2, 2), binclock.dotsize * 3 + 5 * step)
+
+ if binclock.show_seconds then
+ binclock.paintdot(cr, string.sub(sec, 1, 1), binclock.dotsize * 4 + 7 * step)
+ binclock.paintdot(cr, string.sub(sec, 2, 2), binclock.dotsize * 5 + 8 * step)
+ end
+ end,
+ layout = wibox.widget.base.make_widget
+ }
+
+ binclock.timer = gears.timer {
+ autostart = true,
+ timeout = binclock.show_seconds and 1 or 60,
+ callback = function()
+ binclock.widget:emit_signal("widget::redraw_needed")
+ end
+ }
+
+ return binclock
+end
+
+return factory
diff --git a/Phranque/awesome/themes/powerarrow/icons/ac.png b/Phranque/awesome/themes/powerarrow/icons/ac.png
new file mode 100644
index 00000000..453af4e1
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/ac.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/awesome.png b/Phranque/awesome/themes/powerarrow/icons/awesome.png
new file mode 100644
index 00000000..4cabb6c3
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/awesome.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/battery.png b/Phranque/awesome/themes/powerarrow/icons/battery.png
new file mode 100644
index 00000000..7dd78a57
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/battery.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/battery_empty.png b/Phranque/awesome/themes/powerarrow/icons/battery_empty.png
new file mode 100644
index 00000000..763aa93b
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/battery_empty.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/battery_low.png b/Phranque/awesome/themes/powerarrow/icons/battery_low.png
new file mode 100644
index 00000000..626222f6
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/battery_low.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/cpu.png b/Phranque/awesome/themes/powerarrow/icons/cpu.png
new file mode 100644
index 00000000..c28a7873
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/cpu.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/floating.png b/Phranque/awesome/themes/powerarrow/icons/floating.png
new file mode 100644
index 00000000..079fea4f
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/floating.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/fwd.png b/Phranque/awesome/themes/powerarrow/icons/fwd.png
new file mode 100644
index 00000000..e9b875e7
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/fwd.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/hdd.png b/Phranque/awesome/themes/powerarrow/icons/hdd.png
new file mode 100644
index 00000000..8922eaa4
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/hdd.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/mail.png b/Phranque/awesome/themes/powerarrow/icons/mail.png
new file mode 100644
index 00000000..be656ede
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/mail.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/mail_on.png b/Phranque/awesome/themes/powerarrow/icons/mail_on.png
new file mode 100644
index 00000000..ce8984ff
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/mail_on.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/mem.png b/Phranque/awesome/themes/powerarrow/icons/mem.png
new file mode 100644
index 00000000..ee6d480e
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/mem.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/net.png b/Phranque/awesome/themes/powerarrow/icons/net.png
new file mode 100644
index 00000000..769bfec2
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/net.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/next.png b/Phranque/awesome/themes/powerarrow/icons/next.png
new file mode 100644
index 00000000..9a625931
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/next.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/note.png b/Phranque/awesome/themes/powerarrow/icons/note.png
new file mode 100644
index 00000000..19866aae
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/note.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/note_on.png b/Phranque/awesome/themes/powerarrow/icons/note_on.png
new file mode 100644
index 00000000..aae5ca39
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/note_on.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/pacman.png b/Phranque/awesome/themes/powerarrow/icons/pacman.png
new file mode 100644
index 00000000..575144a0
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/pacman.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/pause.png b/Phranque/awesome/themes/powerarrow/icons/pause.png
new file mode 100644
index 00000000..d4546e67
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/pause.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/phones.png b/Phranque/awesome/themes/powerarrow/icons/phones.png
new file mode 100644
index 00000000..bec0e574
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/phones.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/play.png b/Phranque/awesome/themes/powerarrow/icons/play.png
new file mode 100644
index 00000000..b228f747
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/play.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/prev.png b/Phranque/awesome/themes/powerarrow/icons/prev.png
new file mode 100644
index 00000000..4c4ca40b
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/prev.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/rwd.png b/Phranque/awesome/themes/powerarrow/icons/rwd.png
new file mode 100644
index 00000000..ac3d6f4b
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/rwd.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/scissors.png b/Phranque/awesome/themes/powerarrow/icons/scissors.png
new file mode 100644
index 00000000..30f867db
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/scissors.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/square_sel.png b/Phranque/awesome/themes/powerarrow/icons/square_sel.png
new file mode 100755
index 00000000..874a3a55
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/square_sel.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/square_unsel.png b/Phranque/awesome/themes/powerarrow/icons/square_unsel.png
new file mode 100755
index 00000000..979642f6
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/square_unsel.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/stop.png b/Phranque/awesome/themes/powerarrow/icons/stop.png
new file mode 100644
index 00000000..207915ba
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/stop.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/submenu.png b/Phranque/awesome/themes/powerarrow/icons/submenu.png
new file mode 100755
index 00000000..e9a6b7c5
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/submenu.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/task.png b/Phranque/awesome/themes/powerarrow/icons/task.png
new file mode 100644
index 00000000..962b51a9
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/task.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/temp.png b/Phranque/awesome/themes/powerarrow/icons/temp.png
new file mode 100644
index 00000000..7952de8d
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/temp.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/tile.png b/Phranque/awesome/themes/powerarrow/icons/tile.png
new file mode 100644
index 00000000..01fd1b5f
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/tile.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/tilebottom.png b/Phranque/awesome/themes/powerarrow/icons/tilebottom.png
new file mode 100644
index 00000000..bff1f42b
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/tilebottom.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/tileleft.png b/Phranque/awesome/themes/powerarrow/icons/tileleft.png
new file mode 100644
index 00000000..205140a6
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/tileleft.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/tiletop.png b/Phranque/awesome/themes/powerarrow/icons/tiletop.png
new file mode 100644
index 00000000..148cb9b7
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/tiletop.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/close_focus.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/close_focus.png
new file mode 100644
index 00000000..cccbadca
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/close_focus.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/close_normal.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/close_normal.png
new file mode 100644
index 00000000..cb0c3faa
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/close_normal.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/floating_focus_active.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/floating_focus_active.png
new file mode 100644
index 00000000..77ea1271
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/floating_focus_active.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/floating_focus_inactive.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/floating_focus_inactive.png
new file mode 100644
index 00000000..d254a73d
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/floating_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/floating_normal_active.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/floating_normal_active.png
new file mode 100644
index 00000000..c5ce3ded
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/floating_normal_active.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/floating_normal_inactive.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/floating_normal_inactive.png
new file mode 100644
index 00000000..850b6028
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/floating_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/maximized_focus_active.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/maximized_focus_active.png
new file mode 100644
index 00000000..9fc04833
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/maximized_focus_active.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/maximized_focus_inactive.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/maximized_focus_inactive.png
new file mode 100644
index 00000000..f684470e
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/maximized_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/maximized_normal_active.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/maximized_normal_active.png
new file mode 100644
index 00000000..4aa4196f
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/maximized_normal_active.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/maximized_normal_inactive.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/maximized_normal_inactive.png
new file mode 100644
index 00000000..bfe01a3a
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/maximized_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/ontop_focus_active.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/ontop_focus_active.png
new file mode 100644
index 00000000..86e61b7f
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/ontop_focus_active.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/ontop_focus_inactive.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/ontop_focus_inactive.png
new file mode 100644
index 00000000..286d4399
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/ontop_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/ontop_normal_active.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/ontop_normal_active.png
new file mode 100644
index 00000000..fcee7723
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/ontop_normal_active.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/ontop_normal_inactive.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/ontop_normal_inactive.png
new file mode 100644
index 00000000..a6286269
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/ontop_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/sticky_focus_active.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/sticky_focus_active.png
new file mode 100644
index 00000000..5af45c14
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/sticky_focus_active.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/sticky_focus_inactive.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/sticky_focus_inactive.png
new file mode 100644
index 00000000..22d74534
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/sticky_focus_inactive.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/sticky_normal_active.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/sticky_normal_active.png
new file mode 100644
index 00000000..aad079ed
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/sticky_normal_active.png differ
diff --git a/Phranque/awesome/themes/powerarrow/icons/titlebar/sticky_normal_inactive.png b/Phranque/awesome/themes/powerarrow/icons/titlebar/sticky_normal_inactive.png
new file mode 100644
index 00000000..5f3e655e
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/icons/titlebar/sticky_normal_inactive.png differ
diff --git a/Phranque/awesome/themes/powerarrow/theme.lua b/Phranque/awesome/themes/powerarrow/theme.lua
new file mode 100644
index 00000000..15dcf555
--- /dev/null
+++ b/Phranque/awesome/themes/powerarrow/theme.lua
@@ -0,0 +1,388 @@
+--[[
+
+ Powerarrow Awesome WM theme
+ github.com/lcpz
+
+--]]
+
+local gears = require("gears")
+local lain = require("lain")
+local awful = require("awful")
+local wibox = require("wibox")
+
+local math, string, os = math, string, os
+local my_table = awful.util.table or gears.table -- 4.{0,1} compatibility
+
+local theme = {}
+theme.dir = os.getenv("HOME") .. "/.config/awesome/themes/powerarrow"
+theme.wallpaper = theme.dir .. "/wall.png"
+theme.font = "xos4 Terminus 9"
+theme.fg_normal = "#FEFEFE"
+theme.fg_focus = "#32D6FF"
+theme.fg_urgent = "#C83F11"
+theme.bg_normal = "#222222"
+theme.bg_focus = "#1E2320"
+theme.bg_urgent = "#3F3F3F"
+theme.taglist_fg_focus = "#00CCFF"
+theme.tasklist_bg_focus = "#222222"
+theme.tasklist_fg_focus = "#00CCFF"
+theme.border_width = 1
+theme.border_normal = "#3F3F3F"
+theme.border_focus = "#6F6F6F"
+theme.border_marked = "#CC9393"
+theme.titlebar_bg_focus = "#3F3F3F"
+theme.titlebar_bg_normal = "#3F3F3F"
+theme.titlebar_bg_focus = theme.bg_focus
+theme.titlebar_bg_normal = theme.bg_normal
+theme.titlebar_fg_focus = theme.fg_focus
+theme.menu_height = 16
+theme.menu_width = 140
+theme.menu_submenu_icon = theme.dir .. "/icons/submenu.png"
+theme.awesome_icon = theme.dir .. "/icons/awesome.png"
+theme.taglist_squares_sel = theme.dir .. "/icons/square_sel.png"
+theme.taglist_squares_unsel = theme.dir .. "/icons/square_unsel.png"
+theme.layout_tile = theme.dir .. "/icons/tile.png"
+theme.layout_tileleft = theme.dir .. "/icons/tileleft.png"
+theme.layout_tilebottom = theme.dir .. "/icons/tilebottom.png"
+theme.layout_tiletop = theme.dir .. "/icons/tiletop.png"
+theme.layout_fairv = theme.dir .. "/icons/fairv.png"
+theme.layout_fairh = theme.dir .. "/icons/fairh.png"
+theme.layout_spiral = theme.dir .. "/icons/spiral.png"
+theme.layout_dwindle = theme.dir .. "/icons/dwindle.png"
+theme.layout_max = theme.dir .. "/icons/max.png"
+theme.layout_fullscreen = theme.dir .. "/icons/fullscreen.png"
+theme.layout_magnifier = theme.dir .. "/icons/magnifier.png"
+theme.layout_floating = theme.dir .. "/icons/floating.png"
+theme.widget_ac = theme.dir .. "/icons/ac.png"
+theme.widget_battery = theme.dir .. "/icons/battery.png"
+theme.widget_battery_low = theme.dir .. "/icons/battery_low.png"
+theme.widget_battery_empty = theme.dir .. "/icons/battery_empty.png"
+theme.widget_mem = theme.dir .. "/icons/mem.png"
+theme.widget_cpu = theme.dir .. "/icons/cpu.png"
+theme.widget_temp = theme.dir .. "/icons/temp.png"
+theme.widget_net = theme.dir .. "/icons/net.png"
+theme.widget_hdd = theme.dir .. "/icons/hdd.png"
+theme.widget_music = theme.dir .. "/icons/note.png"
+theme.widget_music_on = theme.dir .. "/icons/note_on.png"
+theme.widget_music_pause = theme.dir .. "/icons/pause.png"
+theme.widget_music_stop = theme.dir .. "/icons/stop.png"
+theme.widget_vol = theme.dir .. "/icons/vol.png"
+theme.widget_vol_low = theme.dir .. "/icons/vol_low.png"
+theme.widget_vol_no = theme.dir .. "/icons/vol_no.png"
+theme.widget_vol_mute = theme.dir .. "/icons/vol_mute.png"
+theme.widget_mail = theme.dir .. "/icons/mail.png"
+theme.widget_mail_on = theme.dir .. "/icons/mail_on.png"
+theme.widget_task = theme.dir .. "/icons/task.png"
+theme.widget_scissors = theme.dir .. "/icons/scissors.png"
+theme.tasklist_plain_task_name = true
+theme.tasklist_disable_icon = true
+theme.useless_gap = 0
+theme.titlebar_close_button_focus = theme.dir .. "/icons/titlebar/close_focus.png"
+theme.titlebar_close_button_normal = theme.dir .. "/icons/titlebar/close_normal.png"
+theme.titlebar_ontop_button_focus_active = theme.dir .. "/icons/titlebar/ontop_focus_active.png"
+theme.titlebar_ontop_button_normal_active = theme.dir .. "/icons/titlebar/ontop_normal_active.png"
+theme.titlebar_ontop_button_focus_inactive = theme.dir .. "/icons/titlebar/ontop_focus_inactive.png"
+theme.titlebar_ontop_button_normal_inactive = theme.dir .. "/icons/titlebar/ontop_normal_inactive.png"
+theme.titlebar_sticky_button_focus_active = theme.dir .. "/icons/titlebar/sticky_focus_active.png"
+theme.titlebar_sticky_button_normal_active = theme.dir .. "/icons/titlebar/sticky_normal_active.png"
+theme.titlebar_sticky_button_focus_inactive = theme.dir .. "/icons/titlebar/sticky_focus_inactive.png"
+theme.titlebar_sticky_button_normal_inactive = theme.dir .. "/icons/titlebar/sticky_normal_inactive.png"
+theme.titlebar_floating_button_focus_active = theme.dir .. "/icons/titlebar/floating_focus_active.png"
+theme.titlebar_floating_button_normal_active = theme.dir .. "/icons/titlebar/floating_normal_active.png"
+theme.titlebar_floating_button_focus_inactive = theme.dir .. "/icons/titlebar/floating_focus_inactive.png"
+theme.titlebar_floating_button_normal_inactive = theme.dir .. "/icons/titlebar/floating_normal_inactive.png"
+theme.titlebar_maximized_button_focus_active = theme.dir .. "/icons/titlebar/maximized_focus_active.png"
+theme.titlebar_maximized_button_normal_active = theme.dir .. "/icons/titlebar/maximized_normal_active.png"
+theme.titlebar_maximized_button_focus_inactive = theme.dir .. "/icons/titlebar/maximized_focus_inactive.png"
+theme.titlebar_maximized_button_normal_inactive = theme.dir .. "/icons/titlebar/maximized_normal_inactive.png"
+
+local markup = lain.util.markup
+local separators = lain.util.separators
+
+-- Binary clock
+local binclock = require("themes.powerarrow.binclock"){
+ height = 16,
+ show_seconds = true,
+ color_active = theme.fg_normal,
+ color_inactive = theme.bg_focus
+}
+
+-- Calendar
+theme.cal = lain.widget.cal({
+ --cal = "cal --color=always",
+ attach_to = { binclock.widget },
+ notification_preset = {
+ font = "xos4 Terminus 10",
+ fg = theme.fg_normal,
+ bg = theme.bg_normal
+ }
+})
+
+-- Taskwarrior
+local task = wibox.widget.imagebox(theme.widget_task)
+lain.widget.contrib.task.attach(task, {
+ -- do not colorize output
+ show_cmd = "task | sed -r 's/\\x1B\\[([0-9]{1,2}(;[0-9]{1,2})?)?[mGK]//g'"
+})
+task:buttons(my_table.join(awful.button({}, 1, lain.widget.contrib.task.prompt)))
+
+-- Scissors (xsel copy and paste)
+local scissors = wibox.widget.imagebox(theme.widget_scissors)
+scissors:buttons(my_table.join(awful.button({}, 1, function() awful.spawn.with_shell("xsel | xsel -i -b") end)))
+
+-- Mail IMAP check
+--[[ commented because it needs to be set before use
+local mailicon = wibox.widget.imagebox(theme.widget_mail)
+mailicon:buttons(my_table.join(awful.button({ }, 1, function () awful.spawn(mail) end)))
+theme.mail = lain.widget.imap({
+ timeout = 180,
+ server = "server",
+ mail = "mail",
+ password = "keyring get mail",
+ settings = function()
+ if mailcount > 0 then
+ widget:set_text(" " .. mailcount .. " ")
+ mailicon:set_image(theme.widget_mail_on)
+ else
+ widget:set_text("")
+ mailicon:set_image(theme.widget_mail)
+ end
+ end
+})
+--]]
+
+-- ALSA volume
+theme.volume = lain.widget.alsabar({
+ --togglechannel = "IEC958,3",
+ notification_preset = { font = "xos4 Terminus 10", fg = theme.fg_normal },
+})
+
+-- MPD
+local musicplr = awful.util.terminal .. " -title Music -g 130x34-320+16 -e ncmpcpp"
+local mpdicon = wibox.widget.imagebox(theme.widget_music)
+mpdicon:buttons(my_table.join(
+ awful.button({ modkey }, 1, function () awful.spawn.with_shell(musicplr) end),
+ awful.button({ }, 1, function ()
+ os.execute("mpc prev")
+ theme.mpd.update()
+ end),
+ awful.button({ }, 2, function ()
+ os.execute("mpc toggle")
+ theme.mpd.update()
+ end),
+ awful.button({ }, 3, function ()
+ os.execute("mpc next")
+ theme.mpd.update()
+ end)))
+theme.mpd = lain.widget.mpd({
+ settings = function()
+ if mpd_now.state == "play" then
+ artist = " " .. mpd_now.artist .. " "
+ title = mpd_now.title .. " "
+ mpdicon:set_image(theme.widget_music_on)
+ widget:set_markup(markup.font(theme.font, markup("#FF8466", artist) .. " " .. title))
+ elseif mpd_now.state == "pause" then
+ widget:set_markup(markup.font(theme.font, " mpd paused "))
+ mpdicon:set_image(theme.widget_music_pause)
+ else
+ widget:set_text("")
+ mpdicon:set_image(theme.widget_music)
+ end
+ end
+})
+
+-- MEM
+local memicon = wibox.widget.imagebox(theme.widget_mem)
+local mem = lain.widget.mem({
+ settings = function()
+ widget:set_markup(markup.font(theme.font, " " .. mem_now.used .. "MB "))
+ end
+})
+
+-- CPU
+local cpuicon = wibox.widget.imagebox(theme.widget_cpu)
+local cpu = lain.widget.cpu({
+ settings = function()
+ widget:set_markup(markup.font(theme.font, " " .. cpu_now.usage .. "% "))
+ end
+})
+
+--[[ Coretemp (lm_sensors, per core)
+local tempwidget = awful.widget.watch({awful.util.shell, '-c', 'sensors | grep Core'}, 30,
+function(widget, stdout)
+ local temps = ""
+ for line in stdout:gmatch("[^\r\n]+") do
+ temps = temps .. line:match("+(%d+).*°C") .. "° " -- in Celsius
+ end
+ widget:set_markup(markup.font(theme.font, " " .. temps))
+end)
+--]]
+-- Coretemp (lain, average)
+local temp = lain.widget.temp({
+ settings = function()
+ widget:set_markup(markup.font(theme.font, " " .. coretemp_now .. "°C "))
+ end
+})
+--]]
+local tempicon = wibox.widget.imagebox(theme.widget_temp)
+
+-- / fs
+local fsicon = wibox.widget.imagebox(theme.widget_hdd)
+--[[ commented because it needs Gio/Glib >= 2.54
+theme.fs = lain.widget.fs({
+ notification_preset = { fg = theme.fg_normal, bg = theme.bg_normal, font = "xos4 Terminus 10" },
+ settings = function()
+ local fsp = string.format(" %3.2f %s ", fs_now["/"].free, fs_now["/"].units)
+ widget:set_markup(markup.font(theme.font, fsp))
+ end
+})
+--]]
+
+-- Battery
+local baticon = wibox.widget.imagebox(theme.widget_battery)
+local bat = lain.widget.bat({
+ settings = function()
+ if bat_now.status and bat_now.status ~= "N/A" then
+ if bat_now.ac_status == 1 then
+ widget:set_markup(markup.font(theme.font, " AC "))
+ baticon:set_image(theme.widget_ac)
+ return
+ elseif not bat_now.perc and tonumber(bat_now.perc) <= 5 then
+ baticon:set_image(theme.widget_battery_empty)
+ elseif not bat_now.perc and tonumber(bat_now.perc) <= 15 then
+ baticon:set_image(theme.widget_battery_low)
+ else
+ baticon:set_image(theme.widget_battery)
+ end
+ widget:set_markup(markup.font(theme.font, " " .. bat_now.perc .. "% "))
+ else
+ widget:set_markup()
+ baticon:set_image(theme.widget_ac)
+ end
+ end
+})
+
+-- Net
+local neticon = wibox.widget.imagebox(theme.widget_net)
+local net = lain.widget.net({
+ settings = function()
+ widget:set_markup(markup.fontfg(theme.font, "#FEFEFE", " " .. net_now.received .. " ↓↑ " .. net_now.sent .. " "))
+ end
+})
+
+-- Separators
+local arrow = separators.arrow_left
+
+function theme.powerline_rl(cr, width, height)
+ local arrow_depth, offset = height/2, 0
+
+ -- Avoid going out of the (potential) clip area
+ if arrow_depth < 0 then
+ width = width + 2*arrow_depth
+ offset = -arrow_depth
+ end
+
+ cr:move_to(offset + arrow_depth , 0 )
+ cr:line_to(offset + width , 0 )
+ cr:line_to(offset + width - arrow_depth , height/2 )
+ cr:line_to(offset + width , height )
+ cr:line_to(offset + arrow_depth , height )
+ cr:line_to(offset , height/2 )
+
+ cr:close_path()
+end
+
+local function pl(widget, bgcolor, padding)
+ return wibox.container.background(wibox.container.margin(widget, 16, 16), bgcolor, theme.powerline_rl)
+end
+
+function theme.at_screen_connect(s)
+ -- Quake application
+ s.quake = lain.util.quake({ app = awful.util.terminal })
+
+ -- If wallpaper is a function, call it with the screen
+ local wallpaper = theme.wallpaper
+ if type(wallpaper) == "function" then
+ wallpaper = wallpaper(s)
+ end
+ gears.wallpaper.maximized(wallpaper, s, true)
+
+ -- Tags
+ awful.tag(awful.util.tagnames, s, awful.layout.layouts)
+
+ -- Create a promptbox for each screen
+ s.mypromptbox = awful.widget.prompt()
+ -- Create an imagebox widget which will contains an icon indicating which layout we're using.
+ -- We need one layoutbox per screen.
+ s.mylayoutbox = awful.widget.layoutbox(s)
+ s.mylayoutbox:buttons(my_table.join(
+ awful.button({}, 1, function () awful.layout.inc( 1) end),
+ awful.button({}, 2, function () awful.layout.set( awful.layout.layouts[1] ) end),
+ awful.button({}, 3, function () awful.layout.inc(-1) end),
+ awful.button({}, 4, function () awful.layout.inc( 1) end),
+ awful.button({}, 5, function () awful.layout.inc(-1) end)))
+ -- Create a taglist widget
+ s.mytaglist = awful.widget.taglist(s, awful.widget.taglist.filter.all, awful.util.taglist_buttons)
+
+ -- Create a tasklist widget
+ s.mytasklist = awful.widget.tasklist(s, awful.widget.tasklist.filter.currenttags, awful.util.tasklist_buttons)
+
+ -- Create the wibox
+ s.mywibox = awful.wibar({ position = "top", screen = s, height = 16, bg = theme.bg_normal, fg = theme.fg_normal })
+
+ -- Add widgets to the wibox
+ s.mywibox:setup {
+ layout = wibox.layout.align.horizontal,
+ { -- Left widgets
+ layout = wibox.layout.fixed.horizontal,
+ --spr,
+ s.mytaglist,
+ s.mypromptbox,
+ spr,
+ },
+ s.mytasklist, -- Middle widget
+ { -- Right widgets
+ layout = wibox.layout.fixed.horizontal,
+ wibox.widget.systray(),
+ wibox.container.margin(scissors, 4, 8),
+ --[[ using shapes
+ pl(wibox.widget { mpdicon, theme.mpd.widget, layout = wibox.layout.align.horizontal }, "#343434"),
+ pl(task, "#343434"),
+ --pl(wibox.widget { mailicon, mail and theme.mail.widget, layout = wibox.layout.align.horizontal }, "#343434"),
+ pl(wibox.widget { memicon, mem.widget, layout = wibox.layout.align.horizontal }, "#777E76"),
+ pl(wibox.widget { cpuicon, cpu.widget, layout = wibox.layout.align.horizontal }, "#4B696D"),
+ pl(wibox.widget { tempicon, temp.widget, layout = wibox.layout.align.horizontal }, "#4B3B51"),
+ --pl(wibox.widget { fsicon, theme.fs and theme.fs.widget, layout = wibox.layout.align.horizontal }, "#CB755B"),
+ pl(wibox.widget { baticon, bat.widget, layout = wibox.layout.align.horizontal }, "#8DAA9A"),
+ pl(wibox.widget { neticon, net.widget, layout = wibox.layout.align.horizontal }, "#C0C0A2"),
+ pl(binclock.widget, "#777E76"),
+ --]]
+ -- using separators
+ arrow(theme.bg_normal, "#343434"),
+ wibox.container.background(wibox.container.margin(wibox.widget { mailicon, theme.mail and theme.mail.widget, layout = wibox.layout.align.horizontal }, 4, 7), "#343434"),
+ arrow("#343434", theme.bg_normal),
+ wibox.container.background(wibox.container.margin(wibox.widget { mpdicon, theme.mpd.widget, layout = wibox.layout.align.horizontal }, 3, 6), theme.bg_focus),
+ arrow(theme.bg_normal, "#343434"),
+ wibox.container.background(wibox.container.margin(task, 3, 7), "#343434"),
+ arrow("#343434", "#777E76"),
+ wibox.container.background(wibox.container.margin(wibox.widget { memicon, mem.widget, layout = wibox.layout.align.horizontal }, 2, 3), "#777E76"),
+ arrow("#777E76", "#4B696D"),
+ wibox.container.background(wibox.container.margin(wibox.widget { cpuicon, cpu.widget, layout = wibox.layout.align.horizontal }, 3, 4), "#4B696D"),
+ arrow("#4B696D", "#4B3B51"),
+ wibox.container.background(wibox.container.margin(wibox.widget { tempicon, temp.widget, layout = wibox.layout.align.horizontal }, 4, 4), "#4B3B51"),
+ arrow("#4B3B51", "#CB755B"),
+ wibox.container.background(wibox.container.margin(wibox.widget { fsicon, theme.fs and theme.fs.widget, layout = wibox.layout.align.horizontal }, 3, 3), "#CB755B"),
+ arrow("#CB755B", "#8DAA9A"),
+ wibox.container.background(wibox.container.margin(wibox.widget { baticon, bat.widget, layout = wibox.layout.align.horizontal }, 3, 3), "#8DAA9A"),
+ arrow("#8DAA9A", "#C0C0A2"),
+ wibox.container.background(wibox.container.margin(wibox.widget { nil, neticon, net.widget, layout = wibox.layout.align.horizontal }, 3, 3), "#C0C0A2"),
+ arrow("#C0C0A2", "#777E76"),
+ wibox.container.background(wibox.container.margin(binclock.widget, 4, 8), "#777E76"),
+ arrow("#777E76", "alpha"),
+ --]]
+ s.mylayoutbox,
+ },
+ }
+end
+
+return theme
diff --git a/Phranque/awesome/themes/powerarrow/wall.png b/Phranque/awesome/themes/powerarrow/wall.png
new file mode 100644
index 00000000..0989db92
Binary files /dev/null and b/Phranque/awesome/themes/powerarrow/wall.png differ
diff --git a/Phranque/awesome/themes/rainbow/icons/awesome.png b/Phranque/awesome/themes/rainbow/icons/awesome.png
new file mode 100644
index 00000000..d99fd37d
Binary files /dev/null and b/Phranque/awesome/themes/rainbow/icons/awesome.png differ
diff --git a/Phranque/awesome/themes/rainbow/icons/square_sel.png b/Phranque/awesome/themes/rainbow/icons/square_sel.png
new file mode 100644
index 00000000..532719f8
Binary files /dev/null and b/Phranque/awesome/themes/rainbow/icons/square_sel.png differ
diff --git a/Phranque/awesome/themes/rainbow/icons/square_unsel.png b/Phranque/awesome/themes/rainbow/icons/square_unsel.png
new file mode 100644
index 00000000..72dea5b8
Binary files /dev/null and b/Phranque/awesome/themes/rainbow/icons/square_unsel.png differ
diff --git a/Phranque/awesome/themes/rainbow/icons/submenu.png b/Phranque/awesome/themes/rainbow/icons/submenu.png
new file mode 100644
index 00000000..26ab82a0
Binary files /dev/null and b/Phranque/awesome/themes/rainbow/icons/submenu.png differ
diff --git a/Phranque/awesome/themes/rainbow/theme.lua b/Phranque/awesome/themes/rainbow/theme.lua
new file mode 100644
index 00000000..3402c5e6
--- /dev/null
+++ b/Phranque/awesome/themes/rainbow/theme.lua
@@ -0,0 +1,275 @@
+--[[
+
+ Rainbow Awesome WM theme 2.0
+ github.com/lcpz
+
+--]]
+
+local gears = require("gears")
+local lain = require("lain")
+local awful = require("awful")
+local wibox = require("wibox")
+
+local os = os
+local my_table = awful.util.table or gears.table -- 4.{0,1} compatibility
+
+local theme = {}
+theme.default_dir = require("awful.util").get_themes_dir() .. "default"
+theme.dir = os.getenv("HOME") .. "/.config/awesome/themes/rainbow"
+theme.wallpaper = theme.dir .. "/wall.png"
+theme.font = "Misc Tamsyn 10.5"
+theme.fg_normal = "#9E9E9E"
+theme.fg_focus = "#EBEBFF"
+theme.bg_normal = "#242424"
+theme.bg_focus = "#242424"
+theme.fg_urgent = "#000000"
+theme.bg_urgent = "#FFFFFF"
+theme.border_width = 1
+theme.border_normal = "#242424"
+theme.border_focus = "#EBEBFF"
+theme.taglist_fg_focus = "#EDEFFF"
+theme.taglist_bg_focus = "#242424"
+theme.menu_height = 16
+theme.menu_width = 140
+theme.ocol = ""
+theme.tasklist_sticky = theme.ocol .. "[S]"
+theme.tasklist_ontop = theme.ocol .. "[T]"
+theme.tasklist_floating = theme.ocol .. "[F]"
+theme.tasklist_maximized_horizontal = theme.ocol .. "[M] "
+theme.tasklist_maximized_vertical = ""
+theme.tasklist_disable_icon = true
+theme.awesome_icon = theme.dir .."/icons/awesome.png"
+theme.menu_submenu_icon = theme.dir .."/icons/submenu.png"
+theme.taglist_squares_sel = theme.dir .. "/icons/square_sel.png"
+theme.taglist_squares_unsel = theme.dir .. "/icons/square_unsel.png"
+theme.useless_gap = 8
+theme.layout_txt_tile = "[t]"
+theme.layout_txt_tileleft = "[l]"
+theme.layout_txt_tilebottom = "[b]"
+theme.layout_txt_tiletop = "[tt]"
+theme.layout_txt_fairv = "[fv]"
+theme.layout_txt_fairh = "[fh]"
+theme.layout_txt_spiral = "[s]"
+theme.layout_txt_dwindle = "[d]"
+theme.layout_txt_max = "[m]"
+theme.layout_txt_fullscreen = "[F]"
+theme.layout_txt_magnifier = "[M]"
+theme.layout_txt_floating = "[*]"
+theme.titlebar_close_button_normal = theme.default_dir.."/titlebar/close_normal.png"
+theme.titlebar_close_button_focus = theme.default_dir.."/titlebar/close_focus.png"
+theme.titlebar_minimize_button_normal = theme.default_dir.."/titlebar/minimize_normal.png"
+theme.titlebar_minimize_button_focus = theme.default_dir.."/titlebar/minimize_focus.png"
+theme.titlebar_ontop_button_normal_inactive = theme.default_dir.."/titlebar/ontop_normal_inactive.png"
+theme.titlebar_ontop_button_focus_inactive = theme.default_dir.."/titlebar/ontop_focus_inactive.png"
+theme.titlebar_ontop_button_normal_active = theme.default_dir.."/titlebar/ontop_normal_active.png"
+theme.titlebar_ontop_button_focus_active = theme.default_dir.."/titlebar/ontop_focus_active.png"
+theme.titlebar_sticky_button_normal_inactive = theme.default_dir.."/titlebar/sticky_normal_inactive.png"
+theme.titlebar_sticky_button_focus_inactive = theme.default_dir.."/titlebar/sticky_focus_inactive.png"
+theme.titlebar_sticky_button_normal_active = theme.default_dir.."/titlebar/sticky_normal_active.png"
+theme.titlebar_sticky_button_focus_active = theme.default_dir.."/titlebar/sticky_focus_active.png"
+theme.titlebar_floating_button_normal_inactive = theme.default_dir.."/titlebar/floating_normal_inactive.png"
+theme.titlebar_floating_button_focus_inactive = theme.default_dir.."/titlebar/floating_focus_inactive.png"
+theme.titlebar_floating_button_normal_active = theme.default_dir.."/titlebar/floating_normal_active.png"
+theme.titlebar_floating_button_focus_active = theme.default_dir.."/titlebar/floating_focus_active.png"
+theme.titlebar_maximized_button_normal_inactive = theme.default_dir.."/titlebar/maximized_normal_inactive.png"
+theme.titlebar_maximized_button_focus_inactive = theme.default_dir.."/titlebar/maximized_focus_inactive.png"
+theme.titlebar_maximized_button_normal_active = theme.default_dir.."/titlebar/maximized_normal_active.png"
+theme.titlebar_maximized_button_focus_active = theme.default_dir.."/titlebar/maximized_focus_active.png"
+
+-- lain related
+theme.layout_txt_cascade = "[cascade]"
+theme.layout_txt_cascadetile = "[cascadetile]"
+theme.layout_txt_centerwork = "[centerwork]"
+theme.layout_txt_termfair = "[termfair]"
+theme.layout_txt_centerfair = "[centerfair]"
+
+local markup = lain.util.markup
+local white = theme.fg_focus
+local gray = theme.fg_normal
+
+-- Textclock
+local mytextclock = wibox.widget.textclock(markup(white, " %H:%M "))
+mytextclock.font = theme.font
+
+-- Calendar
+theme.cal = lain.widget.cal({
+ attach_to = { mytextclock },
+ notification_preset = {
+ font = "Misc Tamsyn 11",
+ fg = white,
+ bg = theme.bg_normal
+ }
+})
+
+-- Mail IMAP check
+--[[ commented because it needs to be set before use
+theme.mail = lain.widget.imap({
+ timeout = 180,
+ server = "server",
+ mail = "mail",
+ password = "keyring get mail",
+ settings = function()
+ mail_notification_preset.fg = white
+
+ mail = ""
+ count = ""
+
+ if mailcount > 0 then
+ mail = "Mail "
+ count = mailcount .. " "
+ end
+
+ widget:set_markup(markup.font(theme.font, markup(gray, mail) .. markup(white, count)))
+ end
+})
+--]]
+
+-- MPD
+theme.mpd = lain.widget.mpd({
+ settings = function()
+ mpd_notification_preset.fg = white
+
+ artist = mpd_now.artist .. " "
+ title = mpd_now.title .. " "
+
+ if mpd_now.state == "pause" then
+ artist = "mpd "
+ title = "paused "
+ elseif mpd_now.state == "stop" then
+ artist = ""
+ title = ""
+ end
+
+ widget:set_markup(markup.font(theme.font, markup(gray, artist) .. markup(white, title)))
+ end
+})
+
+-- /home fs
+--[[ commented because it needs Gio/Glib >= 2.54
+theme.fs = lain.widget.fs({
+ notification_preset = { fg = white, bg = theme.bg_normal, font = "Misc Tamsyn 10.5" },
+ settings = function()
+ local fs_header, fs_p = "", ""
+
+ if fs_now["/home"].percentage >= 90 then
+ fs_header = " Hdd "
+ fs_p = fs_now["/home"].percentage
+ end
+
+ widget:set_markup(markup.font(theme.font, markup(gray, fs_header) .. markup(white, fs_p)))
+ end
+})
+--]]
+
+-- ALSA volume bar
+theme.volume = lain.widget.alsabar({
+ ticks = true, width = 67,
+ notification_preset = { font = theme.font }
+})
+theme.volume.tooltip.wibox.fg = theme.fg_focus
+theme.volume.tooltip.wibox.font = theme.font
+theme.volume.bar:buttons(my_table.join (
+ awful.button({}, 1, function()
+ awful.spawn(string.format("%s -e alsamixer", terminal))
+ end),
+ awful.button({}, 2, function()
+ os.execute(string.format("%s set %s 100%%", theme.volume.cmd, theme.volume.channel))
+ theme.volume.update()
+ end),
+ awful.button({}, 3, function()
+ os.execute(string.format("%s set %s toggle", theme.volume.cmd, theme.volume.togglechannel or theme.volume.channel))
+ theme.volume.update()
+ end),
+ awful.button({}, 4, function()
+ os.execute(string.format("%s set %s 1%%+", theme.volume.cmd, theme.volume.channel))
+ theme.volume.update()
+ end),
+ awful.button({}, 5, function()
+ os.execute(string.format("%s set %s 1%%-", theme.volume.cmd, theme.volume.channel))
+ theme.volume.update()
+ end)
+))
+local volumebg = wibox.container.background(theme.volume.bar, "#585858", gears.shape.rectangle)
+local volumewidget = wibox.container.margin(volumebg, 7, 7, 5, 5)
+
+-- Weather
+theme.weather = lain.widget.weather({
+ city_id = 2643743, -- placeholder (London)
+ notification_preset = { font = theme.font, fg = white }
+})
+
+-- Separators
+local first = wibox.widget.textbox(markup.font("Misc Tamsyn 4", " "))
+local spr = wibox.widget.textbox(' ')
+
+local function update_txt_layoutbox(s)
+ -- Writes a string representation of the current layout in a textbox widget
+ local txt_l = theme["layout_txt_" .. awful.layout.getname(awful.layout.get(s))] or ""
+ s.mytxtlayoutbox:set_text(txt_l)
+end
+
+function theme.at_screen_connect(s)
+ -- Quake application
+ s.quake = lain.util.quake({ app = awful.util.terminal })
+
+ -- If wallpaper is a function, call it with the screen
+ local wallpaper = theme.wallpaper
+ if type(wallpaper) == "function" then
+ wallpaper = wallpaper(s)
+ end
+ gears.wallpaper.maximized(wallpaper, s, true)
+
+ -- Tags
+ awful.tag(awful.util.tagnames, s, awful.layout.layouts)
+
+ -- Create a promptbox for each screen
+ s.mypromptbox = awful.widget.prompt()
+
+ -- Textual layoutbox
+ s.mytxtlayoutbox = wibox.widget.textbox(theme["layout_txt_" .. awful.layout.getname(awful.layout.get(s))])
+ awful.tag.attached_connect_signal(s, "property::selected", function () update_txt_layoutbox(s) end)
+ awful.tag.attached_connect_signal(s, "property::layout", function () update_txt_layoutbox(s) end)
+ s.mytxtlayoutbox:buttons(my_table.join(
+ awful.button({}, 1, function() awful.layout.inc(1) end),
+ awful.button({}, 2, function () awful.layout.set( awful.layout.layouts[1] ) end),
+ awful.button({}, 3, function() awful.layout.inc(-1) end),
+ awful.button({}, 4, function() awful.layout.inc(1) end),
+ awful.button({}, 5, function() awful.layout.inc(-1) end)))
+
+ -- Create a taglist widget
+ s.mytaglist = awful.widget.taglist(s, awful.widget.taglist.filter.all, awful.util.taglist_buttons)
+
+ -- Create a tasklist widget
+ s.mytasklist = awful.widget.tasklist(s, awful.widget.tasklist.filter.currenttags, awful.util.tasklist_buttons)
+
+ -- Create the wibox
+ s.mywibox = awful.wibar({ position = "top", screen = s, height = 18, bg = theme.bg_normal, fg = theme.fg_normal })
+
+ -- Add widgets to the wibox
+ s.mywibox:setup {
+ layout = wibox.layout.align.horizontal,
+ { -- Left widgets
+ layout = wibox.layout.fixed.horizontal,
+ first,
+ s.mytaglist,
+ spr,
+ s.mytxtlayoutbox,
+ --spr,
+ s.mypromptbox,
+ spr,
+ },
+ s.mytasklist, -- Middle widget
+ { -- Right widgets
+ layout = wibox.layout.fixed.horizontal,
+ wibox.widget.systray(),
+ spr,
+ theme.mpd.widget,
+ --theme.mail.widget,
+ --theme.fs.widget,
+ volumewidget,
+ mytextclock,
+ },
+ }
+end
+
+return theme
diff --git a/Phranque/awesome/themes/rainbow/wall.png b/Phranque/awesome/themes/rainbow/wall.png
new file mode 100644
index 00000000..6bed2383
Binary files /dev/null and b/Phranque/awesome/themes/rainbow/wall.png differ
diff --git a/Phranque/awesome/themes/steamburn/icons/awesome.png b/Phranque/awesome/themes/steamburn/icons/awesome.png
new file mode 100644
index 00000000..0306bdf2
Binary files /dev/null and b/Phranque/awesome/themes/steamburn/icons/awesome.png differ
diff --git a/Phranque/awesome/themes/steamburn/icons/square_sel.png b/Phranque/awesome/themes/steamburn/icons/square_sel.png
new file mode 100644
index 00000000..2bcabb8e
Binary files /dev/null and b/Phranque/awesome/themes/steamburn/icons/square_sel.png differ
diff --git a/Phranque/awesome/themes/steamburn/icons/square_unsel.png b/Phranque/awesome/themes/steamburn/icons/square_unsel.png
new file mode 100644
index 00000000..14c4fe47
Binary files /dev/null and b/Phranque/awesome/themes/steamburn/icons/square_unsel.png differ
diff --git a/Phranque/awesome/themes/steamburn/icons/submenu.png b/Phranque/awesome/themes/steamburn/icons/submenu.png
new file mode 100644
index 00000000..12b97c6a
Binary files /dev/null and b/Phranque/awesome/themes/steamburn/icons/submenu.png differ
diff --git a/Phranque/awesome/themes/steamburn/theme.lua b/Phranque/awesome/themes/steamburn/theme.lua
new file mode 100644
index 00000000..acb709d4
--- /dev/null
+++ b/Phranque/awesome/themes/steamburn/theme.lua
@@ -0,0 +1,274 @@
+--[[
+
+ Steamburn Awesome WM theme 3.0
+ github.com/lcpz
+
+--]]
+
+local gears = require("gears")
+local lain = require("lain")
+local awful = require("awful")
+local wibox = require("wibox")
+
+local os = os
+local my_table = awful.util.table or gears.table -- 4.{0,1} compatibility
+
+local theme = {}
+theme.zenburn_dir = require("awful.util").get_themes_dir() .. "zenburn"
+theme.dir = os.getenv("HOME") .. "/.config/awesome/themes/steamburn"
+theme.wallpaper = theme.dir .. "/wall.png"
+theme.font = "Misc Tamsyn 10.5"
+theme.fg_normal = "#e2ccb0"
+theme.fg_focus = "#d88166"
+theme.fg_urgent = "#CC9393"
+theme.bg_normal = "#140c0b"
+theme.bg_focus = "#140c0b"
+theme.bg_urgent = "#2a1f1e"
+theme.border_width = 1
+theme.border_normal = "#302627"
+theme.border_focus = "#c2745b"
+theme.border_marked = "#CC9393"
+theme.taglist_fg_focus = "#d88166"
+theme.tasklist_bg_focus = "#140c0b"
+theme.tasklist_fg_focus = "#d88166"
+theme.taglist_squares_sel = theme.dir .. "/icons/square_sel.png"
+theme.taglist_squares_unsel = theme.dir .. "/icons/square_unsel.png"
+theme.menu_height = 16
+theme.menu_width = 140
+theme.awesome_icon = theme.dir .."/icons/awesome.png"
+theme.menu_submenu_icon = theme.dir .. "/icons/submenu.png"
+theme.layout_txt_tile = "[t]"
+theme.layout_txt_tileleft = "[l]"
+theme.layout_txt_tilebottom = "[b]"
+theme.layout_txt_tiletop = "[tt]"
+theme.layout_txt_fairv = "[fv]"
+theme.layout_txt_fairh = "[fh]"
+theme.layout_txt_spiral = "[s]"
+theme.layout_txt_dwindle = "[d]"
+theme.layout_txt_max = "[m]"
+theme.layout_txt_fullscreen = "[F]"
+theme.layout_txt_magnifier = "[M]"
+theme.layout_txt_floating = "[|]"
+theme.tasklist_plain_task_name = true
+theme.tasklist_disable_icon = true
+theme.useless_gap = 0
+theme.titlebar_close_button_normal = theme.zenburn_dir.."/titlebar/close_normal.png"
+theme.titlebar_close_button_focus = theme.zenburn_dir.."/titlebar/close_focus.png"
+theme.titlebar_minimize_button_normal = theme.zenburn_dir.."/titlebar/minimize_normal.png"
+theme.titlebar_minimize_button_focus = theme.zenburn_dir.."/titlebar/minimize_focus.png"
+theme.titlebar_ontop_button_normal_inactive = theme.zenburn_dir.."/titlebar/ontop_normal_inactive.png"
+theme.titlebar_ontop_button_focus_inactive = theme.zenburn_dir.."/titlebar/ontop_focus_inactive.png"
+theme.titlebar_ontop_button_normal_active = theme.zenburn_dir.."/titlebar/ontop_normal_active.png"
+theme.titlebar_ontop_button_focus_active = theme.zenburn_dir.."/titlebar/ontop_focus_active.png"
+theme.titlebar_sticky_button_normal_inactive = theme.zenburn_dir.."/titlebar/sticky_normal_inactive.png"
+theme.titlebar_sticky_button_focus_inactive = theme.zenburn_dir.."/titlebar/sticky_focus_inactive.png"
+theme.titlebar_sticky_button_normal_active = theme.zenburn_dir.."/titlebar/sticky_normal_active.png"
+theme.titlebar_sticky_button_focus_active = theme.zenburn_dir.."/titlebar/sticky_focus_active.png"
+theme.titlebar_floating_button_normal_inactive = theme.zenburn_dir.."/titlebar/floating_normal_inactive.png"
+theme.titlebar_floating_button_focus_inactive = theme.zenburn_dir.."/titlebar/floating_focus_inactive.png"
+theme.titlebar_floating_button_normal_active = theme.zenburn_dir.."/titlebar/floating_normal_active.png"
+theme.titlebar_floating_button_focus_active = theme.zenburn_dir.."/titlebar/floating_focus_active.png"
+theme.titlebar_maximized_button_normal_inactive = theme.zenburn_dir.."/titlebar/maximized_normal_inactive.png"
+theme.titlebar_maximized_button_focus_inactive = theme.zenburn_dir.."/titlebar/maximized_focus_inactive.png"
+theme.titlebar_maximized_button_normal_active = theme.zenburn_dir.."/titlebar/maximized_normal_active.png"
+theme.titlebar_maximized_button_focus_active = theme.zenburn_dir.."/titlebar/maximized_focus_active.png"
+
+-- lain related
+theme.layout_txt_termfair = "[termfair]"
+theme.layout_txt_centerfair = "[centerfair]"
+
+local markup = lain.util.markup
+local gray = "#94928F"
+
+-- Textclock
+local mytextclock = wibox.widget.textclock(" %H:%M ")
+mytextclock.font = theme.font
+
+-- Calendar
+theme.cal = lain.widget.cal({
+ attach_to = { mytextclock },
+ notification_preset = {
+ font = "Misc Tamsyn 11",
+ fg = theme.fg_normal,
+ bg = theme.bg_normal
+ }
+})
+
+-- Mail IMAP check
+--[[ commented because it needs to be set before use
+theme.mail = lain.widget.imap({
+ timeout = 180,
+ server = "server",
+ mail = "mail",
+ password = "keyring get mail",
+ settings = function()
+ mail = ""
+ count = ""
+
+ if mailcount > 0 then
+ mail = "Mail "
+ count = mailcount .. " "
+ end
+
+ widget:set_markup(markup(gray, mail) .. count)
+ end
+})
+--]]
+
+-- MPD
+theme.mpd = lain.widget.mpd({
+ settings = function()
+ artist = mpd_now.artist .. " "
+ title = mpd_now.title .. " "
+
+ if mpd_now.state == "pause" then
+ artist = "mpd "
+ title = "paused "
+ elseif mpd_now.state == "stop" then
+ artist = ""
+ title = ""
+ end
+
+ widget:set_markup(markup.font(theme.font, markup(gray, artist) .. title))
+ end
+})
+
+-- CPU
+local cpu = lain.widget.sysload({
+ settings = function()
+ widget:set_markup(markup.font(theme.font, markup(gray, " Cpu ") .. load_1 .. " "))
+ end
+})
+
+-- MEM
+local mem = lain.widget.mem({
+ settings = function()
+ widget:set_markup(markup.font(theme.font, markup(gray, " Mem ") .. mem_now.used .. " "))
+ end
+})
+
+-- /home fs
+--[[ commented because it needs Gio/Glib >= 2.54
+theme.fs = lain.widget.fs({
+ partition = "/home",
+ notification_preset = { fg = theme.fg_normal, bg = theme.bg_normal, font = "Misc Tamsyn 10.5" },
+})
+--]]
+
+-- Battery
+local bat = lain.widget.bat({
+ settings = function()
+ local perc = bat_now.perc
+ if bat_now.ac_status == 1 then perc = perc .. " Plug" end
+ widget:set_markup(markup.font(theme.font, markup(gray, " Bat ") .. perc .. " "))
+ end
+})
+
+-- Net checker
+local net = lain.widget.net({
+ settings = function()
+ if net_now.state == "up" then net_state = "On"
+ else net_state = "Off" end
+ widget:set_markup(markup.font(theme.font, markup(gray, " Net ") .. net_state .. " "))
+ end
+})
+
+-- ALSA volume
+theme.volume = lain.widget.alsa({
+ settings = function()
+ header = " Vol "
+ vlevel = volume_now.level
+
+ if volume_now.status == "off" then
+ vlevel = vlevel .. "M "
+ else
+ vlevel = vlevel .. " "
+ end
+
+ widget:set_markup(markup.font(theme.font, markup(gray, header) .. vlevel))
+ end
+})
+
+-- Weather
+theme.weather = lain.widget.weather({
+ city_id = 2643743, -- placeholder (London)
+})
+
+-- Separators
+local first = wibox.widget.textbox(markup.font("Misc Tamsyn 4", " "))
+local spr = wibox.widget.textbox(' ')
+
+local function update_txt_layoutbox(s)
+ -- Writes a string representation of the current layout in a textbox widget
+ local txt_l = theme["layout_txt_" .. awful.layout.getname(awful.layout.get(s))] or ""
+ s.mytxtlayoutbox:set_text(txt_l)
+end
+
+function theme.at_screen_connect(s)
+ -- Quake application
+ s.quake = lain.util.quake({ app = awful.util.terminal })
+
+ -- If wallpaper is a function, call it with the screen
+ local wallpaper = theme.wallpaper
+ if type(wallpaper) == "function" then
+ wallpaper = wallpaper(s)
+ end
+ gears.wallpaper.maximized(wallpaper, s, true)
+
+ -- Tags
+ awful.tag(awful.util.tagnames, s, awful.layout.layouts)
+
+ -- Create a promptbox for each screen
+ s.mypromptbox = awful.widget.prompt()
+
+ -- Textual layoutbox
+ s.mytxtlayoutbox = wibox.widget.textbox(theme["layout_txt_" .. awful.layout.getname(awful.layout.get(s))])
+ awful.tag.attached_connect_signal(s, "property::selected", function () update_txt_layoutbox(s) end)
+ awful.tag.attached_connect_signal(s, "property::layout", function () update_txt_layoutbox(s) end)
+ s.mytxtlayoutbox:buttons(my_table.join(
+ awful.button({}, 1, function() awful.layout.inc(1) end),
+ awful.button({}, 2, function () awful.layout.set( awful.layout.layouts[1] ) end),
+ awful.button({}, 3, function() awful.layout.inc(-1) end),
+ awful.button({}, 4, function() awful.layout.inc(1) end),
+ awful.button({}, 5, function() awful.layout.inc(-1) end)))
+
+ -- Create a taglist widget
+ s.mytaglist = awful.widget.taglist(s, awful.widget.taglist.filter.all, awful.util.taglist_buttons)
+
+ -- Create a tasklist widget
+ s.mytasklist = awful.widget.tasklist(s, awful.widget.tasklist.filter.currenttags, awful.util.tasklist_buttons)
+
+ -- Create the wibox
+ s.mywibox = awful.wibar({ position = "top", screen = s, height = 18 })
+
+ -- Add widgets to the wibox
+ s.mywibox:setup {
+ layout = wibox.layout.align.horizontal,
+ { -- Left widgets
+ layout = wibox.layout.fixed.horizontal,
+ first,
+ s.mytaglist,
+ spr,
+ s.mytxtlayoutbox,
+ --spr,
+ s.mypromptbox,
+ spr,
+ },
+ s.mytasklist, -- Middle widget
+ { -- Right widgets
+ layout = wibox.layout.fixed.horizontal,
+ wibox.widget.systray(),
+ spr,
+ theme.mpd.widget,
+ --theme.mail.widget,
+ cpu.widget,
+ mem.widget,
+ bat.widget,
+ net.widget,
+ theme.volume.widget,
+ mytextclock
+ },
+ }
+end
+
+return theme
diff --git a/Phranque/awesome/themes/steamburn/wall.png b/Phranque/awesome/themes/steamburn/wall.png
new file mode 100644
index 00000000..57359d16
Binary files /dev/null and b/Phranque/awesome/themes/steamburn/wall.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/awesome.png b/Phranque/awesome/themes/vertex/icons/awesome.png
new file mode 100644
index 00000000..cbded649
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/awesome.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/bat-000-charging.png b/Phranque/awesome/themes/vertex/icons/bat-000-charging.png
new file mode 100644
index 00000000..cc2abdd3
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/bat-000-charging.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/bat-000.png b/Phranque/awesome/themes/vertex/icons/bat-000.png
new file mode 100644
index 00000000..550d8f05
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/bat-000.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/bat-020-charging.png b/Phranque/awesome/themes/vertex/icons/bat-020-charging.png
new file mode 100644
index 00000000..bd5e788b
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/bat-020-charging.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/bat-020.png b/Phranque/awesome/themes/vertex/icons/bat-020.png
new file mode 100644
index 00000000..c8525eb0
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/bat-020.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/bat-040-charging.png b/Phranque/awesome/themes/vertex/icons/bat-040-charging.png
new file mode 100644
index 00000000..5d20c111
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/bat-040-charging.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/bat-040.png b/Phranque/awesome/themes/vertex/icons/bat-040.png
new file mode 100644
index 00000000..ad378bfe
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/bat-040.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/bat-060-charging.png b/Phranque/awesome/themes/vertex/icons/bat-060-charging.png
new file mode 100644
index 00000000..ea7c2938
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/bat-060-charging.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/bat-060.png b/Phranque/awesome/themes/vertex/icons/bat-060.png
new file mode 100644
index 00000000..5883885f
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/bat-060.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/bat-080-charging.png b/Phranque/awesome/themes/vertex/icons/bat-080-charging.png
new file mode 100644
index 00000000..b6330449
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/bat-080-charging.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/bat-080.png b/Phranque/awesome/themes/vertex/icons/bat-080.png
new file mode 100644
index 00000000..d3fb9712
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/bat-080.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/bat-100-charging.png b/Phranque/awesome/themes/vertex/icons/bat-100-charging.png
new file mode 100644
index 00000000..f2d74194
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/bat-100-charging.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/bat-100.png b/Phranque/awesome/themes/vertex/icons/bat-100.png
new file mode 100644
index 00000000..5852e69e
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/bat-100.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/bat-charged.png b/Phranque/awesome/themes/vertex/icons/bat-charged.png
new file mode 100644
index 00000000..bb6b657b
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/bat-charged.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/ethernet-connected.png b/Phranque/awesome/themes/vertex/icons/ethernet-connected.png
new file mode 100644
index 00000000..6894931d
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/ethernet-connected.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/ethernet-disconnected.png b/Phranque/awesome/themes/vertex/icons/ethernet-disconnected.png
new file mode 100644
index 00000000..cfd73bbb
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/ethernet-disconnected.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/panel.png b/Phranque/awesome/themes/vertex/icons/panel.png
new file mode 100644
index 00000000..2f429e5d
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/panel.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/volume-high.png b/Phranque/awesome/themes/vertex/icons/volume-high.png
new file mode 100644
index 00000000..8fc8279f
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/volume-high.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/volume-low.png b/Phranque/awesome/themes/vertex/icons/volume-low.png
new file mode 100644
index 00000000..7a0682fe
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/volume-low.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/volume-medium.png b/Phranque/awesome/themes/vertex/icons/volume-medium.png
new file mode 100644
index 00000000..2b5c888f
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/volume-medium.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/volume-muted-blocked.png b/Phranque/awesome/themes/vertex/icons/volume-muted-blocked.png
new file mode 100644
index 00000000..6dea15b8
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/volume-muted-blocked.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/volume-muted.png b/Phranque/awesome/themes/vertex/icons/volume-muted.png
new file mode 100644
index 00000000..7d08ac99
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/volume-muted.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/volume-off.png b/Phranque/awesome/themes/vertex/icons/volume-off.png
new file mode 100644
index 00000000..8a6d8c56
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/volume-off.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/wireless-disconnected.png b/Phranque/awesome/themes/vertex/icons/wireless-disconnected.png
new file mode 100644
index 00000000..f98d8b4a
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/wireless-disconnected.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/wireless-full.png b/Phranque/awesome/themes/vertex/icons/wireless-full.png
new file mode 100644
index 00000000..3f83dec8
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/wireless-full.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/wireless-high.png b/Phranque/awesome/themes/vertex/icons/wireless-high.png
new file mode 100644
index 00000000..b3622ec2
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/wireless-high.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/wireless-low.png b/Phranque/awesome/themes/vertex/icons/wireless-low.png
new file mode 100644
index 00000000..3f1e7400
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/wireless-low.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/wireless-medium.png b/Phranque/awesome/themes/vertex/icons/wireless-medium.png
new file mode 100644
index 00000000..6bd0ec53
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/wireless-medium.png differ
diff --git a/Phranque/awesome/themes/vertex/icons/wireless-none.png b/Phranque/awesome/themes/vertex/icons/wireless-none.png
new file mode 100644
index 00000000..e47b4786
Binary files /dev/null and b/Phranque/awesome/themes/vertex/icons/wireless-none.png differ
diff --git a/Phranque/awesome/themes/vertex/theme.lua b/Phranque/awesome/themes/vertex/theme.lua
new file mode 100644
index 00000000..50caa726
--- /dev/null
+++ b/Phranque/awesome/themes/vertex/theme.lua
@@ -0,0 +1,467 @@
+--[[
+
+ Vertex Awesome WM theme
+ github.com/lcpz
+
+--]]
+
+local gears = require("gears")
+local lain = require("lain")
+local awful = require("awful")
+local wibox = require("wibox")
+
+local math, string, tag, tonumber, type, os = math, string, tag, tonumber, type, os
+local my_table = awful.util.table or gears.table -- 4.{0,1} compatibility
+
+local theme = {}
+theme.default_dir = require("awful.util").get_themes_dir() .. "default"
+theme.icon_dir = os.getenv("HOME") .. "/.config/awesome/themes/vertex/icons"
+theme.wallpaper = os.getenv("HOME") .. "/.config/awesome/themes/vertex/wall.png"
+theme.font = "Roboto Bold 10"
+theme.taglist_font = "FontAwesome 17"
+theme.fg_normal = "#FFFFFF"
+theme.fg_focus = "#6A95EB"
+theme.bg_focus = "#303030"
+theme.bg_focus2 = "#3762B8"
+theme.bg_normal = "#242424"
+theme.fg_urgent = "#CC9393"
+theme.bg_urgent = "#006B8E"
+theme.border_width = 4
+theme.border_normal = "#252525"
+theme.border_focus = "#7CA2EE"
+theme.tooltip_border_color = theme.fg_focus
+theme.tooltip_border_width = theme.border_width
+theme.menu_height = 24
+theme.menu_width = 140
+theme.awesome_icon = theme.icon_dir .. "/awesome.png"
+theme.taglist_squares_sel = gears.surface.load_from_shape(3, 30, gears.shape.rectangle, theme.fg_focus)
+theme.taglist_squares_unsel = gears.surface.load_from_shape(3, 30, gears.shape.rectangle, theme.bg_focus2)
+theme.panelbg = theme.icon_dir .. "/panel.png"
+theme.bat000charging = theme.icon_dir .. "/bat-000-charging.png"
+theme.bat000 = theme.icon_dir .. "/bat-000.png"
+theme.bat020charging = theme.icon_dir .. "/bat-020-charging.png"
+theme.bat020 = theme.icon_dir .. "/bat-020.png"
+theme.bat040charging = theme.icon_dir .. "/bat-040-charging.png"
+theme.bat040 = theme.icon_dir .. "/bat-040.png"
+theme.bat060charging = theme.icon_dir .. "/bat-060-charging.png"
+theme.bat060 = theme.icon_dir .. "/bat-060.png"
+theme.bat080charging = theme.icon_dir .. "/bat-080-charging.png"
+theme.bat080 = theme.icon_dir .. "/bat-080.png"
+theme.bat100charging = theme.icon_dir .. "/bat-100-charging.png"
+theme.bat100 = theme.icon_dir .. "/bat-100.png"
+theme.batcharged = theme.icon_dir .. "/bat-charged.png"
+theme.ethon = theme.icon_dir .. "/ethernet-connected.png"
+theme.ethoff = theme.icon_dir .. "/ethernet-disconnected.png"
+theme.volhigh = theme.icon_dir .. "/volume-high.png"
+theme.vollow = theme.icon_dir .. "/volume-low.png"
+theme.volmed = theme.icon_dir .. "/volume-medium.png"
+theme.volmutedblocked = theme.icon_dir .. "/volume-muted-blocked.png"
+theme.volmuted = theme.icon_dir .. "/volume-muted.png"
+theme.voloff = theme.icon_dir .. "/volume-off.png"
+theme.wifidisc = theme.icon_dir .. "/wireless-disconnected.png"
+theme.wififull = theme.icon_dir .. "/wireless-full.png"
+theme.wifihigh = theme.icon_dir .. "/wireless-high.png"
+theme.wifilow = theme.icon_dir .. "/wireless-low.png"
+theme.wifimed = theme.icon_dir .. "/wireless-medium.png"
+theme.wifinone = theme.icon_dir .. "/wireless-none.png"
+theme.layout_fairh = theme.default_dir.."/layouts/fairhw.png"
+theme.layout_fairv = theme.default_dir.."/layouts/fairvw.png"
+theme.layout_floating = theme.default_dir.."/layouts/floatingw.png"
+theme.layout_magnifier = theme.default_dir.."/layouts/magnifierw.png"
+theme.layout_max = theme.default_dir.."/layouts/maxw.png"
+theme.layout_fullscreen = theme.default_dir.."/layouts/fullscreenw.png"
+theme.layout_tilebottom = theme.default_dir.."/layouts/tilebottomw.png"
+theme.layout_tileleft = theme.default_dir.."/layouts/tileleftw.png"
+theme.layout_tile = theme.default_dir.."/layouts/tilew.png"
+theme.layout_tiletop = theme.default_dir.."/layouts/tiletopw.png"
+theme.layout_spiral = theme.default_dir.."/layouts/spiralw.png"
+theme.layout_dwindle = theme.default_dir.."/layouts/dwindlew.png"
+theme.layout_cornernw = theme.default_dir.."/layouts/cornernww.png"
+theme.layout_cornerne = theme.default_dir.."/layouts/cornernew.png"
+theme.layout_cornersw = theme.default_dir.."/layouts/cornersww.png"
+theme.layout_cornerse = theme.default_dir.."/layouts/cornersew.png"
+theme.tasklist_plain_task_name = true
+theme.tasklist_disable_icon = true
+theme.useless_gap = 10
+theme.titlebar_close_button_normal = theme.default_dir.."/titlebar/close_normal.png"
+theme.titlebar_close_button_focus = theme.default_dir.."/titlebar/close_focus.png"
+theme.titlebar_minimize_button_normal = theme.default_dir.."/titlebar/minimize_normal.png"
+theme.titlebar_minimize_button_focus = theme.default_dir.."/titlebar/minimize_focus.png"
+theme.titlebar_ontop_button_normal_inactive = theme.default_dir.."/titlebar/ontop_normal_inactive.png"
+theme.titlebar_ontop_button_focus_inactive = theme.default_dir.."/titlebar/ontop_focus_inactive.png"
+theme.titlebar_ontop_button_normal_active = theme.default_dir.."/titlebar/ontop_normal_active.png"
+theme.titlebar_ontop_button_focus_active = theme.default_dir.."/titlebar/ontop_focus_active.png"
+theme.titlebar_sticky_button_normal_inactive = theme.default_dir.."/titlebar/sticky_normal_inactive.png"
+theme.titlebar_sticky_button_focus_inactive = theme.default_dir.."/titlebar/sticky_focus_inactive.png"
+theme.titlebar_sticky_button_normal_active = theme.default_dir.."/titlebar/sticky_normal_active.png"
+theme.titlebar_sticky_button_focus_active = theme.default_dir.."/titlebar/sticky_focus_active.png"
+theme.titlebar_floating_button_normal_inactive = theme.default_dir.."/titlebar/floating_normal_inactive.png"
+theme.titlebar_floating_button_focus_inactive = theme.default_dir.."/titlebar/floating_focus_inactive.png"
+theme.titlebar_floating_button_normal_active = theme.default_dir.."/titlebar/floating_normal_active.png"
+theme.titlebar_floating_button_focus_active = theme.default_dir.."/titlebar/floating_focus_active.png"
+theme.titlebar_maximized_button_normal_inactive = theme.default_dir.."/titlebar/maximized_normal_inactive.png"
+theme.titlebar_maximized_button_focus_inactive = theme.default_dir.."/titlebar/maximized_focus_inactive.png"
+theme.titlebar_maximized_button_normal_active = theme.default_dir.."/titlebar/maximized_normal_active.png"
+theme.titlebar_maximized_button_focus_active = theme.default_dir.."/titlebar/maximized_focus_active.png"
+
+-- http://fontawesome.io/cheatsheet
+awful.util.tagnames = { "", "", "", "", "", "", "", "" }
+
+local markup = lain.util.markup
+
+-- Clock
+--os.setlocale(os.getenv("LANG")) -- to localize the clock
+local mytextclock = wibox.widget.textclock(markup("#FFFFFF", "%a %d %b, %H:%M"))
+mytextclock.font = theme.font
+theme.cal = lain.widget.cal({
+ attach_to = { mytextclock },
+ notification_preset = {
+ fg = "#FFFFFF",
+ bg = theme.bg_normal,
+ position = "top_middle",
+ font = "Monospace 10"
+ }
+})
+
+-- Battery
+local baticon = wibox.widget.imagebox(theme.bat000)
+local battooltip = awful.tooltip({
+ objects = { baticon },
+ margin_leftright = 15,
+ margin_topbottom = 12
+})
+battooltip.wibox.fg = theme.fg_normal
+battooltip.textbox.font = theme.font
+battooltip.timeout = 0
+battooltip:set_shape(function(cr, width, height)
+ gears.shape.infobubble(cr, width, height, corner_radius, arrow_size, width - 35)
+end)
+local bat = lain.widget.bat({
+ settings = function()
+ local index, perc = "bat", tonumber(bat_now.perc) or 0
+
+ if perc <= 7 then
+ index = index .. "000"
+ elseif perc <= 20 then
+ index = index .. "020"
+ elseif perc <= 40 then
+ index = index .. "040"
+ elseif perc <= 60 then
+ index = index .. "060"
+ elseif perc <= 80 then
+ index = index .. "080"
+ elseif perc <= 100 then
+ index = index .. "100"
+ end
+
+ if bat_now.ac_status == 1 then
+ index = index .. "charging"
+ end
+
+ baticon:set_image(theme[index])
+ battooltip:set_markup(string.format("\n%s%%, %s", perc, bat_now.time))
+ end
+})
+
+-- MPD
+theme.mpd = lain.widget.mpd({
+ music_dir = "/mnt/storage/Downloads/Music",
+ settings = function()
+ if mpd_now.state == "play" then
+ title = mpd_now.title
+ artist = " " .. mpd_now.artist .. " "
+ elseif mpd_now.state == "pause" then
+ title = "mpd "
+ artist = "paused "
+ else
+ title = ""
+ artist = ""
+ end
+
+ widget:set_markup(markup.font(theme.font, title .. markup(theme.fg_focus, artist)))
+ end
+})
+
+-- ALSA volume
+local volicon = wibox.widget.imagebox()
+theme.volume = lain.widget.alsabar({
+ --togglechannel = "IEC958,3",
+ notification_preset = { font = "Monospace 12", fg = theme.fg_normal },
+ settings = function()
+ local index, perc = "", tonumber(volume_now.level) or 0
+
+ if volume_now.status == "off" then
+ index = "volmutedblocked"
+ else
+ if perc <= 5 then
+ index = "volmuted"
+ elseif perc <= 25 then
+ index = "vollow"
+ elseif perc <= 75 then
+ index = "volmed"
+ else
+ index = "volhigh"
+ end
+ end
+
+ volicon:set_image(theme[index])
+ end
+})
+volicon:buttons(my_table.join (
+ awful.button({}, 1, function()
+ awful.spawn(string.format("%s -e alsamixer", awful.util.terminal))
+ end),
+ awful.button({}, 2, function()
+ os.execute(string.format("%s set %s 100%%", theme.volume.cmd, theme.volume.channel))
+ theme.volume.notify()
+ end),
+ awful.button({}, 3, function()
+ os.execute(string.format("%s set %s toggle", theme.volume.cmd, theme.volume.togglechannel or theme.volume.channel))
+ theme.volume.notify()
+ end),
+ awful.button({}, 4, function()
+ os.execute(string.format("%s set %s 1%%+", theme.volume.cmd, theme.volume.channel))
+ theme.volume.notify()
+ end),
+ awful.button({}, 5, function()
+ os.execute(string.format("%s set %s 1%%-", theme.volume.cmd, theme.volume.channel))
+ theme.volume.notify()
+ end)
+))
+
+-- Wifi carrier and signal strength
+local wificon = wibox.widget.imagebox(theme.wifidisc)
+local wifitooltip = awful.tooltip({
+ objects = { wificon },
+ margin_leftright = 15,
+ margin_topbottom = 15
+})
+wifitooltip.wibox.fg = theme.fg_normal
+wifitooltip.textbox.font = theme.font
+wifitooltip.timeout = 0
+wifitooltip:set_shape(function(cr, width, height)
+ gears.shape.infobubble(cr, width, height, corner_radius, arrow_size, width - 120)
+end)
+local mywifisig = awful.widget.watch(
+ { awful.util.shell, "-c", "awk 'NR==3 {printf(\"%d-%.0f\\n\",$2, $3*10/7)}' /proc/net/wireless; iw dev wlan0 link" },
+ 2,
+ function(widget, stdout)
+ local carrier, perc = stdout:match("(%d)-(%d+)")
+ local tiptext = stdout:gsub("(%d)-(%d+)", ""):gsub("%s+$", "")
+ perc = tonumber(perc)
+
+ if carrier == "1" or not perc then
+ wificon:set_image(theme.wifidisc)
+ wifitooltip:set_markup("No carrier")
+ else
+ if perc <= 5 then
+ wificon:set_image(theme.wifinone)
+ elseif perc <= 25 then
+ wificon:set_image(theme.wifilow)
+ elseif perc <= 50 then
+ wificon:set_image(theme.wifimed)
+ elseif perc <= 75 then
+ wificon:set_image(theme.wifihigh)
+ else
+ wificon:set_image(theme.wififull)
+ end
+ wifitooltip:set_markup(tiptext)
+ end
+ end
+)
+wificon:connect_signal("button::press", function() awful.spawn(string.format("%s -e wavemon", awful.util.terminal)) end)
+
+-- Weather
+theme.weather = lain.widget.weather({
+ city_id = 2643743, -- placeholder (London)
+ notification_preset = { font = "Monospace 10" },
+ settings = function()
+ units = math.floor(weather_now["main"]["temp"])
+ widget:set_markup(" " .. markup.font(theme.font, units .. "°C") .. " ")
+ end
+})
+
+-- Launcher
+local mylauncher = awful.widget.button({image = theme.awesome_icon})
+mylauncher:connect_signal("button::press", function() awful.util.mymainmenu:toggle() end)
+
+-- Separators
+local space = wibox.widget.textbox(" ")
+local rspace1 = wibox.widget.textbox()
+local rspace0 = wibox.widget.textbox()
+local rspace2 = wibox.widget.textbox()
+local rspace3 = wibox.widget.textbox()
+local tspace1 = wibox.widget.textbox()
+tspace1.forced_width = 18
+rspace1.forced_width = 16
+rspace0.forced_width = 18
+rspace2.forced_width = 19
+rspace3.forced_width = 21
+
+local lspace1 = wibox.widget.textbox()
+local lspace2 = wibox.widget.textbox()
+local lspace3 = wibox.widget.textbox()
+lspace1.forced_height = 18
+lspace2.forced_height = 10
+lspace3.forced_height = 16
+
+local barcolor = gears.color({
+ type = "linear",
+ from = { 0, 46 },
+ to = { 46, 46 },
+ stops = { {0, theme.bg_focus}, {0.9, theme.bg_focus2} }
+})
+
+local barcolor2 = gears.color({
+ type = "linear",
+ from = { 0, 46 },
+ to = { 46, 46 },
+ stops = { {0, "#323232"}, {1, theme.bg_normal} }
+})
+
+local dockshape = function(cr, width, height)
+ gears.shape.partially_rounded_rect(cr, width, height, false, true, true, false, 6)
+end
+
+function theme.vertical_wibox(s)
+ -- Create the vertical wibox
+ s.dockheight = (35 * s.workarea.height)/100
+
+ s.myleftwibox = wibox({ screen = s, x=0, y=s.workarea.height/2 - s.dockheight/2, width = 6, height = s.dockheight, fg = theme.fg_normal, bg = barcolor2, ontop = true, visible = true, type = "dock" })
+
+ if s.index > 1 and s.myleftwibox.y == 0 then
+ s.myleftwibox.y = screen[1].myleftwibox.y
+ end
+
+ -- Add widgets to the vertical wibox
+ s.myleftwibox:setup {
+ layout = wibox.layout.align.vertical,
+ {
+ layout = wibox.layout.fixed.vertical,
+ lspace1,
+ s.mytaglist,
+ lspace2,
+ s.layoutb,
+ wibox.container.margin(mylauncher, 5, 8, 13, 0),
+ },
+ }
+
+ -- Add toggling functionalities
+ s.docktimer = gears.timer{ timeout = 2 }
+ s.docktimer:connect_signal("timeout", function()
+ local s = awful.screen.focused()
+ s.myleftwibox.width = 9
+ s.layoutb.visible = false
+ mylauncher.visible = false
+ if s.docktimer.started then
+ s.docktimer:stop()
+ end
+ end)
+ tag.connect_signal("property::selected", function(t)
+ local s = t.screen or awful.screen.focused()
+ s.myleftwibox.width = 38
+ s.layoutb.visible = true
+ mylauncher.visible = true
+ gears.surface.apply_shape_bounding(s.myleftwibox, dockshape)
+ if not s.docktimer.started then
+ s.docktimer:start()
+ end
+ end)
+
+ s.myleftwibox:connect_signal("mouse::leave", function()
+ local s = awful.screen.focused()
+ s.myleftwibox.width = 9
+ s.layoutb.visible = false
+ mylauncher.visible = false
+ end)
+
+ s.myleftwibox:connect_signal("mouse::enter", function()
+ local s = awful.screen.focused()
+ s.myleftwibox.width = 38
+ s.layoutb.visible = true
+ mylauncher.visible = true
+ gears.surface.apply_shape_bounding(s.myleftwibox, dockshape)
+ end)
+end
+
+
+function theme.at_screen_connect(s)
+ -- Quake application
+ s.quake = lain.util.quake({ app = awful.util.terminal, border = theme.border_width })
+
+ -- If wallpaper is a function, call it with the screen
+ local wallpaper = theme.wallpaper
+ if type(wallpaper) == "function" then
+ wallpaper = wallpaper(s)
+ end
+ gears.wallpaper.maximized(wallpaper, s, true)
+
+ -- Tags
+ awful.tag(awful.util.tagnames, s, awful.layout.layouts)
+
+ -- Create a promptbox for each screen
+ s.mypromptbox = awful.widget.prompt()
+ s.mypromptbox.bg = "#00000000"
+
+ -- Create an imagebox widget which will contains an icon indicating which layout we're using.
+ -- We need one layoutbox per screen.
+ s.mylayoutbox = awful.widget.layoutbox(s)
+ s.mylayoutbox:buttons(my_table.join(
+ awful.button({}, 1, function () awful.layout.inc( 1) end),
+ awful.button({}, 2, function () awful.layout.set( awful.layout.layouts[1] ) end),
+ awful.button({}, 3, function () awful.layout.inc(-1) end),
+ awful.button({}, 4, function () awful.layout.inc( 1) end),
+ awful.button({}, 5, function () awful.layout.inc(-1) end)))
+ s.layoutb = wibox.container.margin(s.mylayoutbox, 8, 11, 3, 3)
+
+ -- Create a taglist widget
+ s.mytaglist = awful.widget.taglist(s, awful.widget.taglist.filter.all, awful.util.taglist_buttons, {
+ font = theme.taglist_font,
+ shape = gears.shape.rectangle,
+ spacing = 10,
+ square_unsel = theme.square_unsel,
+ bg_focus = barcolor
+ }, nil, wibox.layout.fixed.vertical())
+
+ -- Create a tasklist widget
+ s.mytasklist = awful.widget.tasklist(s, awful.widget.tasklist.filter.focused, awful.util.tasklist_buttons, { bg_focus = "#00000000" })
+
+ -- Create the wibox
+ s.mywibox = awful.wibar({ position = "top", screen = s, height = 25, bg = gears.color.create_png_pattern(theme.panelbg) })
+
+ local wiboxlayout = wibox.layout.align.horizontal()
+ wiboxlayout.expand = "none"
+ -- Add widgets to the wibox
+ s.mywibox:setup {
+ layout = wiboxlayout,
+ { -- Left widgets
+ layout = wibox.layout.fixed.horizontal,
+ s.mypromptbox,
+ tspace1,
+ s.mytasklist,
+ },
+ { -- Middle widgets
+ layout = wibox.layout.fixed.horizontal,
+ mytextclock,
+ },
+ { -- Right widgets
+ layout = wibox.layout.fixed.horizontal,
+ wibox.widget { nil, nil, theme.mpd.widget, layout = wibox.layout.align.horizontal },
+ rspace0,
+ theme.weather.icon,
+ theme.weather.widget,
+ rspace1,
+ wificon,
+ rspace0,
+ volicon,
+ rspace2,
+ baticon,
+ rspace3,
+ wibox.widget.systray(),
+ },
+ }
+
+ gears.timer.delayed_call(theme.vertical_wibox, s)
+end
+
+return theme
diff --git a/Phranque/awesome/themes/vertex/wall.png b/Phranque/awesome/themes/vertex/wall.png
new file mode 100644
index 00000000..863b1bfe
Binary files /dev/null and b/Phranque/awesome/themes/vertex/wall.png differ
diff --git a/Phranque/awesome/xrandr.lua b/Phranque/awesome/xrandr.lua
new file mode 100644
index 00000000..bd90c19a
--- /dev/null
+++ b/Phranque/awesome/xrandr.lua
@@ -0,0 +1,135 @@
+--- Separating Multiple Monitor functions as a separeted module (taken from awesome wiki)
+
+local awful = require("awful")
+local naughty = require("naughty")
+
+-- A path to a fancy icon
+local icon_path = ""
+
+-- Get active outputs
+local function outputs()
+ local outputs = {}
+ local xrandr = io.popen("xrandr -q --current")
+
+ if xrandr then
+ for line in xrandr:lines() do
+ local output = line:match("^([%w-]+) connected ")
+ if output then
+ outputs[#outputs + 1] = output
+ end
+ end
+ xrandr:close()
+ end
+
+ return outputs
+end
+
+local function arrange(out)
+ -- We need to enumerate all permutations of horizontal outputs.
+
+ local choices = {}
+ local previous = { {} }
+ for i = 1, #out do
+ -- Find all permutation of length `i`: we take the permutation
+ -- of length `i-1` and for each of them, we create new
+ -- permutations by adding each output at the end of it if it is
+ -- not already present.
+ local new = {}
+ for _, p in pairs(previous) do
+ for _, o in pairs(out) do
+ if not awful.util.table.hasitem(p, o) then
+ new[#new + 1] = awful.util.table.join(p, {o})
+ end
+ end
+ end
+ choices = awful.util.table.join(choices, new)
+ previous = new
+ end
+
+ return choices
+end
+
+-- Build available choices
+local function menu()
+ local menu = {}
+ local out = outputs()
+ local choices = arrange(out)
+
+ for _, choice in pairs(choices) do
+ local cmd = "xrandr"
+ -- Enabled outputs
+ for i, o in pairs(choice) do
+ cmd = cmd .. " --output " .. o .. " --auto"
+ if i > 1 then
+ cmd = cmd .. " --right-of " .. choice[i-1]
+ end
+ end
+ -- Disabled outputs
+ for _, o in pairs(out) do
+ if not awful.util.table.hasitem(choice, o) then
+ cmd = cmd .. " --output " .. o .. " --off"
+ end
+ end
+
+ local label = ""
+ if #choice == 1 then
+ label = 'Only ' .. choice[1] .. ''
+ else
+ for i, o in pairs(choice) do
+ if i > 1 then label = label .. " + " end
+ label = label .. '' .. o .. ''
+ end
+ end
+
+ menu[#menu + 1] = { label, cmd }
+ end
+
+ return menu
+end
+
+-- Display xrandr notifications from choices
+local state = { cid = nil }
+
+local function naughty_destroy_callback(reason)
+ if reason == naughty.notificationClosedReason.expired or
+ reason == naughty.notificationClosedReason.dismissedByUser then
+ local action = state.index and state.menu[state.index - 1][2]
+ if action then
+ awful.util.spawn(action, false)
+ state.index = nil
+ end
+ end
+end
+
+local function xrandr()
+ -- Build the list of choices
+ if not state.index then
+ state.menu = menu()
+ state.index = 1
+ end
+
+ -- Select one and display the appropriate notification
+ local label, action
+ local next = state.menu[state.index]
+ state.index = state.index + 1
+
+ if not next then
+ label = "Keep the current configuration"
+ state.index = nil
+ else
+ label, action = unpack(next)
+ end
+ state.cid = naughty.notify({ text = label,
+ icon = icon_path,
+ timeout = 4,
+ screen = mouse.screen,
+ replaces_id = state.cid,
+ destroy = naughty_destroy_callback}).id
+end
+
+return {
+ outputs = outputs,
+ arrange = arrange,
+ menu = menu,
+ xrandr = xrandr
+}
diff --git a/Phranque/compton.conf b/Phranque/compton.conf
deleted file mode 100644
index ec67bfbf..00000000
--- a/Phranque/compton.conf
+++ /dev/null
@@ -1,238 +0,0 @@
-# Thank you code_nomad: http://9m.no/ꪯ鵞
-
-#################################
-#
-# Backend
-#
-#################################
-
-# Backend to use: "xrender" or "glx".
-# GLX backend is typically much faster but depends on a sane driver.
-backend = "xrender";
-
-#################################
-#
-# GLX backend
-#
-#################################
-
-glx-no-stencil = true;
-
-# GLX backend: Copy unmodified regions from front buffer instead of redrawing them all.
-# My tests with nvidia-drivers show a 10% decrease in performance when the whole screen is modified,
-# but a 20% increase when only 1/4 is.
-# My tests on nouveau show terrible slowdown.
-# Useful with --glx-swap-method, as well.
-glx-copy-from-front = true;
-
-# GLX backend: Use MESA_copy_sub_buffer to do partial screen update.
-# My tests on nouveau shows a 200% performance boost when only 1/4 of the screen is updated.
-# May break VSync and is not available on some drivers.
-# Overrides --glx-copy-from-front.
-# glx-use-copysubbuffermesa = true;
-
-# GLX backend: Avoid rebinding pixmap on window damage.
-# Probably could improve performance on rapid window content changes, but is known to break things on some drivers (LLVMpipe).
-# Recommended if it works.
-glx-no-rebind-pixmap = true;
-
-
-# GLX backend: GLX buffer swap method we assume.
-# Could be undefined (0), copy (1), exchange (2), 3-6, or buffer-age (-1).
-# undefined is the slowest and the safest, and the default value.
-# copy is fastest, but may fail on some drivers,
-# 2-6 are gradually slower but safer (6 is still faster than 0).
-# Usually, double buffer means 2, triple buffer means 3.
-# buffer-age means auto-detect using GLX_EXT_buffer_age, supported by some drivers.
-# Useless with --glx-use-copysubbuffermesa.
-# Partially breaks --resize-damage.
-# Defaults to undefined.
-glx-swap-method = "undefined";
-
-#################################
-#
-# Shadows
-#
-#################################
-
-# Enabled client-side shadows on windows.
-shadow = true;
-# Don't draw shadows on DND windows.
-no-dnd-shadow = true;
-# Avoid drawing shadows on dock/panel windows.
-no-dock-shadow = true;
-# Zero the part of the shadow's mask behind the window. Fix some weirdness with ARGB windows.
-clear-shadow = true;
-# The blur radius for shadows. (default 12)
-shadow-radius = 5;
-# The left offset for shadows. (default -15)
-shadow-offset-x = -5;
-# The top offset for shadows. (default -15)
-shadow-offset-y = -5;
-# The translucency for shadows. (default .75)
-shadow-opacity = 0.5;
-
-# Set if you want different colour shadows
-# shadow-red = 0.0;
-# shadow-green = 0.0;
-# shadow-blue = 0.0;
-
-# The shadow exclude options are helpful if you have shadows enabled. Due to the way compton draws its shadows, certain applications will have visual
-#glitches
-# (most applications are fine, only apps that do weird things with xshapes or argb are affected).
-# This list includes all the affected apps I found in my testing. The "! name~=''" part excludes shadows on any "Unknown" windows, this prevents a
-#visual glitch with the XFWM alt tab switcher.
-shadow-exclude = [
- "! name~=''",
- "name = 'Notification'",
- "name = 'Plank'",
- "name = 'Docky'",
- "name = 'Kupfer'",
- "name = 'xfce4-notifyd'",
- "name *= 'VLC'",
- "name *= 'compton'",
- "name *= 'Chromium'",
- "name *= 'Chrome'",
- "class_g = 'Conky'",
- "class_g = 'Kupfer'",
- "class_g = 'Synapse'",
- "class_g ?= 'Notify-osd'",
- "class_g ?= 'Cairo-dock'",
- "class_g ?= 'Xfce4-notifyd'",
- "class_g ?= 'Xfce4-power-manager'",
- "_GTK_FRAME_EXTENTS@:c"
-];
-# Avoid drawing shadow on all shaped windows (see also: --detect-rounded-corners)
-shadow-ignore-shaped = false;
-
-#################################
-#
-# Opacity
-#
-#################################
-
-menu-opacity = 1;
-inactive-opacity = 0.75;
-active-opacity = 1;
-frame-opacity = 1;
-inactive-opacity-override = false;
-alpha-step = 0.06;
-
-# Dim inactive windows. (0.0 - 1.0)
-# inactive-dim = 0.2;
-# Do not let dimness adjust based on window opacity.
-# inactive-dim-fixed = true;
-# Blur background of transparent windows. Bad performance with X Render backend. GLX backend is preferred.
-# blur-background = true;
-# Blur background of opaque windows with transparent frames as well.
-# blur-background-frame = true;
-# Do not let blur radius adjust based on window opacity.
-blur-background-fixed = false;
-blur-background-exclude = [
- "window_type = 'dock'",
- "window_type = 'desktop'"
-];
-
-#################################
-#
-# Fading
-#
-#################################
-
-# Fade windows during opacity changes.
-fading = true;
-# The time between steps in a fade in milliseconds. (default 10).
-fade-delta = 4;
-# Opacity change between steps while fading in. (default 0.028).
-fade-in-step = 0.03;
-# Opacity change between steps while fading out. (default 0.03).
-fade-out-step = 0.03;
-# Fade windows in/out when opening/closing
-# no-fading-openclose = true;
-
-# Specify a list of conditions of windows that should not be faded.
-fade-exclude = [ ];
-
-#################################
-#
-# Other
-#
-#################################
-
-# Try to detect WM windows and mark them as active.
-mark-wmwin-focused = true;
-# Mark all non-WM but override-redirect windows active (e.g. menus).
-mark-ovredir-focused = true;
-# Use EWMH _NET_WM_ACTIVE_WINDOW to determine which window is focused instead of using FocusIn/Out events.
-# Usually more reliable but depends on a EWMH-compliant WM.
-use-ewmh-active-win = true;
-# Detect rounded corners and treat them as rectangular when --shadow-ignore-shaped is on.
-detect-rounded-corners = true;
-
-# Detect _NET_WM_OPACITY on client windows, useful for window managers not passing _NET_WM_OPACITY of client windows to frame windows.
-# This prevents opacity being ignored for some apps.
-# For example without this enabled my xfce4-notifyd is 100% opacity no matter what.
-detect-client-opacity = true;
-
-# Specify refresh rate of the screen.
-# If not specified or 0, compton will try detecting this with X RandR extension.
-refresh-rate = 0;
-
-# Set VSync method. VSync methods currently available:
-# none: No VSync
-# drm: VSync with DRM_IOCTL_WAIT_VBLANK. May only work on some drivers.
-# opengl: Try to VSync with SGI_video_sync OpenGL extension. Only work on some drivers.
-# opengl-oml: Try to VSync with OML_sync_control OpenGL extension. Only work on some drivers.
-# opengl-swc: Try to VSync with SGI_swap_control OpenGL extension. Only work on some drivers. Works only with GLX backend. Known to be most effective
-#on many drivers. Does not actually control paint timing, only buffer swap is affected, so it doesn’t have the effect of --sw-opti unlike other
-#methods.
-#Experimental.
-# opengl-mswc: Try to VSync with MESA_swap_control OpenGL extension. Basically the same as opengl-swc above, except the extension we use.
-# (Note some VSync methods may not be enabled at compile time.)
-vsync = "drm";
-
-# Enable DBE painting mode, intended to use with VSync to (hopefully) eliminate tearing.
-# Reported to have no effect, though.
-dbe = true;
-# Painting on X Composite overlay window. Recommended.
-paint-on-overlay = true;
-
-# Limit compton to repaint at most once every 1 / refresh_rate second to boost performance.
-# This should not be used with --vsync drm/opengl/opengl-oml as they essentially does --sw-opti's job already,
-# unless you wish to specify a lower refresh rate than the actual value.
-sw-opti = true;
-
-# Unredirect all windows if a full-screen opaque window is detected, to maximize performance for full-screen windows, like games.
-# Known to cause flickering when redirecting/unredirecting windows.
-# paint-on-overlay may make the flickering less obvious.
-unredir-if-possible = true;
-
-# Specify a list of conditions of windows that should always be considered focused.
-focus-exclude = [ ];
-
-# Use WM_TRANSIENT_FOR to group windows, and consider windows in the same group focused at the same time.
-detect-transient = true;
-# Use WM_CLIENT_LEADER to group windows, and consider windows in the same group focused at the same time.
-# WM_TRANSIENT_FOR has higher priority if --detect-transient is enabled, too.
-detect-client-leader = true;
-
-#################################
-#
-# Window type settings
-#
-#################################
-
-wintypes:
-{
- tooltip =
- {
- # fade: Fade the particular type of windows.
- fade = true;
- # shadow: Give those windows shadow
- shadow = false;
- # opacity: Default opacity for the type of windows.
- opacity = 0.85;
- # focus: Whether to always consider windows of this type focused.
- focus = true;
- };
-};
diff --git a/Fearnley/compton.conf b/Phranque/compton/compton.conf
similarity index 85%
rename from Fearnley/compton.conf
rename to Phranque/compton/compton.conf
index ec67bfbf..15a520ea 100644
--- a/Fearnley/compton.conf
+++ b/Phranque/compton/compton.conf
@@ -1,238 +1,244 @@
-# Thank you code_nomad: http://9m.no/ꪯ鵞
-
-#################################
-#
-# Backend
-#
-#################################
-
-# Backend to use: "xrender" or "glx".
-# GLX backend is typically much faster but depends on a sane driver.
-backend = "xrender";
-
-#################################
-#
-# GLX backend
-#
-#################################
-
-glx-no-stencil = true;
-
-# GLX backend: Copy unmodified regions from front buffer instead of redrawing them all.
-# My tests with nvidia-drivers show a 10% decrease in performance when the whole screen is modified,
-# but a 20% increase when only 1/4 is.
-# My tests on nouveau show terrible slowdown.
-# Useful with --glx-swap-method, as well.
-glx-copy-from-front = true;
-
-# GLX backend: Use MESA_copy_sub_buffer to do partial screen update.
-# My tests on nouveau shows a 200% performance boost when only 1/4 of the screen is updated.
-# May break VSync and is not available on some drivers.
-# Overrides --glx-copy-from-front.
-# glx-use-copysubbuffermesa = true;
-
-# GLX backend: Avoid rebinding pixmap on window damage.
-# Probably could improve performance on rapid window content changes, but is known to break things on some drivers (LLVMpipe).
-# Recommended if it works.
-glx-no-rebind-pixmap = true;
-
-
-# GLX backend: GLX buffer swap method we assume.
-# Could be undefined (0), copy (1), exchange (2), 3-6, or buffer-age (-1).
-# undefined is the slowest and the safest, and the default value.
-# copy is fastest, but may fail on some drivers,
-# 2-6 are gradually slower but safer (6 is still faster than 0).
-# Usually, double buffer means 2, triple buffer means 3.
-# buffer-age means auto-detect using GLX_EXT_buffer_age, supported by some drivers.
-# Useless with --glx-use-copysubbuffermesa.
-# Partially breaks --resize-damage.
-# Defaults to undefined.
-glx-swap-method = "undefined";
-
-#################################
-#
-# Shadows
-#
-#################################
-
-# Enabled client-side shadows on windows.
-shadow = true;
-# Don't draw shadows on DND windows.
-no-dnd-shadow = true;
-# Avoid drawing shadows on dock/panel windows.
-no-dock-shadow = true;
-# Zero the part of the shadow's mask behind the window. Fix some weirdness with ARGB windows.
-clear-shadow = true;
-# The blur radius for shadows. (default 12)
-shadow-radius = 5;
-# The left offset for shadows. (default -15)
-shadow-offset-x = -5;
-# The top offset for shadows. (default -15)
-shadow-offset-y = -5;
-# The translucency for shadows. (default .75)
-shadow-opacity = 0.5;
-
-# Set if you want different colour shadows
-# shadow-red = 0.0;
-# shadow-green = 0.0;
-# shadow-blue = 0.0;
-
-# The shadow exclude options are helpful if you have shadows enabled. Due to the way compton draws its shadows, certain applications will have visual
-#glitches
-# (most applications are fine, only apps that do weird things with xshapes or argb are affected).
-# This list includes all the affected apps I found in my testing. The "! name~=''" part excludes shadows on any "Unknown" windows, this prevents a
-#visual glitch with the XFWM alt tab switcher.
-shadow-exclude = [
- "! name~=''",
- "name = 'Notification'",
- "name = 'Plank'",
- "name = 'Docky'",
- "name = 'Kupfer'",
- "name = 'xfce4-notifyd'",
- "name *= 'VLC'",
- "name *= 'compton'",
- "name *= 'Chromium'",
- "name *= 'Chrome'",
- "class_g = 'Conky'",
- "class_g = 'Kupfer'",
- "class_g = 'Synapse'",
- "class_g ?= 'Notify-osd'",
- "class_g ?= 'Cairo-dock'",
- "class_g ?= 'Xfce4-notifyd'",
- "class_g ?= 'Xfce4-power-manager'",
- "_GTK_FRAME_EXTENTS@:c"
-];
-# Avoid drawing shadow on all shaped windows (see also: --detect-rounded-corners)
-shadow-ignore-shaped = false;
-
-#################################
-#
-# Opacity
-#
-#################################
-
-menu-opacity = 1;
-inactive-opacity = 0.75;
-active-opacity = 1;
-frame-opacity = 1;
-inactive-opacity-override = false;
-alpha-step = 0.06;
-
-# Dim inactive windows. (0.0 - 1.0)
-# inactive-dim = 0.2;
-# Do not let dimness adjust based on window opacity.
-# inactive-dim-fixed = true;
-# Blur background of transparent windows. Bad performance with X Render backend. GLX backend is preferred.
-# blur-background = true;
-# Blur background of opaque windows with transparent frames as well.
-# blur-background-frame = true;
-# Do not let blur radius adjust based on window opacity.
-blur-background-fixed = false;
-blur-background-exclude = [
- "window_type = 'dock'",
- "window_type = 'desktop'"
-];
-
-#################################
-#
-# Fading
-#
-#################################
-
-# Fade windows during opacity changes.
-fading = true;
-# The time between steps in a fade in milliseconds. (default 10).
-fade-delta = 4;
-# Opacity change between steps while fading in. (default 0.028).
-fade-in-step = 0.03;
-# Opacity change between steps while fading out. (default 0.03).
-fade-out-step = 0.03;
-# Fade windows in/out when opening/closing
-# no-fading-openclose = true;
-
-# Specify a list of conditions of windows that should not be faded.
-fade-exclude = [ ];
-
-#################################
-#
-# Other
-#
-#################################
-
-# Try to detect WM windows and mark them as active.
-mark-wmwin-focused = true;
-# Mark all non-WM but override-redirect windows active (e.g. menus).
-mark-ovredir-focused = true;
-# Use EWMH _NET_WM_ACTIVE_WINDOW to determine which window is focused instead of using FocusIn/Out events.
-# Usually more reliable but depends on a EWMH-compliant WM.
-use-ewmh-active-win = true;
-# Detect rounded corners and treat them as rectangular when --shadow-ignore-shaped is on.
-detect-rounded-corners = true;
-
-# Detect _NET_WM_OPACITY on client windows, useful for window managers not passing _NET_WM_OPACITY of client windows to frame windows.
-# This prevents opacity being ignored for some apps.
-# For example without this enabled my xfce4-notifyd is 100% opacity no matter what.
-detect-client-opacity = true;
-
-# Specify refresh rate of the screen.
-# If not specified or 0, compton will try detecting this with X RandR extension.
-refresh-rate = 0;
-
-# Set VSync method. VSync methods currently available:
-# none: No VSync
-# drm: VSync with DRM_IOCTL_WAIT_VBLANK. May only work on some drivers.
-# opengl: Try to VSync with SGI_video_sync OpenGL extension. Only work on some drivers.
-# opengl-oml: Try to VSync with OML_sync_control OpenGL extension. Only work on some drivers.
-# opengl-swc: Try to VSync with SGI_swap_control OpenGL extension. Only work on some drivers. Works only with GLX backend. Known to be most effective
-#on many drivers. Does not actually control paint timing, only buffer swap is affected, so it doesn’t have the effect of --sw-opti unlike other
-#methods.
-#Experimental.
-# opengl-mswc: Try to VSync with MESA_swap_control OpenGL extension. Basically the same as opengl-swc above, except the extension we use.
-# (Note some VSync methods may not be enabled at compile time.)
-vsync = "drm";
-
-# Enable DBE painting mode, intended to use with VSync to (hopefully) eliminate tearing.
-# Reported to have no effect, though.
-dbe = true;
-# Painting on X Composite overlay window. Recommended.
-paint-on-overlay = true;
-
-# Limit compton to repaint at most once every 1 / refresh_rate second to boost performance.
-# This should not be used with --vsync drm/opengl/opengl-oml as they essentially does --sw-opti's job already,
-# unless you wish to specify a lower refresh rate than the actual value.
-sw-opti = true;
-
-# Unredirect all windows if a full-screen opaque window is detected, to maximize performance for full-screen windows, like games.
-# Known to cause flickering when redirecting/unredirecting windows.
-# paint-on-overlay may make the flickering less obvious.
-unredir-if-possible = true;
-
-# Specify a list of conditions of windows that should always be considered focused.
-focus-exclude = [ ];
-
-# Use WM_TRANSIENT_FOR to group windows, and consider windows in the same group focused at the same time.
-detect-transient = true;
-# Use WM_CLIENT_LEADER to group windows, and consider windows in the same group focused at the same time.
-# WM_TRANSIENT_FOR has higher priority if --detect-transient is enabled, too.
-detect-client-leader = true;
-
-#################################
-#
-# Window type settings
-#
-#################################
-
-wintypes:
-{
- tooltip =
- {
- # fade: Fade the particular type of windows.
- fade = true;
- # shadow: Give those windows shadow
- shadow = false;
- # opacity: Default opacity for the type of windows.
- opacity = 0.85;
- # focus: Whether to always consider windows of this type focused.
- focus = true;
- };
-};
+# Thank you code_nomad: http://9m.no/ꪯ鵞
+# and Arch Wiki contributors: https://wiki.archlinux.org/index.php/Compton
+
+#################################
+#
+# Backend
+#
+#################################
+
+# Backend to use: "xrender" or "glx".
+# GLX backend is typically much faster but depends on a sane driver.
+backend = "glx";
+
+#################################
+#
+# GLX backend
+#
+#################################
+
+glx-no-stencil = true;
+
+# GLX backend: Copy unmodified regions from front buffer instead of redrawing them all.
+# My tests with nvidia-drivers show a 10% decrease in performance when the whole screen is modified,
+# but a 20% increase when only 1/4 is.
+# My tests on nouveau show terrible slowdown.
+# Useful with --glx-swap-method, as well.
+glx-copy-from-front = false;
+
+# GLX backend: Use MESA_copy_sub_buffer to do partial screen update.
+# My tests on nouveau shows a 200% performance boost when only 1/4 of the screen is updated.
+# May break VSync and is not available on some drivers.
+# Overrides --glx-copy-from-front.
+# glx-use-copysubbuffermesa = true;
+
+# GLX backend: Avoid rebinding pixmap on window damage.
+# Probably could improve performance on rapid window content changes, but is known to break things on some drivers (LLVMpipe).
+# Recommended if it works.
+# glx-no-rebind-pixmap = true;
+
+# GLX backend: GLX buffer swap method we assume.
+# Could be undefined (0), copy (1), exchange (2), 3-6, or buffer-age (-1).
+# undefined is the slowest and the safest, and the default value.
+# copy is fastest, but may fail on some drivers,
+# 2-6 are gradually slower but safer (6 is still faster than 0).
+# Usually, double buffer means 2, triple buffer means 3.
+# buffer-age means auto-detect using GLX_EXT_buffer_age, supported by some drivers.
+# Useless with --glx-use-copysubbuffermesa.
+# Partially breaks --resize-damage.
+# Defaults to undefined.
+glx-swap-method = "undefined";
+
+#################################
+#
+# Shadows
+#
+#################################
+
+# Enabled client-side shadows on windows.
+shadow = true;
+# Don't draw shadows on DND windows.
+no-dnd-shadow = true;
+# Avoid drawing shadows on dock/panel windows.
+no-dock-shadow = true;
+# The blur radius for shadows. (default 12)
+shadow-radius = 10;
+# The left offset for shadows. (default -15)
+shadow-offset-x = -5;
+# The top offset for shadows. (default -15)
+shadow-offset-y = -5;
+# The translucency for shadows. (default .75)
+shadow-opacity = 0.5;
+
+# Set if you want different colour shadows
+# shadow-red = 0.0;
+# shadow-green = 0.0;
+# shadow-blue = 0.0;
+
+# The shadow exclude options are helpful if you have shadows enabled. Due to the way compton draws its shadows, certain applications will have visual glitches
+# (most applications are fine, only apps that do weird things with xshapes or argb are affected).
+# This list includes all the affected apps I found in my testing. The "! name~=''" part excludes shadows on any "Unknown" windows, this prevents a visual glitch with the XFWM alt tab switcher.
+shadow-exclude = [
+ "! name~=''",
+ "name = 'Notification'",
+ "name = 'Plank'",
+ "name = 'Docky'",
+ "name = 'Kupfer'",
+ "name = 'xfce4-notifyd'",
+ "name *= 'VLC'",
+ "name *= 'compton'",
+ "name *= 'Chromium'",
+ "name *= 'Chrome'",
+ "class_g = 'Firefox' && argb",
+ "class_g = 'Conky'",
+ "class_g = 'Kupfer'",
+ "class_g = 'Synapse'",
+ "class_g ?= 'Notify-osd'",
+ "class_g ?= 'Cairo-dock'",
+ "class_g ?= 'Xfce4-notifyd'",
+ "class_g ?= 'Xfce4-power-manager'",
+ "_GTK_FRAME_EXTENTS@:c",
+ "_NET_WM_STATE@:32a *= '_NET_WM_STATE_HIDDEN'"
+];
+# Avoid drawing shadow on all shaped windows (see also: --detect-rounded-corners)
+shadow-ignore-shaped = false;
+
+#################################
+#
+# Opacity
+#
+#################################
+
+menu-opacity = 1;
+inactive-opacity = 1;
+active-opacity = 1;
+frame-opacity = 1;
+inactive-opacity-override = false;
+alpha-step = 0.06;
+
+# Dim inactive windows. (0.0 - 1.0)
+# inactive-dim = 0.2;
+# Do not let dimness adjust based on window opacity.
+# inactive-dim-fixed = true;
+# Blur background of transparent windows. Bad performance with X Render backend. GLX backend is preferred.
+# blur-background = true;
+# Blur background of opaque windows with transparent frames as well.
+# blur-background-frame = true;
+# Do not let blur radius adjust based on window opacity.
+blur-background-fixed = false;
+blur-background-exclude = [
+ "window_type = 'dock'",
+ "window_type = 'desktop'"
+];
+
+#################################
+#
+# Fading
+#
+#################################
+
+# Fade windows during opacity changes.
+fading = true;
+# The time between steps in a fade in milliseconds. (default 10).
+fade-delta = 4;
+# Opacity change between steps while fading in. (default 0.028).
+fade-in-step = 0.03;
+# Opacity change between steps while fading out. (default 0.03).
+fade-out-step = 0.03;
+# Fade windows in/out when opening/closing
+# no-fading-openclose = true;
+
+# Specify a list of conditions of windows that should not be faded.
+fade-exclude = [ ];
+
+#################################
+#
+# Other
+#
+#################################
+
+# Try to detect WM windows and mark them as active.
+mark-wmwin-focused = true;
+# Mark all non-WM but override-redirect windows active (e.g. menus).
+mark-ovredir-focused = true;
+# Use EWMH _NET_WM_ACTIVE_WINDOW to determine which window is focused instead of using FocusIn/Out events.
+# Usually more reliable but depends on a EWMH-compliant WM.
+use-ewmh-active-win = true;
+# Detect rounded corners and treat them as rectangular when --shadow-ignore-shaped is on.
+detect-rounded-corners = true;
+
+# Detect _NET_WM_OPACITY on client windows, useful for window managers not passing _NET_WM_OPACITY of client windows to frame windows.
+# This prevents opacity being ignored for some apps.
+# For example without this enabled my xfce4-notifyd is 100% opacity no matter what.
+detect-client-opacity = true;
+
+# Specify refresh rate of the screen.
+# If not specified or 0, compton will try detecting this with X RandR extension.
+refresh-rate = 0;
+
+# Set VSync method. VSync methods currently available:
+# none: No VSync
+# drm: VSync with DRM_IOCTL_WAIT_VBLANK. May only work on some drivers.
+# opengl: Try to VSync with SGI_video_sync OpenGL extension. Only work on some drivers.
+# opengl-oml: Try to VSync with OML_sync_control OpenGL extension. Only work on some drivers.
+# opengl-swc: Try to VSync with SGI_swap_control OpenGL extension. Only work on some drivers. Works only with GLX backend. Known to be most effective on many drivers. Does not actually control paint timing, only buffer swap is affected, so it doesn’t have the effect of --sw-opti unlike other methods. Experimental.
+# opengl-mswc: Try to VSync with MESA_swap_control OpenGL extension. Basically the same as opengl-swc above, except the extension we use.
+# (Note some VSync methods may not be enabled at compile time.)
+vsync = "opengl";
+
+# Enable DBE painting mode, intended to use with VSync to (hopefully) eliminate tearing.
+# Reported to have no effect, though.
+dbe = false;
+
+# Limit compton to repaint at most once every 1 / refresh_rate second to boost performance.
+# This should not be used with --vsync drm/opengl/opengl-oml as they essentially does --sw-opti's job already,
+# unless you wish to specify a lower refresh rate than the actual value.
+sw-opti = true;
+
+# Unredirect all windows if a full-screen opaque window is detected, to maximize performance for full-screen windows, like games.
+# Known to cause flickering when redirecting/unredirecting windows.
+# paint-on-overlay may make the flickering less obvious.
+unredir-if-possible = false;
+
+# Specify a list of conditions of windows that should always be considered focused.
+focus-exclude = [ ];
+
+# Use WM_TRANSIENT_FOR to group windows, and consider windows in the same group focused at the same time.
+detect-transient = true;
+# Use WM_CLIENT_LEADER to group windows, and consider windows in the same group focused at the same time.
+# WM_TRANSIENT_FOR has higher priority if --detect-transient is enabled, too.
+detect-client-leader = true;
+
+#################################
+#
+# Window type settings
+#
+#################################
+
+wintypes:
+{
+ tooltip =
+ {
+ # fade: Fade the particular type of windows.
+ fade = true;
+ # shadow: Give those windows shadow
+ shadow = false;
+ # opacity: Default opacity for the type of windows.
+ opacity = 0.85;
+ # focus: Whether to always consider windows of this type focused.
+ focus = true;
+ };
+};
+
+######################
+#
+# XSync
+# See: https://github.com/yshui/compton/commit/b18d46bcbdc35a3b5620d817dd46fbc76485c20d
+#
+######################
+
+# Attempt to synchronize client applications' draw calls with `XSync()`, used on GLX backend to ensure up-to-date window content is painted.
+xrender-sync = true;
+
+# use X Sync fence to sync clients' draw calls. Needed on nvidia-drivers with GLX backend for some users.
+xrender-sync-fence = true;
diff --git a/Phranque/fish/fishd.Phranque b/Phranque/fish/fishd.Phranque
new file mode 100644
index 00000000..93eb0b8c
--- /dev/null
+++ b/Phranque/fish/fishd.Phranque
@@ -0,0 +1,31 @@
+# This file is automatically generated by the fish.
+# Do NOT edit it directly, your changes will be overwritten.
+SET __fish_init_2_39_8:\x1d
+SET __fish_init_2_3_0:\x1d
+SET fish_color_autosuggestion:555\x1ebrblack
+SET fish_color_cancel:\x2dr
+SET fish_color_command:\x2d\x2dbold
+SET fish_color_comment:red
+SET fish_color_cwd:green
+SET fish_color_cwd_root:red
+SET fish_color_end:brmagenta
+SET fish_color_error:brred
+SET fish_color_escape:bryellow\x1e\x2d\x2dbold
+SET fish_color_history_current:\x2d\x2dbold
+SET fish_color_host:normal
+SET fish_color_match:\x2d\x2dbackground\x3dbrblue
+SET fish_color_normal:normal
+SET fish_color_operator:bryellow
+SET fish_color_param:cyan
+SET fish_color_quote:yellow
+SET fish_color_redirection:brblue
+SET fish_color_search_match:bryellow\x1e\x2d\x2dbackground\x3dbrblack
+SET fish_color_selection:white\x1e\x2d\x2dbold\x1e\x2d\x2dbackground\x3dbrblack
+SET fish_color_user:brgreen
+SET fish_color_valid_path:\x2d\x2dunderline
+SET fish_greeting:Welcome\x20to\x20fish\x2c\x20the\x20friendly\x20interactive\x20shell
+SET fish_key_bindings:fish_default_key_bindings
+SET fish_pager_color_completion:\x1d
+SET fish_pager_color_description:B3A06D\x1eyellow
+SET fish_pager_color_prefix:white\x1e\x2d\x2dbold\x1e\x2d\x2dunderline
+SET fish_pager_color_progress:brwhite\x1e\x2d\x2dbackground\x3dcyan
diff --git a/Phranque/fish/functions/fish_prompt.fish b/Phranque/fish/functions/fish_prompt.fish
new file mode 100644
index 00000000..474752a5
--- /dev/null
+++ b/Phranque/fish/functions/fish_prompt.fish
@@ -0,0 +1,3 @@
+function fish_prompt
+ echo "["(prompt_pwd)"][\$] "
+end
diff --git a/Phranque/i3.conf b/Phranque/i3.conf
deleted file mode 100644
index 37c74b93..00000000
--- a/Phranque/i3.conf
+++ /dev/null
@@ -1,296 +0,0 @@
-##########################################################################################
-# ____ __ _ _ _ ____ __ _ #
-# / __ \ / _(_) ( ) (_)___ \ / _(_) #
-# | | | |_ __ __ _| |_ ___ _|/ ___ _ __) | ___ ___ _ __ | |_ _ __ _ #
-# | | | | '_ \ / _` | _| \ \/ / / __| | ||__ < / __/ _ \| '_ \| _| |/ _` | #
-# | |__| | |_) | (_| | | | |> < \__ \ | |___) | | (_| (_) | | | | | | | (_| | #
-# \____/| .__/ \__,_|_| |_/_/\_\ |___/ |_|____/ _ \___\___/|_| |_|_| |_|\__, | #
-# / ____| | | | _ | __ \| | __/ | #
-# | (___ |_| _ ___| |_ ___ _ __ ___ (_) | |__) | |__ _ __ __ _ _ __ _|___/ _ ___ #
-# \___ \| | | / __| __/ _ \ '_ ` _ \ | ___/| '_ \| '__/ _` | '_ \ / _` | | | |/ _ \ #
-# ____) | |_| \__ \ || __/ | | | | |_ | | | | | | | | (_| | | | | (_| | |_| | __/ #
-# |_____/ \__, |___/\__\___|_| |_| |_(_) |_| |_| |_|_| \__,_|_| |_|\__, |\__,_|\___| #
-# __/ | | | #
-# |___/ |_| #
-##########################################################################################
-# Last Edit: 17/09/2018 #
-##########################################################################################
-
-set $mod Mod1
-set $sup Mod4
-
-# Font for window titles. Will also be used by the bar unless a different font
-# is used in the bar {} block below.
-font xft:Fira Sans:size=8
-
-############################################################
-# _ __ ____ _ _ _ #
-# | |/ / | _ \(_) | (_) #
-# | ' / ___ _ _ | |_) |_ _ __ __| |_ _ __ __ _ ___ #
-# | < / _ \ | | | | _ <| | '_ \ / _` | | '_ \ / _` / __| #
-# | . \ __/ |_| | | |_) | | | | | (_| | | | | | (_| \__ \ #
-# |_|\_\___|\__, | |____/|_|_| |_|\__,_|_|_| |_|\__, |___/ #
-# __/ | __/ | #
-# |___/ |___/ #
-############################################################
-
-## Use Mouse+$mod to drag floating windows to their wanted position
-floating_modifier $mod
-
-# kill focused window
-bindsym $mod+Shift+q kill
-
-# start dmenu (a program launcher)
-# bindsym $mod+d exec dmenu
-
-# Rofi bindings
-bindsym $mod+d exec rofi -show run
-bindsym $mod+shift+d exec rofi -show window
-
-# enable Alt+Tab Functionality
-bindsym $mod+Tab workspace next
-bindsym $mod+Shift+Tab workspace prev
-
-# change focus
-bindsym $mod+j focus left
-bindsym $mod+k focus down
-bindsym $mod+l focus up
-bindsym $mod+semicolon focus right
-
-# alternatively, you can use the cursor keys:
-bindsym $mod+Left focus left
-bindsym $mod+Down focus down
-bindsym $mod+Up focus up
-bindsym $mod+Right focus right
-
-# move focused window
-bindsym $mod+Shift+j move left
-bindsym $mod+Shift+k move down
-bindsym $mod+Shift+l move up
-bindsym $mod+Shift+semicolon move right
-
-# alternatively, you can use the cursor keys:
-bindsym $mod+Shift+Left move left
-bindsym $mod+Shift+Down move down
-bindsym $mod+Shift+Up move up
-bindsym $mod+Shift+Right move right
-
-# split in horizontal orientation
-bindsym $mod+h split h
-
-# split in vertical orientation
-bindsym $mod+v split v
-
-# enter fullscreen mode for the focused container
-bindsym $mod+f fullscreen toggle
-
-# change container layout (stacked, tabbed, toggle split)
-bindsym $mod+s layout stacking
-bindsym $mod+w layout tabbed
-bindsym $mod+e layout toggle split
-
-# toggle tiling / floating
-bindsym $mod+Shift+space floating toggle
-
-# change focus between tiling / floating windows
-bindsym $mod+space focus mode_toggle
-
-# focus the parent container
-bindsym $mod+a focus parent
-
-# reload the configuration file
-bindsym $mod+Shift+c reload
-
-# restart i3 inplace (preserves your layout/session, can be used to upgrade i3)
-bindsym $mod+Shift+r restart
-
-# exit i3 (logs you out of your X session)
-bindsym $mod+Shift+e exec "i3-nagbar -t warning -m 'You pressed the exit shortcut. Do you really want to exit i3? This will end your X session.' -b 'Yes, exit i3' 'i3-msg exit'"
-
-# Pulse Audio Controls
-bindsym XF86AudioRaiseVolume exec --no-startup-id pactl set-sink-volume 0 +5% #increase sound volume
-bindsym XF86AudioLowerVolume exec --no-startup-id pactl set-sink-volume 0 -5% #decrease sound volume
-bindsym XF86AudioMute exec --no-startup-id pactl set-sink-mute 0 toggle # mute sound
-
-# Sreen brightness controls
-bindsym XF86MonBrightnessUp exec --no-startup-id sudo xbacklight -inc 5 # increase screen brightness
-bindsym XF86MonBrightnessDown exec --no-startup-id sudo xbacklight -dec 5 # decrease screen brightness
-
-# Media player controls
-bindsym XF86AudioPlay exec --no-startup-id playerctl play
-bindsym XF86AudioPause exec --no-startup-id playerctl pause
-bindsym XF86AudioNext exec --no-startup-id playerctl next
-bindsym XF86AudioPrev exec --no-startup-id playerctl previous
-
-#############################################################
-# __ __ _ #
-# \ \ / / | | #
-# \ \ /\ / /__ _ __| | _____ _ __ __ _ ___ ___ ___ #
-# \ \/ \/ / _ \| '__| |/ / __| '_ \ / _` |/ __/ _ \/ __| #
-# \ /\ / (_) | | | <\__ \ |_) | (_| | (_| __/\__ \ #
-# \/ \/ \___/|_| |_|\_\___/ .__/ \__,_|\___\___||___/ #
-# | | #
-# |_| #
-# #
-#############################################################
-
-# Define names for default workspaces for which we configure key bindings later on.
-# We use variables to avoid repeating the names in multiple places.
-set $ws1 "1"
-set $ws2 "2"
-set $ws3 "3"
-set $ws4 "4"
-set $ws5 "5"
-set $ws6 "6"
-set $ws7 "7"
-set $ws8 "8"
-set $ws9 "9"
-set $ws10 "10"
-
-# switch to workspace
-bindsym $mod+1 workspace $ws1
-bindsym $mod+2 workspace $ws2
-bindsym $mod+3 workspace $ws3
-bindsym $mod+4 workspace $ws4
-bindsym $mod+5 workspace $ws5
-bindsym $mod+6 workspace $ws6
-bindsym $mod+7 workspace $ws7
-bindsym $mod+8 workspace $ws8
-bindsym $mod+9 workspace $ws9
-bindsym $mod+0 workspace $ws10
-
-# move focused container to workspace
-bindsym $mod+Shift+1 move container to workspace $ws1
-bindsym $mod+Shift+2 move container to workspace $ws2
-bindsym $mod+Shift+3 move container to workspace $ws3
-bindsym $mod+Shift+4 move container to workspace $ws4
-bindsym $mod+Shift+5 move container to workspace $ws5
-bindsym $mod+Shift+6 move container to workspace $ws6
-bindsym $mod+Shift+7 move container to workspace $ws7
-bindsym $mod+Shift+8 move container to workspace $ws8
-bindsym $mod+Shift+9 move container to workspace $ws9
-bindsym $mod+Shift+0 move container to workspace $ws10
-
-##################################
-# __ __ _ #
-# | \/ | | | #
-# | \ / | ___ __| | ___ ___ #
-# | |\/| |/ _ \ / _` |/ _ \/ __| #
-# | | | | (_) | (_| | __/\__ \ #
-# |_| |_|\___/ \__,_|\___||___/ #
-# #
-##################################
-
-# resize window (you can also use the mouse for that)
-bindsym $mod+r mode "resize"
-mode "resize" {
- # These bindings trigger as soon as you enter the resize mode
-
- # Pressing left will shrink the window’s width.
- # Pressing right will grow the window’s width.
- # Pressing up will shrink the window’s height.
- # Pressing down will grow the window’s height.
- bindsym j resize shrink width 10 px or 10 ppt
- bindsym k resize grow height 10 px or 10 ppt
- bindsym l resize shrink height 10 px or 10 ppt
- bindsym semicolon resize grow width 10 px or 10 ppt
-
- # same bindings, but for the arrow keys
- bindsym Left resize shrink width 10 px or 10 ppt
- bindsym Down resize grow height 10 px or 10 ppt
- bindsym Up resize shrink height 10 px or 10 ppt
- bindsym Right resize grow width 10 px or 10 ppt
-
- # back to normal: Enter or Escape or $mod+r
- bindsym Return mode "default"
- bindsym Escape mode "default"
- bindsym $mod+r mode "default"
-}
-
-#Application Launcher
-set $launcher "application launcher: [w]eb [e]mails [t]erminal [f]iles [s]team [a]udio"
-bindsym $mod+Return mode $launcher
-mode $launcher {
- bindsym w exec --no-startup-id "firefox" mode "default"
- bindsym e exec --no-startup-id "thunderbird" mode "default"
- bindsym t exec --no-startup-id "urxvtc" mode "default"
- bindsym s exec --no-startup-id "steam" mode "default"
- bindsym a exec --no-startup-id "pavucontrol" mode "default"
- bindsym f exec --no-startup-id "pcmanfm" mode "default"
- bindsym Return mode "default"
- bindsym Escape mode "default"
-}
-
-#Power Menu
-set $sysmenu "system: [r]eload [e]xit [l]ock [p]oweroff re[b]oot [s]uspend [h]ibernate"
-bindsym $mod+Delete mode $sysmenu
-mode $sysmenu {
- bindsym r restart
- bindsym e exit
- bindsym p exec sudo shutdown -P now
- bindsym b exec sudo shutdown -r now
- bindsym s exec "i3lock -i ~/.lock -f && sudo pm-suspend" mode "default"
- bindsym h exec "i3lock -i ~/.lock -f && sudo pm-hibernate" mode "default"
- bindsym l exec "i3lock -i ~/.lock -f" mode "default"
- bindsym Return mode "default"
- bindsym Escape mode "default"
-}
-
-################################
-# _ ____ ____ #
-# (_)___ \ | _ \ #
-# _ __) | | |_) | __ _ _ __ #
-# | ||__ < | _ < / _` | '__| #
-# | |___) | | |_) | (_| | | #
-# |_|____/ |____/ \__,_|_| #
-# #
-################################
-
-# Start i3bar to display a workspace bar (plus the system information i3blocks finds
-# out, if available)
-bar {
- font pango:Fira Sans 10
- status_command i3blocks -c ~/.config/i3/i3blocks.conf
- colors {
- separator #FFFFFF
- background #000000
- statusline #FFFFFF
- focused_workspace #FFFFFF #FFFFFF #000000
- active_workspace #FFFFFF #FFFFFF #000000
- inactive_workspace #FFFFFF #000000 #FFFFFF
- urgent_workspace #FFFFFF #000000 #FFFFFF
- }
-}
-
-###############################################################
-# _____ _ _ _ _ _ _ _ #
-# |_ _| (_) | (_) | (_) | | (_) #
-# | | _ __ _| |_ _ __ _| |_ ___ __ _| |_ _ ___ _ __ #
-# | | | '_ \| | __| |/ _` | | / __|/ _` | __| |/ _ \| '_ \ #
-# _| |_| | | | | |_| | (_| | | \__ \ (_| | |_| | (_) | | | | #
-# |_____|_| |_|_|\__|_|\__,_|_|_|___/\__,_|\__|_|\___/|_| |_| #
-# #
-###############################################################
-
-# Set Keyboard Layout
-exec --no-startup-id setxkbmap gb
-
-# Set Wallpaper
-exec --no-startup-id feh --bg-fill ~/.wallpaper
-
-# Start RXVT-Unicode Daemon
-exec --no-startup-id urxvtd
-
-# Start Compton
-exec --no-startup-id compton -b --config ~/.config/compton.conf
-
-# Start Network Manager Tray Applet
-exec --no-startup-id nm-applet
-
-# Remove Window Border
-for_window [class="^.*"] border pixel 0
-
-#Configure Gaps
-gaps inner 5
-gaps outer 0
-smart_gaps on
diff --git a/Phranque/i3blocks.conf b/Phranque/i3blocks.conf
deleted file mode 100644
index 00ec0b53..00000000
--- a/Phranque/i3blocks.conf
+++ /dev/null
@@ -1,161 +0,0 @@
-# i3blocks config file
-#
-# Please see man i3blocks for a complete reference!
-# The man page is also hosted at http://vivien.github.io/i3blocks
-#
-# List of valid properties:
-#
-# align
-# color
-# command
-# full_text
-# instance
-# interval
-# label
-# min_width
-# name
-# separator
-# separator_block_width
-# short_text
-# signal
-# urgent
-
-# Global properties
-#
-# The top properties below are applied to every block, but can be overridden.
-# Each block command defaults to the script name to avoid boilerplate.
-command=/usr/libexec/i3blocks/$BLOCK_NAME
-separator_block_width=15
-markup=none
-
-# Volume indicator
-#
-# The first parameter sets the step (and units to display)
-# The second parameter overrides the mixer selection
-# See the script for details.
-[volume]
-label=VOL
-#label=♪
-instance=Master
-#instance=PCM
-interval=once
-signal=10
-separator=false
-
-[brghtness]
-label=LIGHT
-interval=1
-command=xbacklight -get | sed 's/$/%/'
-
-# Memory usage
-#
-# The type defaults to "mem" if the instance is not specified.
-[memory]
-label=RAM
-separator=false
-interval=30
-instance=mem
-
-#[memory]
-#label=SWAP
-#instance=swap
-#separator=false
-#interval=30
-
-# Disk usage
-#
-# The directory defaults to $HOME if the instance is not specified.
-# The script may be called with a optional argument to set the alert
-# (defaults to 10 for 10%).
-[disk]
-label=DISK
-#instance=/mnt/data
-interval=30
-
-# Network interface monitoring
-#
-# If the instance is not specified, use the interface used for default route.
-# The address can be forced to IPv4 or IPv6 with -4 or -6 switches.
-[iface]
-#instance=wlan0
-color=#00FF00
-interval=10
-separator=false
-
-[wifi]
-#instance=wlp3s0
-interval=10
-separator=false
-
-[bandwidth]
-#instance=eth0
-interval=5
-
-# CPU usage
-#
-# The script may be called with -w and -c switches to specify thresholds,
-# see the script for details.
-[cpu_usage]
-label=CPU
-interval=10
-min_width=CPU: 100.00%
-separator=false
-
-#[load_average]
-#interval=10
-
-# Battery indicator
-#
-# The battery instance defaults to 0.
-[battery]
-label=BAT
-#label=⚡
-#instance=1
-interval=30
-
-# Temperature
-#
-# Support multiple chips, though lm-sensors.
-# The script may be called with -w and -c switches to specify thresholds,
-# see the script for details.
-[temperature]
-label=TEMP
-interval=10
-
-
-# Date Time
-#
-[time]
-command=date '+%Y-%m-%d %H:%M:%S'
-interval=5
-
-# Generic media player support
-#
-# This displays "ARTIST - SONG" if a music is playing.
-# Supported players are: spotify, vlc, audacious, xmms2, mplayer, and others.
-#[mediaplayer]
-#instance=spotify
-#interval=5
-#signal=10
-
-# OpenVPN support
-#
-# Support multiple VPN, with colors.
-#[openvpn]
-#interval=20
-
-# Key indicators
-#
-# Add the following bindings to i3 config file:
-#
-# bindsym --release Caps_Lock exec pkill -SIGRTMIN+11 i3blocks
-# bindsym --release Num_Lock exec pkill -SIGRTMIN+11 i3blocks
-#[keyindicator]
-#instance=CAPS
-#interval=once
-#signal=11
-
-#[keyindicator]
-#instance=NUM
-#interval=once
-#signal=11
diff --git a/suckless/st/LICENSE b/suckless/st/LICENSE
new file mode 100644
index 00000000..c356c39d
--- /dev/null
+++ b/suckless/st/LICENSE
@@ -0,0 +1,34 @@
+MIT/X Consortium License
+
+© 2014-2018 Hiltjo Posthuma
+© 2018 Devin J. Pohly
+© 2014-2017 Quentin Rameau
+© 2009-2012 Aurélien APTEL
+© 2008-2017 Anselm R Garbe
+© 2012-2017 Roberto E. Vargas Caballero
+© 2012-2016 Christoph Lohmann <20h at r-36 dot net>
+© 2013 Eon S. Jeon
+© 2013 Alexander Sedov
+© 2013 Mark Edgar
+© 2013-2014 Eric Pruitt
+© 2013 Michael Forney
+© 2013-2014 Markus Teich
+© 2014-2015 Laslo Hunhold
+
+Permission is hereby granted, free of charge, to any person obtaining a
+copy of this software and associated documentation files (the "Software"),
+to deal in the Software without restriction, including without limitation
+the rights to use, copy, modify, merge, publish, distribute, sublicense,
+and/or sell copies of the Software, and to permit persons to whom the
+Software is furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in
+all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
+THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
+FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
+DEALINGS IN THE SOFTWARE.
diff --git a/suckless/st/Makefile b/suckless/st/Makefile
new file mode 100644
index 00000000..0b3cecd0
--- /dev/null
+++ b/suckless/st/Makefile
@@ -0,0 +1,57 @@
+# st - simple terminal
+# See LICENSE file for copyright and license details.
+.POSIX:
+
+include config.mk
+
+SRC = st.c x.c
+OBJ = $(SRC:.c=.o)
+
+all: options st
+
+options:
+ @echo st build options:
+ @echo "CFLAGS = $(STCFLAGS)"
+ @echo "LDFLAGS = $(STLDFLAGS)"
+ @echo "CC = $(CC)"
+
+config.h:
+ cp config.def.h config.h
+
+.c.o:
+ $(CC) $(STCFLAGS) -c $<
+
+st.o: config.h st.h win.h
+x.o: arg.h st.h win.h
+
+$(OBJ): config.h config.mk
+
+st: $(OBJ)
+ $(CC) -o $@ $(OBJ) $(STLDFLAGS)
+
+clean:
+ rm -f st $(OBJ) st-$(VERSION).tar.gz
+
+dist: clean
+ mkdir -p st-$(VERSION)
+ cp -R FAQ LEGACY TODO LICENSE Makefile README config.mk\
+ config.def.h st.info st.1 arg.h st.h win.h $(SRC)\
+ st-$(VERSION)
+ tar -cf - st-$(VERSION) | gzip > st-$(VERSION).tar.gz
+ rm -rf st-$(VERSION)
+
+install: st
+ mkdir -p $(DESTDIR)$(PREFIX)/bin
+ cp -f st $(DESTDIR)$(PREFIX)/bin
+ chmod 755 $(DESTDIR)$(PREFIX)/bin/st
+ mkdir -p $(DESTDIR)$(MANPREFIX)/man1
+ sed "s/VERSION/$(VERSION)/g" < st.1 > $(DESTDIR)$(MANPREFIX)/man1/st.1
+ chmod 644 $(DESTDIR)$(MANPREFIX)/man1/st.1
+ tic -sx st.info
+ @echo Please see the README file regarding the terminfo entry of st.
+
+uninstall:
+ rm -f $(DESTDIR)$(PREFIX)/bin/st
+ rm -f $(DESTDIR)$(MANPREFIX)/man1/st.1
+
+.PHONY: all options clean dist install uninstall
diff --git a/suckless/st/README.md b/suckless/st/README.md
new file mode 100644
index 00000000..f7c9c0b2
--- /dev/null
+++ b/suckless/st/README.md
@@ -0,0 +1,75 @@
+# Luke's build of st - the simple (suckless) terminal
+
+The [suckless terminal (st)](https://st.suckless.org/) with some additional features:
+
++ Compatibility with `Xresources` and `pywal` for dynamic colors.
++ Default solarized colors otherwize.
++ Transparency/alpha, which is also adjustable from `~/.Xresources`.
++ Default font is system "mono" at 14pt, meaning the font will match your system font.
++ Very useful keybinds including:
+ + Copy is alt-c, paste is alt-v or alt-p pastes from primary selection
+ + Zoom in/out or increase font size with Alt+Shift+k/j or u/d for larger intervals.
+ + Hold alt and press either ↑/↓ or the vim keys k/j to move up/down in the terminal.
+ + Shift+Mouse wheel do the same.
+ + Alt-u and Alt-d scroll back/forward in history a page at a time.
+ + Alt-PageUp and Alt-PageDown will do the same.
++ Vertcenter
++ Scrollback
++ updated to latest version 0.8.1
+
+The following additional bindings were added before I forked this:
+
++ Scroll through history -- Shift+PageUp/PageDown or Shift+Mouse wheel
++ Increase/decrease font size -- Shift+Alt+PageUp/PageDown
++ Return to default font size -- Shift+Alt+Home
++ Paste -- Shift+Insert
+
+## Installation for newbs
+
+```
+make
+sudo make install
+```
+
+Obviously, `make` is required to build. `fontconfig` is required for the
+default build, since it asks `fontconfig` for your system monospace font. It
+might be obvious, but `libX11` and `libXft` are required as well. Chances are,
+you have all of this installed already.
+
+On OpenBSD, be sure to edit `config.mk` first and remove `-lrt` from the
+`$LIBS` before compiling.
+
+## How to configure dynamically with Xresrouces
+
+For many key variables, this build of `st` will look for X settings set in
+either `~/.Xdefaults` or `~/.Xresources`. You must run `xrdb` on one of these
+files to load the settings.
+
+For example, you can define your desired fonts, transparency or colors:
+
+```
+*.font: Liberation Mono:pixelsize=12:antialias=true:autohint=true;
+*.alpha: 150
+*.color0: #111
+...
+```
+
+The `alpha` value (for transparency) goes from `0` (transparent) to `255`
+(opaque).
+
+### Colors
+
+To be clear about the color settings:
+
+- This build will use colorized colors by default and as a fallback.
+- If there are Xresources colors defined, those will take priority.
+- But if `wal` has run in your session, its colors will take priority.
+
+Note that when you run `wal`, it will negate the transparency of existing
+windows, but new windows will continue with the previously defined
+transparency.
+
+## Contact
+
+- Luke Smith
+- [https://lukesmith.xyz](https://lukesmith.xyz)
diff --git a/suckless/st/arg.h b/suckless/st/arg.h
new file mode 100644
index 00000000..a22e019e
--- /dev/null
+++ b/suckless/st/arg.h
@@ -0,0 +1,50 @@
+/*
+ * Copy me if you can.
+ * by 20h
+ */
+
+#ifndef ARG_H__
+#define ARG_H__
+
+extern char *argv0;
+
+/* use main(int argc, char *argv[]) */
+#define ARGBEGIN for (argv0 = *argv, argv++, argc--;\
+ argv[0] && argv[0][0] == '-'\
+ && argv[0][1];\
+ argc--, argv++) {\
+ char argc_;\
+ char **argv_;\
+ int brk_;\
+ if (argv[0][1] == '-' && argv[0][2] == '\0') {\
+ argv++;\
+ argc--;\
+ break;\
+ }\
+ int i_;\
+ for (i_ = 1, brk_ = 0, argv_ = argv;\
+ argv[0][i_] && !brk_;\
+ i_++) {\
+ if (argv_ != argv)\
+ break;\
+ argc_ = argv[0][i_];\
+ switch (argc_)
+
+#define ARGEND }\
+ }
+
+#define ARGC() argc_
+
+#define EARGF(x) ((argv[0][i_+1] == '\0' && argv[1] == NULL)?\
+ ((x), abort(), (char *)0) :\
+ (brk_ = 1, (argv[0][i_+1] != '\0')?\
+ (&argv[0][i_+1]) :\
+ (argc--, argv++, argv[0])))
+
+#define ARGF() ((argv[0][i_+1] == '\0' && argv[1] == NULL)?\
+ (char *)0 :\
+ (brk_ = 1, (argv[0][i_+1] != '\0')?\
+ (&argv[0][i_+1]) :\
+ (argc--, argv++, argv[0])))
+
+#endif
diff --git a/config.h b/suckless/st/config.h
similarity index 99%
rename from config.h
rename to suckless/st/config.h
index 2c6fe864..96bb401d 100644
--- a/config.h
+++ b/suckless/st/config.h
@@ -5,8 +5,8 @@
*
* font: see http://freedesktop.org/software/fontconfig/fontconfig-user.html
*/
-static char *font = "Fira Mono for Powerline:pixelsize=12:antialias=true:autohint=true";
-static int borderpx = 5;
+static char *font = "Fira Mono:pixelsize=12:antialias=true:autohint=true";
+static int borderpx = 10;
/*
* What program is execed by st depends of these precedence rules:
diff --git a/suckless/st/config.mk b/suckless/st/config.mk
new file mode 100644
index 00000000..3b00d7e5
--- /dev/null
+++ b/suckless/st/config.mk
@@ -0,0 +1,28 @@
+# st version
+VERSION = 0.8.1
+
+# Customize below to fit your system
+
+# paths
+PREFIX = /usr/local
+MANPREFIX = $(PREFIX)/share/man
+
+X11INC = /usr/X11R6/include
+X11LIB = /usr/X11R6/lib
+
+# includes and libs
+INCS = -I$(X11INC) \
+ `pkg-config --cflags fontconfig` \
+ `pkg-config --cflags freetype2`
+LIBS = -L${X11LIB} -lm -lrt -lX11 -lutil -lXft -lXrender \
+ `pkg-config --libs fontconfig` \
+ `pkg-config --libs freetype2`
+
+# flags
+CPPFLAGS = -DVERSION=\"$(VERSION)\" -D_XOPEN_SOURCE=600
+STCFLAGS = $(INCS) $(CPPFLAGS) $(CFLAGS)
+STLDFLAGS = $(LIBS) $(LDFLAGS)
+
+# compiler and linker
+# CC = c99
+
diff --git a/suckless/st/st b/suckless/st/st
new file mode 100755
index 00000000..02fd7d49
Binary files /dev/null and b/suckless/st/st differ
diff --git a/suckless/st/st.1 b/suckless/st/st.1
new file mode 100644
index 00000000..b3f63b6d
--- /dev/null
+++ b/suckless/st/st.1
@@ -0,0 +1,183 @@
+.TH ST 1 st\-VERSION
+.SH NAME
+st \- simple terminal (Luke Smith (https://lukesmith.xyz)'s build)
+.SH SYNOPSIS
+.B st
+.RB [ \-aiv ]
+.RB [ \-c
+.IR class ]
+.RB [ \-f
+.IR font ]
+.RB [ \-g
+.IR geometry ]
+.RB [ \-n
+.IR name ]
+.RB [ \-o
+.IR iofile ]
+.RB [ \-T
+.IR title ]
+.RB [ \-t
+.IR title ]
+.RB [ \-l
+.IR line ]
+.RB [ \-w
+.IR windowid ]
+.RB [[ \-e ]
+.IR command
+.RI [ arguments ...]]
+.PP
+.B st
+.RB [ \-aiv ]
+.RB [ \-c
+.IR class ]
+.RB [ \-f
+.IR font ]
+.RB [ \-g
+.IR geometry ]
+.RB [ \-n
+.IR name ]
+.RB [ \-o
+.IR iofile ]
+.RB [ \-T
+.IR title ]
+.RB [ \-t
+.IR title ]
+.RB [ \-w
+.IR windowid ]
+.RB \-l
+.IR line
+.RI [ stty_args ...]
+.SH DESCRIPTION
+.B st
+is a simple terminal emulator.
+.SH OPTIONS
+.TP
+.B \-a
+disable alternate screens in terminal
+.TP
+.BI \-c " class"
+defines the window class (default $TERM).
+.TP
+.BI \-f " font"
+defines the
+.I font
+to use when st is run.
+.TP
+.BI \-g " geometry"
+defines the X11 geometry string.
+The form is [=][{xX}][{+-}{+-}]. See
+.BR XParseGeometry (3)
+for further details.
+.TP
+.B \-i
+will fixate the position given with the -g option.
+.TP
+.BI \-n " name"
+defines the window instance name (default $TERM).
+.TP
+.BI \-o " iofile"
+writes all the I/O to
+.I iofile.
+This feature is useful when recording st sessions. A value of "-" means
+standard output.
+.TP
+.BI \-T " title"
+defines the window title (default 'st').
+.TP
+.BI \-t " title"
+defines the window title (default 'st').
+.TP
+.BI \-w " windowid"
+embeds st within the window identified by
+.I windowid
+.TP
+.BI \-l " line"
+use a tty
+.I line
+instead of a pseudo terminal.
+.I line
+should be a (pseudo-)serial device (e.g. /dev/ttyS0 on Linux for serial port
+0).
+When this flag is given
+remaining arguments are used as flags for
+.BR stty(1).
+By default st initializes the serial line to 8 bits, no parity, 1 stop bit
+and a 38400 baud rate. The speed is set by appending it as last argument
+(e.g. 'st -l /dev/ttyS0 115200'). Arguments before the last one are
+.BR stty(1)
+flags. If you want to set odd parity on 115200 baud use for example 'st -l
+/dev/ttyS0 parenb parodd 115200'. Set the number of bits by using for
+example 'st -l /dev/ttyS0 cs7 115200'. See
+.BR stty(1)
+for more arguments and cases.
+.TP
+.B \-v
+prints version information to stderr, then exits.
+.TP
+.BI \-e " command " [ " arguments " "... ]"
+st executes
+.I command
+instead of the shell. If this is used it
+.B must be the last option
+on the command line, as in xterm / rxvt.
+This option is only intended for compatibility,
+and all the remaining arguments are used as a command
+even without it.
+.SH SHORTCUTS
+.TP
+.B Alt-j/k or Alt-Up/Down or Alt-Mouse Wheel
+Scroll up/down one line at a time.
+.TP
+.B Alt-u/d or Alt-Page Up/Page Down
+Scroll up/down one screen at a time.
+.TP
+.B Alt-Shift-k/j or Alt-Shift-Page Up/Page Down or Alt-Shift-Mouse Wheel
+Increase or decrease font size.
+.TP
+.B Alt-Shift-Home
+Reset to default font size.
+.TP
+.B Shift-Insert or Alt-v
+Paste from clipboard.
+.TP
+.B Alt-c
+Copy to clipboard.
+.TP
+.B Alt-p
+Paste/input primary selection.
+.TP
+.B Break
+Send a break in the serial line.
+Break key is obtained in PC keyboards
+pressing at the same time control and pause.
+.TP
+.B Ctrl-Print Screen
+Toggle if st should print to the
+.I iofile.
+.TP
+.B Shift-Print Screen
+Print the full screen to the
+.I iofile.
+.TP
+.B Print Screen
+Print the selection to the
+.I iofile.
+.TP
+.B Ctrl-Shift-i
+Launch dmenu to enter a unicode codepoint and send the corresponding glyph
+to st.
+.SH CUSTOMIZATION
+.B st
+can be customized by creating a custom config.h and (re)compiling the source
+code. This keeps it fast, secure and simple.
+.SH AUTHORS
+See the LICENSE file for the authors.
+.SH LICENSE
+See the LICENSE file for the terms of redistribution.
+.SH SEE ALSO
+.BR tabbed (1),
+.BR utmp (1),
+.BR stty (1)
+.SH BUGS
+See the TODO file in the distribution.
+
diff --git a/suckless/st/st.c b/suckless/st/st.c
new file mode 100644
index 00000000..45d25447
--- /dev/null
+++ b/suckless/st/st.c
@@ -0,0 +1,2685 @@
+/* See LICENSE for license details. */
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+
+#include "st.h"
+#include "win.h"
+
+#if defined(__linux)
+ #include
+#elif defined(__OpenBSD__) || defined(__NetBSD__) || defined(__APPLE__)
+ #include
+#elif defined(__FreeBSD__) || defined(__DragonFly__)
+ #include
+#endif
+
+/* Arbitrary sizes */
+#define UTF_INVALID 0xFFFD
+#define UTF_SIZ 4
+#define ESC_BUF_SIZ (128*UTF_SIZ)
+#define ESC_ARG_SIZ 16
+#define STR_BUF_SIZ ESC_BUF_SIZ
+#define STR_ARG_SIZ ESC_ARG_SIZ
+
+/* macros */
+#define IS_SET(flag) ((term.mode & (flag)) != 0)
+#define NUMMAXLEN(x) ((int)(sizeof(x) * 2.56 + 0.5) + 1)
+#define ISCONTROLC0(c) (BETWEEN(c, 0, 0x1f) || (c) == '\177')
+#define ISCONTROLC1(c) (BETWEEN(c, 0x80, 0x9f))
+#define ISCONTROL(c) (ISCONTROLC0(c) || ISCONTROLC1(c))
+#define ISDELIM(u) (utf8strchr(worddelimiters, u) != NULL)
+
+/* constants */
+#define ISO14755CMD "dmenu -w \"$WINDOWID\" -p codepoint: ] [;]] []] */
+typedef struct {
+ char buf[ESC_BUF_SIZ]; /* raw string */
+ int len; /* raw string length */
+ char priv;
+ int arg[ESC_ARG_SIZ];
+ int narg; /* nb of args */
+ char mode[2];
+} CSIEscape;
+
+/* STR Escape sequence structs */
+/* ESC type [[ [] [;]] ] ESC '\' */
+typedef struct {
+ char type; /* ESC type ... */
+ char buf[STR_BUF_SIZ]; /* raw string */
+ int len; /* raw string length */
+ char *args[STR_ARG_SIZ];
+ int narg; /* nb of args */
+} STREscape;
+
+static void execsh(char *, char **);
+static void stty(char **);
+static void sigchld(int);
+static void ttywriteraw(const char *, size_t);
+
+static void csidump(void);
+static void csihandle(void);
+static void csiparse(void);
+static void csireset(void);
+static int eschandle(uchar);
+static void strdump(void);
+static void strhandle(void);
+static void strparse(void);
+static void strreset(void);
+
+static void tprinter(char *, size_t);
+static void tdumpsel(void);
+static void tdumpline(int);
+static void tdump(void);
+static void tclearregion(int, int, int, int);
+static void tcursor(int);
+static void tdeletechar(int);
+static void tdeleteline(int);
+static void tinsertblank(int);
+static void tinsertblankline(int);
+static int tlinelen(int);
+static void tmoveto(int, int);
+static void tmoveato(int, int);
+static void tnewline(int);
+static void tputtab(int);
+static void tputc(Rune);
+static void treset(void);
+static void tscrollup(int, int, int);
+static void tscrolldown(int, int, int);
+static void tsetattr(int *, int);
+static void tsetchar(Rune, Glyph *, int, int);
+static void tsetdirt(int, int);
+static void tsetscroll(int, int);
+static void tswapscreen(void);
+static void tsetmode(int, int, int *, int);
+static int twrite(const char *, int, int);
+static void tfulldirt(void);
+static void tcontrolcode(uchar );
+static void tdectest(char );
+static void tdefutf8(char);
+static int32_t tdefcolor(int *, int *, int);
+static void tdeftran(char);
+static void tstrsequence(uchar);
+
+static void drawregion(int, int, int, int);
+
+static void selnormalize(void);
+static void selscroll(int, int);
+static void selsnap(int *, int *, int);
+
+static size_t utf8decode(const char *, Rune *, size_t);
+static Rune utf8decodebyte(char, size_t *);
+static char utf8encodebyte(Rune, size_t);
+static char *utf8strchr(char *, Rune);
+static size_t utf8validate(Rune *, size_t);
+
+static char *base64dec(const char *);
+static char base64dec_getc(const char **);
+
+static ssize_t xwrite(int, const char *, size_t);
+
+/* Globals */
+static Term term;
+static Selection sel;
+static CSIEscape csiescseq;
+static STREscape strescseq;
+static int iofd = 1;
+static int cmdfd;
+static pid_t pid;
+
+static uchar utfbyte[UTF_SIZ + 1] = {0x80, 0, 0xC0, 0xE0, 0xF0};
+static uchar utfmask[UTF_SIZ + 1] = {0xC0, 0x80, 0xE0, 0xF0, 0xF8};
+static Rune utfmin[UTF_SIZ + 1] = { 0, 0, 0x80, 0x800, 0x10000};
+static Rune utfmax[UTF_SIZ + 1] = {0x10FFFF, 0x7F, 0x7FF, 0xFFFF, 0x10FFFF};
+
+ssize_t
+xwrite(int fd, const char *s, size_t len)
+{
+ size_t aux = len;
+ ssize_t r;
+
+ while (len > 0) {
+ r = write(fd, s, len);
+ if (r < 0)
+ return r;
+ len -= r;
+ s += r;
+ }
+
+ return aux;
+}
+
+void *
+xmalloc(size_t len)
+{
+ void *p = malloc(len);
+
+ if (!p)
+ die("Out of memory\n");
+
+ return p;
+}
+
+void *
+xrealloc(void *p, size_t len)
+{
+ if ((p = realloc(p, len)) == NULL)
+ die("Out of memory\n");
+
+ return p;
+}
+
+char *
+xstrdup(char *s)
+{
+ if ((s = strdup(s)) == NULL)
+ die("Out of memory\n");
+
+ return s;
+}
+
+size_t
+utf8decode(const char *c, Rune *u, size_t clen)
+{
+ size_t i, j, len, type;
+ Rune udecoded;
+
+ *u = UTF_INVALID;
+ if (!clen)
+ return 0;
+ udecoded = utf8decodebyte(c[0], &len);
+ if (!BETWEEN(len, 1, UTF_SIZ))
+ return 1;
+ for (i = 1, j = 1; i < clen && j < len; ++i, ++j) {
+ udecoded = (udecoded << 6) | utf8decodebyte(c[i], &type);
+ if (type != 0)
+ return j;
+ }
+ if (j < len)
+ return 0;
+ *u = udecoded;
+ utf8validate(u, len);
+
+ return len;
+}
+
+Rune
+utf8decodebyte(char c, size_t *i)
+{
+ for (*i = 0; *i < LEN(utfmask); ++(*i))
+ if (((uchar)c & utfmask[*i]) == utfbyte[*i])
+ return (uchar)c & ~utfmask[*i];
+
+ return 0;
+}
+
+size_t
+utf8encode(Rune u, char *c)
+{
+ size_t len, i;
+
+ len = utf8validate(&u, 0);
+ if (len > UTF_SIZ)
+ return 0;
+
+ for (i = len - 1; i != 0; --i) {
+ c[i] = utf8encodebyte(u, 0);
+ u >>= 6;
+ }
+ c[0] = utf8encodebyte(u, len);
+
+ return len;
+}
+
+char
+utf8encodebyte(Rune u, size_t i)
+{
+ return utfbyte[i] | (u & ~utfmask[i]);
+}
+
+char *
+utf8strchr(char *s, Rune u)
+{
+ Rune r;
+ size_t i, j, len;
+
+ len = strlen(s);
+ for (i = 0, j = 0; i < len; i += j) {
+ if (!(j = utf8decode(&s[i], &r, len - i)))
+ break;
+ if (r == u)
+ return &(s[i]);
+ }
+
+ return NULL;
+}
+
+size_t
+utf8validate(Rune *u, size_t i)
+{
+ if (!BETWEEN(*u, utfmin[i], utfmax[i]) || BETWEEN(*u, 0xD800, 0xDFFF))
+ *u = UTF_INVALID;
+ for (i = 1; *u > utfmax[i]; ++i)
+ ;
+
+ return i;
+}
+
+static const char base64_digits[] = {
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 0, 0,
+ 63, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 0, 0, 0, -1, 0, 0, 0, 0, 1,
+ 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21,
+ 22, 23, 24, 25, 0, 0, 0, 0, 0, 0, 26, 27, 28, 29, 30, 31, 32, 33, 34,
+ 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 0,
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
+ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
+};
+
+char
+base64dec_getc(const char **src)
+{
+ while (**src && !isprint(**src)) (*src)++;
+ return *((*src)++);
+}
+
+char *
+base64dec(const char *src)
+{
+ size_t in_len = strlen(src);
+ char *result, *dst;
+
+ if (in_len % 4)
+ in_len += 4 - (in_len % 4);
+ result = dst = xmalloc(in_len / 4 * 3 + 1);
+ while (*src) {
+ int a = base64_digits[(unsigned char) base64dec_getc(&src)];
+ int b = base64_digits[(unsigned char) base64dec_getc(&src)];
+ int c = base64_digits[(unsigned char) base64dec_getc(&src)];
+ int d = base64_digits[(unsigned char) base64dec_getc(&src)];
+
+ *dst++ = (a << 2) | ((b & 0x30) >> 4);
+ if (c == -1)
+ break;
+ *dst++ = ((b & 0x0f) << 4) | ((c & 0x3c) >> 2);
+ if (d == -1)
+ break;
+ *dst++ = ((c & 0x03) << 6) | d;
+ }
+ *dst = '\0';
+ return result;
+}
+
+void
+selinit(void)
+{
+ sel.mode = SEL_IDLE;
+ sel.snap = 0;
+ sel.ob.x = -1;
+}
+
+int
+tlinelen(int y)
+{
+ int i = term.col;
+
+ if (TLINE(y)[i - 1].mode & ATTR_WRAP)
+ return i;
+
+ while (i > 0 && TLINE(y)[i - 1].u == ' ')
+ --i;
+
+ return i;
+}
+
+void
+selstart(int col, int row, int snap)
+{
+ selclear();
+ sel.mode = SEL_EMPTY;
+ sel.type = SEL_REGULAR;
+ sel.snap = snap;
+ sel.oe.x = sel.ob.x = col;
+ sel.oe.y = sel.ob.y = row;
+ selnormalize();
+
+ if (sel.snap != 0)
+ sel.mode = SEL_READY;
+ tsetdirt(sel.nb.y, sel.ne.y);
+}
+
+void
+selextend(int col, int row, int type, int done)
+{
+ int oldey, oldex, oldsby, oldsey, oldtype;
+
+ if (sel.mode == SEL_IDLE)
+ return;
+ if (done && sel.mode == SEL_EMPTY) {
+ selclear();
+ return;
+ }
+
+ oldey = sel.oe.y;
+ oldex = sel.oe.x;
+ oldsby = sel.nb.y;
+ oldsey = sel.ne.y;
+ oldtype = sel.type;
+
+ sel.alt = IS_SET(MODE_ALTSCREEN);
+ sel.oe.x = col;
+ sel.oe.y = row;
+ selnormalize();
+ sel.type = type;
+
+ if (oldey != sel.oe.y || oldex != sel.oe.x || oldtype != sel.type)
+ tsetdirt(MIN(sel.nb.y, oldsby), MAX(sel.ne.y, oldsey));
+
+ sel.mode = done ? SEL_IDLE : SEL_READY;
+}
+
+void
+selnormalize(void)
+{
+ int i;
+
+ if (sel.type == SEL_REGULAR && sel.ob.y != sel.oe.y) {
+ sel.nb.x = sel.ob.y < sel.oe.y ? sel.ob.x : sel.oe.x;
+ sel.ne.x = sel.ob.y < sel.oe.y ? sel.oe.x : sel.ob.x;
+ } else {
+ sel.nb.x = MIN(sel.ob.x, sel.oe.x);
+ sel.ne.x = MAX(sel.ob.x, sel.oe.x);
+ }
+ sel.nb.y = MIN(sel.ob.y, sel.oe.y);
+ sel.ne.y = MAX(sel.ob.y, sel.oe.y);
+
+ selsnap(&sel.nb.x, &sel.nb.y, -1);
+ selsnap(&sel.ne.x, &sel.ne.y, +1);
+
+ /* expand selection over line breaks */
+ if (sel.type == SEL_RECTANGULAR)
+ return;
+ i = tlinelen(sel.nb.y);
+ if (i < sel.nb.x)
+ sel.nb.x = i;
+ if (tlinelen(sel.ne.y) <= sel.ne.x)
+ sel.ne.x = term.col - 1;
+}
+
+int
+selected(int x, int y)
+{
+ if (sel.mode == SEL_EMPTY || sel.ob.x == -1 ||
+ sel.alt != IS_SET(MODE_ALTSCREEN))
+ return 0;
+
+ if (sel.type == SEL_RECTANGULAR)
+ return BETWEEN(y, sel.nb.y, sel.ne.y)
+ && BETWEEN(x, sel.nb.x, sel.ne.x);
+
+ return BETWEEN(y, sel.nb.y, sel.ne.y)
+ && (y != sel.nb.y || x >= sel.nb.x)
+ && (y != sel.ne.y || x <= sel.ne.x);
+}
+
+void
+selsnap(int *x, int *y, int direction)
+{
+ int newx, newy, xt, yt;
+ int delim, prevdelim;
+ Glyph *gp, *prevgp;
+
+ switch (sel.snap) {
+ case SNAP_WORD:
+ /*
+ * Snap around if the word wraps around at the end or
+ * beginning of a line.
+ */
+ prevgp = &TLINE(*y)[*x];
+ prevdelim = ISDELIM(prevgp->u);
+ for (;;) {
+ newx = *x + direction;
+ newy = *y;
+ if (!BETWEEN(newx, 0, term.col - 1)) {
+ newy += direction;
+ newx = (newx + term.col) % term.col;
+ if (!BETWEEN(newy, 0, term.row - 1))
+ break;
+
+ if (direction > 0)
+ yt = *y, xt = *x;
+ else
+ yt = newy, xt = newx;
+ if (!(TLINE(yt)[xt].mode & ATTR_WRAP))
+ break;
+ }
+
+ if (newx >= tlinelen(newy))
+ break;
+
+ gp = &TLINE(newy)[newx];
+ delim = ISDELIM(gp->u);
+ if (!(gp->mode & ATTR_WDUMMY) && (delim != prevdelim
+ || (delim && gp->u != prevgp->u)))
+ break;
+
+ *x = newx;
+ *y = newy;
+ prevgp = gp;
+ prevdelim = delim;
+ }
+ break;
+ case SNAP_LINE:
+ /*
+ * Snap around if the the previous line or the current one
+ * has set ATTR_WRAP at its end. Then the whole next or
+ * previous line will be selected.
+ */
+ *x = (direction < 0) ? 0 : term.col - 1;
+ if (direction < 0) {
+ for (; *y > 0; *y += direction) {
+ if (!(TLINE(*y-1)[term.col-1].mode
+ & ATTR_WRAP)) {
+ break;
+ }
+ }
+ } else if (direction > 0) {
+ for (; *y < term.row-1; *y += direction) {
+ if (!(TLINE(*y)[term.col-1].mode
+ & ATTR_WRAP)) {
+ break;
+ }
+ }
+ }
+ break;
+ }
+}
+
+char *
+getsel(void)
+{
+ char *str, *ptr;
+ int y, bufsize, lastx, linelen;
+ Glyph *gp, *last;
+
+ if (sel.ob.x == -1)
+ return NULL;
+
+ bufsize = (term.col+1) * (sel.ne.y-sel.nb.y+1) * UTF_SIZ;
+ ptr = str = xmalloc(bufsize);
+
+ /* append every set & selected glyph to the selection */
+ for (y = sel.nb.y; y <= sel.ne.y; y++) {
+ if ((linelen = tlinelen(y)) == 0) {
+ *ptr++ = '\n';
+ continue;
+ }
+
+ if (sel.type == SEL_RECTANGULAR) {
+ gp = &TLINE(y)[sel.nb.x];
+ lastx = sel.ne.x;
+ } else {
+ gp = &TLINE(y)[sel.nb.y == y ? sel.nb.x : 0];
+ lastx = (sel.ne.y == y) ? sel.ne.x : term.col-1;
+ }
+ last = &TLINE(y)[MIN(lastx, linelen-1)];
+ while (last >= gp && last->u == ' ')
+ --last;
+
+ for ( ; gp <= last; ++gp) {
+ if (gp->mode & ATTR_WDUMMY)
+ continue;
+
+ ptr += utf8encode(gp->u, ptr);
+ }
+
+ /*
+ * Copy and pasting of line endings is inconsistent
+ * in the inconsistent terminal and GUI world.
+ * The best solution seems like to produce '\n' when
+ * something is copied from st and convert '\n' to
+ * '\r', when something to be pasted is received by
+ * st.
+ * FIXME: Fix the computer world.
+ */
+ if ((y < sel.ne.y || lastx >= linelen) && !(last->mode & ATTR_WRAP))
+ *ptr++ = '\n';
+ }
+ *ptr = 0;
+ return str;
+}
+
+void
+selclear(void)
+{
+ if (sel.ob.x == -1)
+ return;
+ sel.mode = SEL_IDLE;
+ sel.ob.x = -1;
+ tsetdirt(sel.nb.y, sel.ne.y);
+}
+
+void
+die(const char *errstr, ...)
+{
+ va_list ap;
+
+ va_start(ap, errstr);
+ vfprintf(stderr, errstr, ap);
+ va_end(ap);
+ exit(1);
+}
+
+void
+execsh(char *cmd, char **args)
+{
+ char *sh, *prog;
+ const struct passwd *pw;
+
+ errno = 0;
+ if ((pw = getpwuid(getuid())) == NULL) {
+ if (errno)
+ die("getpwuid:%s\n", strerror(errno));
+ else
+ die("who are you?\n");
+ }
+
+ if ((sh = getenv("SHELL")) == NULL)
+ sh = (pw->pw_shell[0]) ? pw->pw_shell : cmd;
+
+ if (args)
+ prog = args[0];
+ else if (utmp)
+ prog = utmp;
+ else
+ prog = sh;
+ DEFAULT(args, ((char *[]) {prog, NULL}));
+
+ unsetenv("COLUMNS");
+ unsetenv("LINES");
+ unsetenv("TERMCAP");
+ setenv("LOGNAME", pw->pw_name, 1);
+ setenv("USER", pw->pw_name, 1);
+ setenv("SHELL", sh, 1);
+ setenv("HOME", pw->pw_dir, 1);
+ setenv("TERM", termname, 1);
+
+ signal(SIGCHLD, SIG_DFL);
+ signal(SIGHUP, SIG_DFL);
+ signal(SIGINT, SIG_DFL);
+ signal(SIGQUIT, SIG_DFL);
+ signal(SIGTERM, SIG_DFL);
+ signal(SIGALRM, SIG_DFL);
+
+ execvp(prog, args);
+ _exit(1);
+}
+
+void
+sigchld(int a)
+{
+ int stat;
+ pid_t p;
+
+ if ((p = waitpid(pid, &stat, WNOHANG)) < 0)
+ die("Waiting for pid %hd failed: %s\n", pid, strerror(errno));
+
+ if (pid != p)
+ return;
+
+ if (!WIFEXITED(stat) || WEXITSTATUS(stat))
+ die("child finished with error '%d'\n", stat);
+ exit(0);
+}
+
+void
+stty(char **args)
+{
+ char cmd[_POSIX_ARG_MAX], **p, *q, *s;
+ size_t n, siz;
+
+ if ((n = strlen(stty_args)) > sizeof(cmd)-1)
+ die("incorrect stty parameters\n");
+ memcpy(cmd, stty_args, n);
+ q = cmd + n;
+ siz = sizeof(cmd) - n;
+ for (p = args; p && (s = *p); ++p) {
+ if ((n = strlen(s)) > siz-1)
+ die("stty parameter length too long\n");
+ *q++ = ' ';
+ memcpy(q, s, n);
+ q += n;
+ siz -= n + 1;
+ }
+ *q = '\0';
+ if (system(cmd) != 0)
+ perror("Couldn't call stty");
+}
+
+int
+ttynew(char *line, char *cmd, char *out, char **args)
+{
+ int m, s;
+
+ if (out) {
+ term.mode |= MODE_PRINT;
+ iofd = (!strcmp(out, "-")) ?
+ 1 : open(out, O_WRONLY | O_CREAT, 0666);
+ if (iofd < 0) {
+ fprintf(stderr, "Error opening %s:%s\n",
+ out, strerror(errno));
+ }
+ }
+
+ if (line) {
+ if ((cmdfd = open(line, O_RDWR)) < 0)
+ die("open line failed: %s\n", strerror(errno));
+ dup2(cmdfd, 0);
+ stty(args);
+ return cmdfd;
+ }
+
+ /* seems to work fine on linux, openbsd and freebsd */
+ if (openpty(&m, &s, NULL, NULL, NULL) < 0)
+ die("openpty failed: %s\n", strerror(errno));
+
+ switch (pid = fork()) {
+ case -1:
+ die("fork failed\n");
+ break;
+ case 0:
+ close(iofd);
+ setsid(); /* create a new process group */
+ dup2(s, 0);
+ dup2(s, 1);
+ dup2(s, 2);
+ if (ioctl(s, TIOCSCTTY, NULL) < 0)
+ die("ioctl TIOCSCTTY failed: %s\n", strerror(errno));
+ close(s);
+ close(m);
+ execsh(cmd, args);
+ break;
+ default:
+ close(s);
+ cmdfd = m;
+ signal(SIGCHLD, sigchld);
+ break;
+ }
+ return cmdfd;
+}
+
+size_t
+ttyread(void)
+{
+ static char buf[BUFSIZ];
+ static int buflen = 0;
+ int written;
+ int ret;
+
+ /* append read bytes to unprocessed bytes */
+ if ((ret = read(cmdfd, buf+buflen, LEN(buf)-buflen)) < 0)
+ die("Couldn't read from shell: %s\n", strerror(errno));
+ buflen += ret;
+
+ written = twrite(buf, buflen, 0);
+ buflen -= written;
+ /* keep any uncomplete utf8 char for the next call */
+ if (buflen > 0)
+ memmove(buf, buf + written, buflen);
+
+ if (term.scr > 0 && term.scr < HISTSIZE-1)
+ term.scr++;
+
+ return ret;
+}
+
+void
+ttywrite(const char *s, size_t n, int may_echo)
+{
+ const char *next;
+ Arg arg = (Arg) { .i = term.scr };
+
+ kscrolldown(&arg);
+
+ if (may_echo && IS_SET(MODE_ECHO))
+ twrite(s, n, 1);
+
+ if (!IS_SET(MODE_CRLF)) {
+ ttywriteraw(s, n);
+ return;
+ }
+
+ /* This is similar to how the kernel handles ONLCR for ttys */
+ while (n > 0) {
+ if (*s == '\r') {
+ next = s + 1;
+ ttywriteraw("\r\n", 2);
+ } else {
+ next = memchr(s, '\r', n);
+ DEFAULT(next, s + n);
+ ttywriteraw(s, next - s);
+ }
+ n -= next - s;
+ s = next;
+ }
+}
+
+void
+ttywriteraw(const char *s, size_t n)
+{
+ fd_set wfd, rfd;
+ ssize_t r;
+ size_t lim = 256;
+
+ /*
+ * Remember that we are using a pty, which might be a modem line.
+ * Writing too much will clog the line. That's why we are doing this
+ * dance.
+ * FIXME: Migrate the world to Plan 9.
+ */
+ while (n > 0) {
+ FD_ZERO(&wfd);
+ FD_ZERO(&rfd);
+ FD_SET(cmdfd, &wfd);
+ FD_SET(cmdfd, &rfd);
+
+ /* Check if we can write. */
+ if (pselect(cmdfd+1, &rfd, &wfd, NULL, NULL, NULL) < 0) {
+ if (errno == EINTR)
+ continue;
+ die("select failed: %s\n", strerror(errno));
+ }
+ if (FD_ISSET(cmdfd, &wfd)) {
+ /*
+ * Only write the bytes written by ttywrite() or the
+ * default of 256. This seems to be a reasonable value
+ * for a serial line. Bigger values might clog the I/O.
+ */
+ if ((r = write(cmdfd, s, (n < lim)? n : lim)) < 0)
+ goto write_error;
+ if (r < n) {
+ /*
+ * We weren't able to write out everything.
+ * This means the buffer is getting full
+ * again. Empty it.
+ */
+ if (n < lim)
+ lim = ttyread();
+ n -= r;
+ s += r;
+ } else {
+ /* All bytes have been written. */
+ break;
+ }
+ }
+ if (FD_ISSET(cmdfd, &rfd))
+ lim = ttyread();
+ }
+ return;
+
+write_error:
+ die("write error on tty: %s\n", strerror(errno));
+}
+
+void
+ttyresize(int tw, int th)
+{
+ struct winsize w;
+
+ w.ws_row = term.row;
+ w.ws_col = term.col;
+ w.ws_xpixel = tw;
+ w.ws_ypixel = th;
+ if (ioctl(cmdfd, TIOCSWINSZ, &w) < 0)
+ fprintf(stderr, "Couldn't set window size: %s\n", strerror(errno));
+}
+
+void
+ttyhangup()
+{
+ /* Send SIGHUP to shell */
+ kill(pid, SIGHUP);
+}
+
+int
+tattrset(int attr)
+{
+ int i, j;
+
+ for (i = 0; i < term.row-1; i++) {
+ for (j = 0; j < term.col-1; j++) {
+ if (term.line[i][j].mode & attr)
+ return 1;
+ }
+ }
+
+ return 0;
+}
+
+void
+tsetdirt(int top, int bot)
+{
+ int i;
+
+ LIMIT(top, 0, term.row-1);
+ LIMIT(bot, 0, term.row-1);
+
+ for (i = top; i <= bot; i++)
+ term.dirty[i] = 1;
+}
+
+void
+tsetdirtattr(int attr)
+{
+ int i, j;
+
+ for (i = 0; i < term.row-1; i++) {
+ for (j = 0; j < term.col-1; j++) {
+ if (term.line[i][j].mode & attr) {
+ tsetdirt(i, i);
+ break;
+ }
+ }
+ }
+}
+
+void
+tfulldirt(void)
+{
+ tsetdirt(0, term.row-1);
+}
+
+void
+tcursor(int mode)
+{
+ static TCursor c[2];
+ int alt = IS_SET(MODE_ALTSCREEN);
+
+ if (mode == CURSOR_SAVE) {
+ c[alt] = term.c;
+ } else if (mode == CURSOR_LOAD) {
+ term.c = c[alt];
+ tmoveto(c[alt].x, c[alt].y);
+ }
+}
+
+void
+treset(void)
+{
+ uint i;
+
+ term.c = (TCursor){{
+ .mode = ATTR_NULL,
+ .fg = defaultfg,
+ .bg = defaultbg
+ }, .x = 0, .y = 0, .state = CURSOR_DEFAULT};
+
+ memset(term.tabs, 0, term.col * sizeof(*term.tabs));
+ for (i = tabspaces; i < term.col; i += tabspaces)
+ term.tabs[i] = 1;
+ term.top = 0;
+ term.bot = term.row - 1;
+ term.mode = MODE_WRAP|MODE_UTF8;
+ memset(term.trantbl, CS_USA, sizeof(term.trantbl));
+ term.charset = 0;
+
+ for (i = 0; i < 2; i++) {
+ tmoveto(0, 0);
+ tcursor(CURSOR_SAVE);
+ tclearregion(0, 0, term.col-1, term.row-1);
+ tswapscreen();
+ }
+}
+
+void
+tnew(int col, int row)
+{
+ term = (Term){ .c = { .attr = { .fg = defaultfg, .bg = defaultbg } } };
+ tresize(col, row);
+ treset();
+}
+
+void
+tswapscreen(void)
+{
+ Line *tmp = term.line;
+
+ term.line = term.alt;
+ term.alt = tmp;
+ term.mode ^= MODE_ALTSCREEN;
+ tfulldirt();
+}
+
+void
+kscrolldown(const Arg* a)
+{
+ int n = a->i;
+
+ if (n < 0)
+ n = term.row + n;
+
+ if (n > term.scr)
+ n = term.scr;
+
+ if (term.scr > 0) {
+ term.scr -= n;
+ selscroll(0, -n);
+ tfulldirt();
+ }
+}
+
+void
+kscrollup(const Arg* a)
+{
+ int n = a->i;
+
+ if (n < 0)
+ n = term.row + n;
+
+ if (term.scr <= HISTSIZE-n) {
+ term.scr += n;
+ selscroll(0, n);
+ tfulldirt();
+ }
+}
+
+
+void
+tscrolldown(int orig, int n, int copyhist)
+{
+ int i;
+ Line temp;
+
+ LIMIT(n, 0, term.bot-orig+1);
+
+ if (copyhist) {
+ term.histi = (term.histi - 1 + HISTSIZE) % HISTSIZE;
+ temp = term.hist[term.histi];
+ term.hist[term.histi] = term.line[term.bot];
+ term.line[term.bot] = temp;
+ }
+
+ tsetdirt(orig, term.bot-n);
+ tclearregion(0, term.bot-n+1, term.col-1, term.bot);
+
+ for (i = term.bot; i >= orig+n; i--) {
+ temp = term.line[i];
+ term.line[i] = term.line[i-n];
+ term.line[i-n] = temp;
+ }
+
+ selscroll(orig, n);
+}
+
+void
+tscrollup(int orig, int n, int copyhist)
+{
+ int i;
+ Line temp;
+
+ LIMIT(n, 0, term.bot-orig+1);
+
+ if (copyhist) {
+ term.histi = (term.histi + 1) % HISTSIZE;
+ temp = term.hist[term.histi];
+ term.hist[term.histi] = term.line[orig];
+ term.line[orig] = temp;
+ }
+
+ tclearregion(0, orig, term.col-1, orig+n-1);
+ tsetdirt(orig+n, term.bot);
+
+ for (i = orig; i <= term.bot-n; i++) {
+ temp = term.line[i];
+ term.line[i] = term.line[i+n];
+ term.line[i+n] = temp;
+ }
+
+ selscroll(orig, -n);
+}
+
+void
+selscroll(int orig, int n)
+{
+ if (sel.ob.x == -1)
+ return;
+
+ if (BETWEEN(sel.ob.y, orig, term.bot) || BETWEEN(sel.oe.y, orig, term.bot)) {
+ if ((sel.ob.y += n) > term.bot || (sel.oe.y += n) < term.top) {
+ selclear();
+ return;
+ }
+ if (sel.type == SEL_RECTANGULAR) {
+ if (sel.ob.y < term.top)
+ sel.ob.y = term.top;
+ if (sel.oe.y > term.bot)
+ sel.oe.y = term.bot;
+ } else {
+ if (sel.ob.y < term.top) {
+ sel.ob.y = term.top;
+ sel.ob.x = 0;
+ }
+ if (sel.oe.y > term.bot) {
+ sel.oe.y = term.bot;
+ sel.oe.x = term.col;
+ }
+ }
+ selnormalize();
+ }
+}
+
+void
+tnewline(int first_col)
+{
+ int y = term.c.y;
+
+ if (y == term.bot) {
+ tscrollup(term.top, 1, 1);
+ } else {
+ y++;
+ }
+ tmoveto(first_col ? 0 : term.c.x, y);
+}
+
+void
+csiparse(void)
+{
+ char *p = csiescseq.buf, *np;
+ long int v;
+
+ csiescseq.narg = 0;
+ if (*p == '?') {
+ csiescseq.priv = 1;
+ p++;
+ }
+
+ csiescseq.buf[csiescseq.len] = '\0';
+ while (p < csiescseq.buf+csiescseq.len) {
+ np = NULL;
+ v = strtol(p, &np, 10);
+ if (np == p)
+ v = 0;
+ if (v == LONG_MAX || v == LONG_MIN)
+ v = -1;
+ csiescseq.arg[csiescseq.narg++] = v;
+ p = np;
+ if (*p != ';' || csiescseq.narg == ESC_ARG_SIZ)
+ break;
+ p++;
+ }
+ csiescseq.mode[0] = *p++;
+ csiescseq.mode[1] = (p < csiescseq.buf+csiescseq.len) ? *p : '\0';
+}
+
+/* for absolute user moves, when decom is set */
+void
+tmoveato(int x, int y)
+{
+ tmoveto(x, y + ((term.c.state & CURSOR_ORIGIN) ? term.top: 0));
+}
+
+void
+tmoveto(int x, int y)
+{
+ int miny, maxy;
+
+ if (term.c.state & CURSOR_ORIGIN) {
+ miny = term.top;
+ maxy = term.bot;
+ } else {
+ miny = 0;
+ maxy = term.row - 1;
+ }
+ term.c.state &= ~CURSOR_WRAPNEXT;
+ term.c.x = LIMIT(x, 0, term.col-1);
+ term.c.y = LIMIT(y, miny, maxy);
+}
+
+void
+tsetchar(Rune u, Glyph *attr, int x, int y)
+{
+ static char *vt100_0[62] = { /* 0x41 - 0x7e */
+ "↑", "↓", "→", "←", "█", "▚", "☃", /* A - G */
+ 0, 0, 0, 0, 0, 0, 0, 0, /* H - O */
+ 0, 0, 0, 0, 0, 0, 0, 0, /* P - W */
+ 0, 0, 0, 0, 0, 0, 0, " ", /* X - _ */
+ "◆", "▒", "␉", "␌", "␍", "␊", "°", "±", /* ` - g */
+ "", "␋", "┘", "┐", "┌", "└", "┼", "⎺", /* h - o */
+ "⎻", "─", "⎼", "⎽", "├", "┤", "┴", "┬", /* p - w */
+ "│", "≤", "≥", "π", "≠", "£", "·", /* x - ~ */
+ };
+
+ /*
+ * The table is proudly stolen from rxvt.
+ */
+ if (term.trantbl[term.charset] == CS_GRAPHIC0 &&
+ BETWEEN(u, 0x41, 0x7e) && vt100_0[u - 0x41])
+ utf8decode(vt100_0[u - 0x41], &u, UTF_SIZ);
+
+ if (term.line[y][x].mode & ATTR_WIDE) {
+ if (x+1 < term.col) {
+ term.line[y][x+1].u = ' ';
+ term.line[y][x+1].mode &= ~ATTR_WDUMMY;
+ }
+ } else if (term.line[y][x].mode & ATTR_WDUMMY) {
+ term.line[y][x-1].u = ' ';
+ term.line[y][x-1].mode &= ~ATTR_WIDE;
+ }
+
+ term.dirty[y] = 1;
+ term.line[y][x] = *attr;
+ term.line[y][x].u = u;
+}
+
+void
+tclearregion(int x1, int y1, int x2, int y2)
+{
+ int x, y, temp;
+ Glyph *gp;
+
+ if (x1 > x2)
+ temp = x1, x1 = x2, x2 = temp;
+ if (y1 > y2)
+ temp = y1, y1 = y2, y2 = temp;
+
+ LIMIT(x1, 0, term.col-1);
+ LIMIT(x2, 0, term.col-1);
+ LIMIT(y1, 0, term.row-1);
+ LIMIT(y2, 0, term.row-1);
+
+ for (y = y1; y <= y2; y++) {
+ term.dirty[y] = 1;
+ for (x = x1; x <= x2; x++) {
+ gp = &term.line[y][x];
+ if (selected(x, y))
+ selclear();
+ gp->fg = term.c.attr.fg;
+ gp->bg = term.c.attr.bg;
+ gp->mode = 0;
+ gp->u = ' ';
+ }
+ }
+}
+
+void
+tdeletechar(int n)
+{
+ int dst, src, size;
+ Glyph *line;
+
+ LIMIT(n, 0, term.col - term.c.x);
+
+ dst = term.c.x;
+ src = term.c.x + n;
+ size = term.col - src;
+ line = term.line[term.c.y];
+
+ memmove(&line[dst], &line[src], size * sizeof(Glyph));
+ tclearregion(term.col-n, term.c.y, term.col-1, term.c.y);
+}
+
+void
+tinsertblank(int n)
+{
+ int dst, src, size;
+ Glyph *line;
+
+ LIMIT(n, 0, term.col - term.c.x);
+
+ dst = term.c.x + n;
+ src = term.c.x;
+ size = term.col - dst;
+ line = term.line[term.c.y];
+
+ memmove(&line[dst], &line[src], size * sizeof(Glyph));
+ tclearregion(src, term.c.y, dst - 1, term.c.y);
+}
+
+void
+tinsertblankline(int n)
+{
+ if (BETWEEN(term.c.y, term.top, term.bot))
+ tscrolldown(term.c.y, n, 0);
+}
+
+void
+tdeleteline(int n)
+{
+ if (BETWEEN(term.c.y, term.top, term.bot))
+ tscrollup(term.c.y, n, 0);
+}
+
+int32_t
+tdefcolor(int *attr, int *npar, int l)
+{
+ int32_t idx = -1;
+ uint r, g, b;
+
+ switch (attr[*npar + 1]) {
+ case 2: /* direct color in RGB space */
+ if (*npar + 4 >= l) {
+ fprintf(stderr,
+ "erresc(38): Incorrect number of parameters (%d)\n",
+ *npar);
+ break;
+ }
+ r = attr[*npar + 2];
+ g = attr[*npar + 3];
+ b = attr[*npar + 4];
+ *npar += 4;
+ if (!BETWEEN(r, 0, 255) || !BETWEEN(g, 0, 255) || !BETWEEN(b, 0, 255))
+ fprintf(stderr, "erresc: bad rgb color (%u,%u,%u)\n",
+ r, g, b);
+ else
+ idx = TRUECOLOR(r, g, b);
+ break;
+ case 5: /* indexed color */
+ if (*npar + 2 >= l) {
+ fprintf(stderr,
+ "erresc(38): Incorrect number of parameters (%d)\n",
+ *npar);
+ break;
+ }
+ *npar += 2;
+ if (!BETWEEN(attr[*npar], 0, 255))
+ fprintf(stderr, "erresc: bad fgcolor %d\n", attr[*npar]);
+ else
+ idx = attr[*npar];
+ break;
+ case 0: /* implemented defined (only foreground) */
+ case 1: /* transparent */
+ case 3: /* direct color in CMY space */
+ case 4: /* direct color in CMYK space */
+ default:
+ fprintf(stderr,
+ "erresc(38): gfx attr %d unknown\n", attr[*npar]);
+ break;
+ }
+
+ return idx;
+}
+
+void
+tsetattr(int *attr, int l)
+{
+ int i;
+ int32_t idx;
+
+ for (i = 0; i < l; i++) {
+ switch (attr[i]) {
+ case 0:
+ term.c.attr.mode &= ~(
+ ATTR_BOLD |
+ ATTR_FAINT |
+ ATTR_ITALIC |
+ ATTR_UNDERLINE |
+ ATTR_BLINK |
+ ATTR_REVERSE |
+ ATTR_INVISIBLE |
+ ATTR_STRUCK );
+ term.c.attr.fg = defaultfg;
+ term.c.attr.bg = defaultbg;
+ break;
+ case 1:
+ term.c.attr.mode |= ATTR_BOLD;
+ break;
+ case 2:
+ term.c.attr.mode |= ATTR_FAINT;
+ break;
+ case 3:
+ term.c.attr.mode |= ATTR_ITALIC;
+ break;
+ case 4:
+ term.c.attr.mode |= ATTR_UNDERLINE;
+ break;
+ case 5: /* slow blink */
+ /* FALLTHROUGH */
+ case 6: /* rapid blink */
+ term.c.attr.mode |= ATTR_BLINK;
+ break;
+ case 7:
+ term.c.attr.mode |= ATTR_REVERSE;
+ break;
+ case 8:
+ term.c.attr.mode |= ATTR_INVISIBLE;
+ break;
+ case 9:
+ term.c.attr.mode |= ATTR_STRUCK;
+ break;
+ case 22:
+ term.c.attr.mode &= ~(ATTR_BOLD | ATTR_FAINT);
+ break;
+ case 23:
+ term.c.attr.mode &= ~ATTR_ITALIC;
+ break;
+ case 24:
+ term.c.attr.mode &= ~ATTR_UNDERLINE;
+ break;
+ case 25:
+ term.c.attr.mode &= ~ATTR_BLINK;
+ break;
+ case 27:
+ term.c.attr.mode &= ~ATTR_REVERSE;
+ break;
+ case 28:
+ term.c.attr.mode &= ~ATTR_INVISIBLE;
+ break;
+ case 29:
+ term.c.attr.mode &= ~ATTR_STRUCK;
+ break;
+ case 38:
+ if ((idx = tdefcolor(attr, &i, l)) >= 0)
+ term.c.attr.fg = idx;
+ break;
+ case 39:
+ term.c.attr.fg = defaultfg;
+ break;
+ case 48:
+ if ((idx = tdefcolor(attr, &i, l)) >= 0)
+ term.c.attr.bg = idx;
+ break;
+ case 49:
+ term.c.attr.bg = defaultbg;
+ break;
+ default:
+ if (BETWEEN(attr[i], 30, 37)) {
+ term.c.attr.fg = attr[i] - 30;
+ } else if (BETWEEN(attr[i], 40, 47)) {
+ term.c.attr.bg = attr[i] - 40;
+ } else if (BETWEEN(attr[i], 90, 97)) {
+ term.c.attr.fg = attr[i] - 90 + 8;
+ } else if (BETWEEN(attr[i], 100, 107)) {
+ term.c.attr.bg = attr[i] - 100 + 8;
+ } else {
+ fprintf(stderr,
+ "erresc(default): gfx attr %d unknown\n",
+ attr[i]), csidump();
+ }
+ break;
+ }
+ }
+}
+
+void
+tsetscroll(int t, int b)
+{
+ int temp;
+
+ LIMIT(t, 0, term.row-1);
+ LIMIT(b, 0, term.row-1);
+ if (t > b) {
+ temp = t;
+ t = b;
+ b = temp;
+ }
+ term.top = t;
+ term.bot = b;
+}
+
+void
+tsetmode(int priv, int set, int *args, int narg)
+{
+ int alt, *lim;
+
+ for (lim = args + narg; args < lim; ++args) {
+ if (priv) {
+ switch (*args) {
+ case 1: /* DECCKM -- Cursor key */
+ xsetmode(set, MODE_APPCURSOR);
+ break;
+ case 5: /* DECSCNM -- Reverse video */
+ xsetmode(set, MODE_REVERSE);
+ break;
+ case 6: /* DECOM -- Origin */
+ MODBIT(term.c.state, set, CURSOR_ORIGIN);
+ tmoveato(0, 0);
+ break;
+ case 7: /* DECAWM -- Auto wrap */
+ MODBIT(term.mode, set, MODE_WRAP);
+ break;
+ case 0: /* Error (IGNORED) */
+ case 2: /* DECANM -- ANSI/VT52 (IGNORED) */
+ case 3: /* DECCOLM -- Column (IGNORED) */
+ case 4: /* DECSCLM -- Scroll (IGNORED) */
+ case 8: /* DECARM -- Auto repeat (IGNORED) */
+ case 18: /* DECPFF -- Printer feed (IGNORED) */
+ case 19: /* DECPEX -- Printer extent (IGNORED) */
+ case 42: /* DECNRCM -- National characters (IGNORED) */
+ case 12: /* att610 -- Start blinking cursor (IGNORED) */
+ break;
+ case 25: /* DECTCEM -- Text Cursor Enable Mode */
+ xsetmode(!set, MODE_HIDE);
+ break;
+ case 9: /* X10 mouse compatibility mode */
+ xsetpointermotion(0);
+ xsetmode(0, MODE_MOUSE);
+ xsetmode(set, MODE_MOUSEX10);
+ break;
+ case 1000: /* 1000: report button press */
+ xsetpointermotion(0);
+ xsetmode(0, MODE_MOUSE);
+ xsetmode(set, MODE_MOUSEBTN);
+ break;
+ case 1002: /* 1002: report motion on button press */
+ xsetpointermotion(0);
+ xsetmode(0, MODE_MOUSE);
+ xsetmode(set, MODE_MOUSEMOTION);
+ break;
+ case 1003: /* 1003: enable all mouse motions */
+ xsetpointermotion(set);
+ xsetmode(0, MODE_MOUSE);
+ xsetmode(set, MODE_MOUSEMANY);
+ break;
+ case 1004: /* 1004: send focus events to tty */
+ xsetmode(set, MODE_FOCUS);
+ break;
+ case 1006: /* 1006: extended reporting mode */
+ xsetmode(set, MODE_MOUSESGR);
+ break;
+ case 1034:
+ xsetmode(set, MODE_8BIT);
+ break;
+ case 1049: /* swap screen & set/restore cursor as xterm */
+ if (!allowaltscreen)
+ break;
+ tcursor((set) ? CURSOR_SAVE : CURSOR_LOAD);
+ /* FALLTHROUGH */
+ case 47: /* swap screen */
+ case 1047:
+ if (!allowaltscreen)
+ break;
+ alt = IS_SET(MODE_ALTSCREEN);
+ if (alt) {
+ tclearregion(0, 0, term.col-1,
+ term.row-1);
+ }
+ if (set ^ alt) /* set is always 1 or 0 */
+ tswapscreen();
+ if (*args != 1049)
+ break;
+ /* FALLTHROUGH */
+ case 1048:
+ tcursor((set) ? CURSOR_SAVE : CURSOR_LOAD);
+ break;
+ case 2004: /* 2004: bracketed paste mode */
+ xsetmode(set, MODE_BRCKTPASTE);
+ break;
+ /* Not implemented mouse modes. See comments there. */
+ case 1001: /* mouse highlight mode; can hang the
+ terminal by design when implemented. */
+ case 1005: /* UTF-8 mouse mode; will confuse
+ applications not supporting UTF-8
+ and luit. */
+ case 1015: /* urxvt mangled mouse mode; incompatible
+ and can be mistaken for other control
+ codes. */
+ default:
+ fprintf(stderr,
+ "erresc: unknown private set/reset mode %d\n",
+ *args);
+ break;
+ }
+ } else {
+ switch (*args) {
+ case 0: /* Error (IGNORED) */
+ break;
+ case 2:
+ xsetmode(set, MODE_KBDLOCK);
+ break;
+ case 4: /* IRM -- Insertion-replacement */
+ MODBIT(term.mode, set, MODE_INSERT);
+ break;
+ case 12: /* SRM -- Send/Receive */
+ MODBIT(term.mode, !set, MODE_ECHO);
+ break;
+ case 20: /* LNM -- Linefeed/new line */
+ MODBIT(term.mode, set, MODE_CRLF);
+ break;
+ default:
+ fprintf(stderr,
+ "erresc: unknown set/reset mode %d\n",
+ *args);
+ break;
+ }
+ }
+ }
+}
+
+void
+csihandle(void)
+{
+ char buf[40];
+ int len;
+
+ switch (csiescseq.mode[0]) {
+ default:
+ unknown:
+ fprintf(stderr, "erresc: unknown csi ");
+ csidump();
+ /* die(""); */
+ break;
+ case '@': /* ICH -- Insert blank char */
+ DEFAULT(csiescseq.arg[0], 1);
+ tinsertblank(csiescseq.arg[0]);
+ break;
+ case 'A': /* CUU -- Cursor Up */
+ DEFAULT(csiescseq.arg[0], 1);
+ tmoveto(term.c.x, term.c.y-csiescseq.arg[0]);
+ break;
+ case 'B': /* CUD -- Cursor Down */
+ case 'e': /* VPR --Cursor Down */
+ DEFAULT(csiescseq.arg[0], 1);
+ tmoveto(term.c.x, term.c.y+csiescseq.arg[0]);
+ break;
+ case 'i': /* MC -- Media Copy */
+ switch (csiescseq.arg[0]) {
+ case 0:
+ tdump();
+ break;
+ case 1:
+ tdumpline(term.c.y);
+ break;
+ case 2:
+ tdumpsel();
+ break;
+ case 4:
+ term.mode &= ~MODE_PRINT;
+ break;
+ case 5:
+ term.mode |= MODE_PRINT;
+ break;
+ }
+ break;
+ case 'c': /* DA -- Device Attributes */
+ if (csiescseq.arg[0] == 0)
+ ttywrite(vtiden, strlen(vtiden), 0);
+ break;
+ case 'C': /* CUF -- Cursor Forward */
+ case 'a': /* HPR -- Cursor Forward */
+ DEFAULT(csiescseq.arg[0], 1);
+ tmoveto(term.c.x+csiescseq.arg[0], term.c.y);
+ break;
+ case 'D': /* CUB -- Cursor Backward */
+ DEFAULT(csiescseq.arg[0], 1);
+ tmoveto(term.c.x-csiescseq.arg[0], term.c.y);
+ break;
+ case 'E': /* CNL -- Cursor Down and first col */
+ DEFAULT(csiescseq.arg[0], 1);
+ tmoveto(0, term.c.y+csiescseq.arg[0]);
+ break;
+ case 'F': /* CPL -- Cursor Up and first col */
+ DEFAULT(csiescseq.arg[0], 1);
+ tmoveto(0, term.c.y-csiescseq.arg[0]);
+ break;
+ case 'g': /* TBC -- Tabulation clear */
+ switch (csiescseq.arg[0]) {
+ case 0: /* clear current tab stop */
+ term.tabs[term.c.x] = 0;
+ break;
+ case 3: /* clear all the tabs */
+ memset(term.tabs, 0, term.col * sizeof(*term.tabs));
+ break;
+ default:
+ goto unknown;
+ }
+ break;
+ case 'G': /* CHA -- Move to */
+ case '`': /* HPA */
+ DEFAULT(csiescseq.arg[0], 1);
+ tmoveto(csiescseq.arg[0]-1, term.c.y);
+ break;
+ case 'H': /* CUP -- Move to */
+ case 'f': /* HVP */
+ DEFAULT(csiescseq.arg[0], 1);
+ DEFAULT(csiescseq.arg[1], 1);
+ tmoveato(csiescseq.arg[1]-1, csiescseq.arg[0]-1);
+ break;
+ case 'I': /* CHT -- Cursor Forward Tabulation tab stops */
+ DEFAULT(csiescseq.arg[0], 1);
+ tputtab(csiescseq.arg[0]);
+ break;
+ case 'J': /* ED -- Clear screen */
+ switch (csiescseq.arg[0]) {
+ case 0: /* below */
+ tclearregion(term.c.x, term.c.y, term.col-1, term.c.y);
+ if (term.c.y < term.row-1) {
+ tclearregion(0, term.c.y+1, term.col-1,
+ term.row-1);
+ }
+ break;
+ case 1: /* above */
+ if (term.c.y > 1)
+ tclearregion(0, 0, term.col-1, term.c.y-1);
+ tclearregion(0, term.c.y, term.c.x, term.c.y);
+ break;
+ case 2: /* all */
+ tclearregion(0, 0, term.col-1, term.row-1);
+ break;
+ default:
+ goto unknown;
+ }
+ break;
+ case 'K': /* EL -- Clear line */
+ switch (csiescseq.arg[0]) {
+ case 0: /* right */
+ tclearregion(term.c.x, term.c.y, term.col-1,
+ term.c.y);
+ break;
+ case 1: /* left */
+ tclearregion(0, term.c.y, term.c.x, term.c.y);
+ break;
+ case 2: /* all */
+ tclearregion(0, term.c.y, term.col-1, term.c.y);
+ break;
+ }
+ break;
+ case 'S': /* SU -- Scroll line up */
+ DEFAULT(csiescseq.arg[0], 1);
+ tscrollup(term.top, csiescseq.arg[0], 0);
+ break;
+ case 'T': /* SD -- Scroll line down */
+ DEFAULT(csiescseq.arg[0], 1);
+ tscrolldown(term.top, csiescseq.arg[0], 0);
+ break;
+ case 'L': /* IL -- Insert blank lines */
+ DEFAULT(csiescseq.arg[0], 1);
+ tinsertblankline(csiescseq.arg[0]);
+ break;
+ case 'l': /* RM -- Reset Mode */
+ tsetmode(csiescseq.priv, 0, csiescseq.arg, csiescseq.narg);
+ break;
+ case 'M': /* DL -- Delete lines */
+ DEFAULT(csiescseq.arg[0], 1);
+ tdeleteline(csiescseq.arg[0]);
+ break;
+ case 'X': /* ECH -- Erase char */
+ DEFAULT(csiescseq.arg[0], 1);
+ tclearregion(term.c.x, term.c.y,
+ term.c.x + csiescseq.arg[0] - 1, term.c.y);
+ break;
+ case 'P': /* DCH -- Delete char */
+ DEFAULT(csiescseq.arg[0], 1);
+ tdeletechar(csiescseq.arg[0]);
+ break;
+ case 'Z': /* CBT -- Cursor Backward Tabulation tab stops */
+ DEFAULT(csiescseq.arg[0], 1);
+ tputtab(-csiescseq.arg[0]);
+ break;
+ case 'd': /* VPA -- Move to */
+ DEFAULT(csiescseq.arg[0], 1);
+ tmoveato(term.c.x, csiescseq.arg[0]-1);
+ break;
+ case 'h': /* SM -- Set terminal mode */
+ tsetmode(csiescseq.priv, 1, csiescseq.arg, csiescseq.narg);
+ break;
+ case 'm': /* SGR -- Terminal attribute (color) */
+ tsetattr(csiescseq.arg, csiescseq.narg);
+ break;
+ case 'n': /* DSR – Device Status Report (cursor position) */
+ if (csiescseq.arg[0] == 6) {
+ len = snprintf(buf, sizeof(buf),"\033[%i;%iR",
+ term.c.y+1, term.c.x+1);
+ ttywrite(buf, len, 0);
+ }
+ break;
+ case 'r': /* DECSTBM -- Set Scrolling Region */
+ if (csiescseq.priv) {
+ goto unknown;
+ } else {
+ DEFAULT(csiescseq.arg[0], 1);
+ DEFAULT(csiescseq.arg[1], term.row);
+ tsetscroll(csiescseq.arg[0]-1, csiescseq.arg[1]-1);
+ tmoveato(0, 0);
+ }
+ break;
+ case 's': /* DECSC -- Save cursor position (ANSI.SYS) */
+ tcursor(CURSOR_SAVE);
+ break;
+ case 'u': /* DECRC -- Restore cursor position (ANSI.SYS) */
+ tcursor(CURSOR_LOAD);
+ break;
+ case ' ':
+ switch (csiescseq.mode[1]) {
+ case 'q': /* DECSCUSR -- Set Cursor Style */
+ if (xsetcursor(csiescseq.arg[0]))
+ goto unknown;
+ break;
+ default:
+ goto unknown;
+ }
+ break;
+ }
+}
+
+void
+csidump(void)
+{
+ int i;
+ uint c;
+
+ fprintf(stderr, "ESC[");
+ for (i = 0; i < csiescseq.len; i++) {
+ c = csiescseq.buf[i] & 0xff;
+ if (isprint(c)) {
+ putc(c, stderr);
+ } else if (c == '\n') {
+ fprintf(stderr, "(\\n)");
+ } else if (c == '\r') {
+ fprintf(stderr, "(\\r)");
+ } else if (c == 0x1b) {
+ fprintf(stderr, "(\\e)");
+ } else {
+ fprintf(stderr, "(%02x)", c);
+ }
+ }
+ putc('\n', stderr);
+}
+
+void
+csireset(void)
+{
+ memset(&csiescseq, 0, sizeof(csiescseq));
+}
+
+void
+strhandle(void)
+{
+ char *p = NULL;
+ int j, narg, par;
+
+ term.esc &= ~(ESC_STR_END|ESC_STR);
+ strparse();
+ par = (narg = strescseq.narg) ? atoi(strescseq.args[0]) : 0;
+
+ switch (strescseq.type) {
+ case ']': /* OSC -- Operating System Command */
+ switch (par) {
+ case 0:
+ case 1:
+ case 2:
+ if (narg > 1)
+ xsettitle(strescseq.args[1]);
+ return;
+ case 52:
+ if (narg > 2) {
+ char *dec;
+
+ dec = base64dec(strescseq.args[2]);
+ if (dec) {
+ xsetsel(dec);
+ xclipcopy();
+ } else {
+ fprintf(stderr, "erresc: invalid base64\n");
+ }
+ }
+ return;
+ case 4: /* color set */
+ if (narg < 3)
+ break;
+ p = strescseq.args[2];
+ /* FALLTHROUGH */
+ case 104: /* color reset, here p = NULL */
+ j = (narg > 1) ? atoi(strescseq.args[1]) : -1;
+ if (xsetcolorname(j, p)) {
+ fprintf(stderr, "erresc: invalid color %s\n", p);
+ } else {
+ /*
+ * TODO if defaultbg color is changed, borders
+ * are dirty
+ */
+ redraw();
+ }
+ return;
+ }
+ break;
+ case 'k': /* old title set compatibility */
+ xsettitle(strescseq.args[0]);
+ return;
+ case 'P': /* DCS -- Device Control String */
+ term.mode |= ESC_DCS;
+ case '_': /* APC -- Application Program Command */
+ case '^': /* PM -- Privacy Message */
+ return;
+ }
+
+ fprintf(stderr, "erresc: unknown str ");
+ strdump();
+}
+
+void
+strparse(void)
+{
+ int c;
+ char *p = strescseq.buf;
+
+ strescseq.narg = 0;
+ strescseq.buf[strescseq.len] = '\0';
+
+ if (*p == '\0')
+ return;
+
+ while (strescseq.narg < STR_ARG_SIZ) {
+ strescseq.args[strescseq.narg++] = p;
+ while ((c = *p) != ';' && c != '\0')
+ ++p;
+ if (c == '\0')
+ return;
+ *p++ = '\0';
+ }
+}
+
+void
+strdump(void)
+{
+ int i;
+ uint c;
+
+ fprintf(stderr, "ESC%c", strescseq.type);
+ for (i = 0; i < strescseq.len; i++) {
+ c = strescseq.buf[i] & 0xff;
+ if (c == '\0') {
+ putc('\n', stderr);
+ return;
+ } else if (isprint(c)) {
+ putc(c, stderr);
+ } else if (c == '\n') {
+ fprintf(stderr, "(\\n)");
+ } else if (c == '\r') {
+ fprintf(stderr, "(\\r)");
+ } else if (c == 0x1b) {
+ fprintf(stderr, "(\\e)");
+ } else {
+ fprintf(stderr, "(%02x)", c);
+ }
+ }
+ fprintf(stderr, "ESC\\\n");
+}
+
+void
+strreset(void)
+{
+ memset(&strescseq, 0, sizeof(strescseq));
+}
+
+void
+sendbreak(const Arg *arg)
+{
+ if (tcsendbreak(cmdfd, 0))
+ perror("Error sending break");
+}
+
+void
+tprinter(char *s, size_t len)
+{
+ if (iofd != -1 && xwrite(iofd, s, len) < 0) {
+ perror("Error writing to output file");
+ close(iofd);
+ iofd = -1;
+ }
+}
+
+void
+iso14755(const Arg *arg)
+{
+ FILE *p;
+ char *us, *e, codepoint[9], uc[UTF_SIZ];
+ unsigned long utf32;
+
+ if (!(p = popen(ISO14755CMD, "r")))
+ return;
+
+ us = fgets(codepoint, sizeof(codepoint), p);
+ pclose(p);
+
+ if (!us || *us == '\0' || *us == '-' || strlen(us) > 7)
+ return;
+ if ((utf32 = strtoul(us, &e, 16)) == ULONG_MAX ||
+ (*e != '\n' && *e != '\0'))
+ return;
+
+ ttywrite(uc, utf8encode(utf32, uc), 1);
+}
+
+void
+toggleprinter(const Arg *arg)
+{
+ term.mode ^= MODE_PRINT;
+}
+
+void
+printscreen(const Arg *arg)
+{
+ tdump();
+}
+
+void
+printsel(const Arg *arg)
+{
+ tdumpsel();
+}
+
+void
+tdumpsel(void)
+{
+ char *ptr;
+
+ if ((ptr = getsel())) {
+ tprinter(ptr, strlen(ptr));
+ free(ptr);
+ }
+}
+
+void
+tdumpline(int n)
+{
+ char buf[UTF_SIZ];
+ Glyph *bp, *end;
+
+ bp = &term.line[n][0];
+ end = &bp[MIN(tlinelen(n), term.col) - 1];
+ if (bp != end || bp->u != ' ') {
+ for ( ;bp <= end; ++bp)
+ tprinter(buf, utf8encode(bp->u, buf));
+ }
+ tprinter("\n", 1);
+}
+
+void
+tdump(void)
+{
+ int i;
+
+ for (i = 0; i < term.row; ++i)
+ tdumpline(i);
+}
+
+void
+tputtab(int n)
+{
+ uint x = term.c.x;
+
+ if (n > 0) {
+ while (x < term.col && n--)
+ for (++x; x < term.col && !term.tabs[x]; ++x)
+ /* nothing */ ;
+ } else if (n < 0) {
+ while (x > 0 && n++)
+ for (--x; x > 0 && !term.tabs[x]; --x)
+ /* nothing */ ;
+ }
+ term.c.x = LIMIT(x, 0, term.col-1);
+}
+
+void
+tdefutf8(char ascii)
+{
+ if (ascii == 'G')
+ term.mode |= MODE_UTF8;
+ else if (ascii == '@')
+ term.mode &= ~MODE_UTF8;
+}
+
+void
+tdeftran(char ascii)
+{
+ static char cs[] = "0B";
+ static int vcs[] = {CS_GRAPHIC0, CS_USA};
+ char *p;
+
+ if ((p = strchr(cs, ascii)) == NULL) {
+ fprintf(stderr, "esc unhandled charset: ESC ( %c\n", ascii);
+ } else {
+ term.trantbl[term.icharset] = vcs[p - cs];
+ }
+}
+
+void
+tdectest(char c)
+{
+ int x, y;
+
+ if (c == '8') { /* DEC screen alignment test. */
+ for (x = 0; x < term.col; ++x) {
+ for (y = 0; y < term.row; ++y)
+ tsetchar('E', &term.c.attr, x, y);
+ }
+ }
+}
+
+void
+tstrsequence(uchar c)
+{
+ strreset();
+
+ switch (c) {
+ case 0x90: /* DCS -- Device Control String */
+ c = 'P';
+ term.esc |= ESC_DCS;
+ break;
+ case 0x9f: /* APC -- Application Program Command */
+ c = '_';
+ break;
+ case 0x9e: /* PM -- Privacy Message */
+ c = '^';
+ break;
+ case 0x9d: /* OSC -- Operating System Command */
+ c = ']';
+ break;
+ }
+ strescseq.type = c;
+ term.esc |= ESC_STR;
+}
+
+void
+tcontrolcode(uchar ascii)
+{
+ switch (ascii) {
+ case '\t': /* HT */
+ tputtab(1);
+ return;
+ case '\b': /* BS */
+ tmoveto(term.c.x-1, term.c.y);
+ return;
+ case '\r': /* CR */
+ tmoveto(0, term.c.y);
+ return;
+ case '\f': /* LF */
+ case '\v': /* VT */
+ case '\n': /* LF */
+ /* go to first col if the mode is set */
+ tnewline(IS_SET(MODE_CRLF));
+ return;
+ case '\a': /* BEL */
+ if (term.esc & ESC_STR_END) {
+ /* backwards compatibility to xterm */
+ strhandle();
+ } else {
+ xbell();
+ }
+ break;
+ case '\033': /* ESC */
+ csireset();
+ term.esc &= ~(ESC_CSI|ESC_ALTCHARSET|ESC_TEST);
+ term.esc |= ESC_START;
+ return;
+ case '\016': /* SO (LS1 -- Locking shift 1) */
+ case '\017': /* SI (LS0 -- Locking shift 0) */
+ term.charset = 1 - (ascii - '\016');
+ return;
+ case '\032': /* SUB */
+ tsetchar('?', &term.c.attr, term.c.x, term.c.y);
+ case '\030': /* CAN */
+ csireset();
+ break;
+ case '\005': /* ENQ (IGNORED) */
+ case '\000': /* NUL (IGNORED) */
+ case '\021': /* XON (IGNORED) */
+ case '\023': /* XOFF (IGNORED) */
+ case 0177: /* DEL (IGNORED) */
+ return;
+ case 0x80: /* TODO: PAD */
+ case 0x81: /* TODO: HOP */
+ case 0x82: /* TODO: BPH */
+ case 0x83: /* TODO: NBH */
+ case 0x84: /* TODO: IND */
+ break;
+ case 0x85: /* NEL -- Next line */
+ tnewline(1); /* always go to first col */
+ break;
+ case 0x86: /* TODO: SSA */
+ case 0x87: /* TODO: ESA */
+ break;
+ case 0x88: /* HTS -- Horizontal tab stop */
+ term.tabs[term.c.x] = 1;
+ break;
+ case 0x89: /* TODO: HTJ */
+ case 0x8a: /* TODO: VTS */
+ case 0x8b: /* TODO: PLD */
+ case 0x8c: /* TODO: PLU */
+ case 0x8d: /* TODO: RI */
+ case 0x8e: /* TODO: SS2 */
+ case 0x8f: /* TODO: SS3 */
+ case 0x91: /* TODO: PU1 */
+ case 0x92: /* TODO: PU2 */
+ case 0x93: /* TODO: STS */
+ case 0x94: /* TODO: CCH */
+ case 0x95: /* TODO: MW */
+ case 0x96: /* TODO: SPA */
+ case 0x97: /* TODO: EPA */
+ case 0x98: /* TODO: SOS */
+ case 0x99: /* TODO: SGCI */
+ break;
+ case 0x9a: /* DECID -- Identify Terminal */
+ ttywrite(vtiden, strlen(vtiden), 0);
+ break;
+ case 0x9b: /* TODO: CSI */
+ case 0x9c: /* TODO: ST */
+ break;
+ case 0x90: /* DCS -- Device Control String */
+ case 0x9d: /* OSC -- Operating System Command */
+ case 0x9e: /* PM -- Privacy Message */
+ case 0x9f: /* APC -- Application Program Command */
+ tstrsequence(ascii);
+ return;
+ }
+ /* only CAN, SUB, \a and C1 chars interrupt a sequence */
+ term.esc &= ~(ESC_STR_END|ESC_STR);
+}
+
+/*
+ * returns 1 when the sequence is finished and it hasn't to read
+ * more characters for this sequence, otherwise 0
+ */
+int
+eschandle(uchar ascii)
+{
+ switch (ascii) {
+ case '[':
+ term.esc |= ESC_CSI;
+ return 0;
+ case '#':
+ term.esc |= ESC_TEST;
+ return 0;
+ case '%':
+ term.esc |= ESC_UTF8;
+ return 0;
+ case 'P': /* DCS -- Device Control String */
+ case '_': /* APC -- Application Program Command */
+ case '^': /* PM -- Privacy Message */
+ case ']': /* OSC -- Operating System Command */
+ case 'k': /* old title set compatibility */
+ tstrsequence(ascii);
+ return 0;
+ case 'n': /* LS2 -- Locking shift 2 */
+ case 'o': /* LS3 -- Locking shift 3 */
+ term.charset = 2 + (ascii - 'n');
+ break;
+ case '(': /* GZD4 -- set primary charset G0 */
+ case ')': /* G1D4 -- set secondary charset G1 */
+ case '*': /* G2D4 -- set tertiary charset G2 */
+ case '+': /* G3D4 -- set quaternary charset G3 */
+ term.icharset = ascii - '(';
+ term.esc |= ESC_ALTCHARSET;
+ return 0;
+ case 'D': /* IND -- Linefeed */
+ if (term.c.y == term.bot) {
+ tscrollup(term.top, 1, 1);
+ } else {
+ tmoveto(term.c.x, term.c.y+1);
+ }
+ break;
+ case 'E': /* NEL -- Next line */
+ tnewline(1); /* always go to first col */
+ break;
+ case 'H': /* HTS -- Horizontal tab stop */
+ term.tabs[term.c.x] = 1;
+ break;
+ case 'M': /* RI -- Reverse index */
+ if (term.c.y == term.top) {
+ tscrolldown(term.top, 1, 1);
+ } else {
+ tmoveto(term.c.x, term.c.y-1);
+ }
+ break;
+ case 'Z': /* DECID -- Identify Terminal */
+ ttywrite(vtiden, strlen(vtiden), 0);
+ break;
+ case 'c': /* RIS -- Reset to inital state */
+ treset();
+ resettitle();
+ xloadcols();
+ break;
+ case '=': /* DECPAM -- Application keypad */
+ xsetmode(1, MODE_APPKEYPAD);
+ break;
+ case '>': /* DECPNM -- Normal keypad */
+ xsetmode(0, MODE_APPKEYPAD);
+ break;
+ case '7': /* DECSC -- Save Cursor */
+ tcursor(CURSOR_SAVE);
+ break;
+ case '8': /* DECRC -- Restore Cursor */
+ tcursor(CURSOR_LOAD);
+ break;
+ case '\\': /* ST -- String Terminator */
+ if (term.esc & ESC_STR_END)
+ strhandle();
+ break;
+ default:
+ fprintf(stderr, "erresc: unknown sequence ESC 0x%02X '%c'\n",
+ (uchar) ascii, isprint(ascii)? ascii:'.');
+ break;
+ }
+ return 1;
+}
+
+void
+tputc(Rune u)
+{
+ char c[UTF_SIZ];
+ int control;
+ int width, len;
+ Glyph *gp;
+
+ control = ISCONTROL(u);
+ if (!IS_SET(MODE_UTF8) && !IS_SET(MODE_SIXEL)) {
+ c[0] = u;
+ width = len = 1;
+ } else {
+ len = utf8encode(u, c);
+ if (!control && (width = wcwidth(u)) == -1) {
+ memcpy(c, "\357\277\275", 4); /* UTF_INVALID */
+ width = 1;
+ }
+ }
+
+ if (IS_SET(MODE_PRINT))
+ tprinter(c, len);
+
+ /*
+ * STR sequence must be checked before anything else
+ * because it uses all following characters until it
+ * receives a ESC, a SUB, a ST or any other C1 control
+ * character.
+ */
+ if (term.esc & ESC_STR) {
+ if (u == '\a' || u == 030 || u == 032 || u == 033 ||
+ ISCONTROLC1(u)) {
+ term.esc &= ~(ESC_START|ESC_STR|ESC_DCS);
+ if (IS_SET(MODE_SIXEL)) {
+ /* TODO: render sixel */;
+ term.mode &= ~MODE_SIXEL;
+ return;
+ }
+ term.esc |= ESC_STR_END;
+ goto check_control_code;
+ }
+
+
+ if (IS_SET(MODE_SIXEL)) {
+ /* TODO: implement sixel mode */
+ return;
+ }
+ if (term.esc&ESC_DCS && strescseq.len == 0 && u == 'q')
+ term.mode |= MODE_SIXEL;
+
+ if (strescseq.len+len >= sizeof(strescseq.buf)-1) {
+ /*
+ * Here is a bug in terminals. If the user never sends
+ * some code to stop the str or esc command, then st
+ * will stop responding. But this is better than
+ * silently failing with unknown characters. At least
+ * then users will report back.
+ *
+ * In the case users ever get fixed, here is the code:
+ */
+ /*
+ * term.esc = 0;
+ * strhandle();
+ */
+ return;
+ }
+
+ memmove(&strescseq.buf[strescseq.len], c, len);
+ strescseq.len += len;
+ return;
+ }
+
+check_control_code:
+ /*
+ * Actions of control codes must be performed as soon they arrive
+ * because they can be embedded inside a control sequence, and
+ * they must not cause conflicts with sequences.
+ */
+ if (control) {
+ tcontrolcode(u);
+ /*
+ * control codes are not shown ever
+ */
+ return;
+ } else if (term.esc & ESC_START) {
+ if (term.esc & ESC_CSI) {
+ csiescseq.buf[csiescseq.len++] = u;
+ if (BETWEEN(u, 0x40, 0x7E)
+ || csiescseq.len >= \
+ sizeof(csiescseq.buf)-1) {
+ term.esc = 0;
+ csiparse();
+ csihandle();
+ }
+ return;
+ } else if (term.esc & ESC_UTF8) {
+ tdefutf8(u);
+ } else if (term.esc & ESC_ALTCHARSET) {
+ tdeftran(u);
+ } else if (term.esc & ESC_TEST) {
+ tdectest(u);
+ } else {
+ if (!eschandle(u))
+ return;
+ /* sequence already finished */
+ }
+ term.esc = 0;
+ /*
+ * All characters which form part of a sequence are not
+ * printed
+ */
+ return;
+ }
+ if (sel.ob.x != -1 && BETWEEN(term.c.y, sel.ob.y, sel.oe.y))
+ selclear();
+
+ gp = &term.line[term.c.y][term.c.x];
+ if (IS_SET(MODE_WRAP) && (term.c.state & CURSOR_WRAPNEXT)) {
+ gp->mode |= ATTR_WRAP;
+ tnewline(1);
+ gp = &term.line[term.c.y][term.c.x];
+ }
+
+ if (IS_SET(MODE_INSERT) && term.c.x+width < term.col)
+ memmove(gp+width, gp, (term.col - term.c.x - width) * sizeof(Glyph));
+
+ if (term.c.x+width > term.col) {
+ tnewline(1);
+ gp = &term.line[term.c.y][term.c.x];
+ }
+
+ tsetchar(u, &term.c.attr, term.c.x, term.c.y);
+
+ if (width == 2) {
+ gp->mode |= ATTR_WIDE;
+ if (term.c.x+1 < term.col) {
+ gp[1].u = '\0';
+ gp[1].mode = ATTR_WDUMMY;
+ }
+ }
+ if (term.c.x+width < term.col) {
+ tmoveto(term.c.x+width, term.c.y);
+ } else {
+ term.c.state |= CURSOR_WRAPNEXT;
+ }
+}
+
+int
+twrite(const char *buf, int buflen, int show_ctrl)
+{
+ int charsize;
+ Rune u;
+ int n;
+
+ for (n = 0; n < buflen; n += charsize) {
+ if (IS_SET(MODE_UTF8) && !IS_SET(MODE_SIXEL)) {
+ /* process a complete utf8 char */
+ charsize = utf8decode(buf + n, &u, buflen - n);
+ if (charsize == 0)
+ break;
+ } else {
+ u = buf[n] & 0xFF;
+ charsize = 1;
+ }
+ if (show_ctrl && ISCONTROL(u)) {
+ if (u & 0x80) {
+ u &= 0x7f;
+ tputc('^');
+ tputc('[');
+ } else if (u != '\n' && u != '\r' && u != '\t') {
+ u ^= 0x40;
+ tputc('^');
+ }
+ }
+ tputc(u);
+ }
+ return n;
+}
+
+void
+tresize(int col, int row)
+{
+ int i, j;
+ int minrow = MIN(row, term.row);
+ int mincol = MIN(col, term.col);
+ int *bp;
+ TCursor c;
+
+ if (col < 1 || row < 1) {
+ fprintf(stderr,
+ "tresize: error resizing to %dx%d\n", col, row);
+ return;
+ }
+
+ /*
+ * slide screen to keep cursor where we expect it -
+ * tscrollup would work here, but we can optimize to
+ * memmove because we're freeing the earlier lines
+ */
+ for (i = 0; i <= term.c.y - row; i++) {
+ free(term.line[i]);
+ free(term.alt[i]);
+ }
+ /* ensure that both src and dst are not NULL */
+ if (i > 0) {
+ memmove(term.line, term.line + i, row * sizeof(Line));
+ memmove(term.alt, term.alt + i, row * sizeof(Line));
+ }
+ for (i += row; i < term.row; i++) {
+ free(term.line[i]);
+ free(term.alt[i]);
+ }
+
+ /* resize to new height */
+ term.line = xrealloc(term.line, row * sizeof(Line));
+ term.alt = xrealloc(term.alt, row * sizeof(Line));
+ term.dirty = xrealloc(term.dirty, row * sizeof(*term.dirty));
+ term.tabs = xrealloc(term.tabs, col * sizeof(*term.tabs));
+
+ for (i = 0; i < HISTSIZE; i++) {
+ term.hist[i] = xrealloc(term.hist[i], col * sizeof(Glyph));
+ for (j = mincol; j < col; j++) {
+ term.hist[i][j] = term.c.attr;
+ term.hist[i][j].u = ' ';
+ }
+ }
+
+ /* resize each r w to new width, zero-pad if needed */
+ for (i = 0; i < minrow; i++) {
+ term.line[i] = xrealloc(term.line[i], col * sizeof(Glyph));
+ term.alt[i] = xrealloc(term.alt[i], col * sizeof(Glyph));
+ }
+
+ /* allocate any new rows */
+ for (/* i = minrow */; i < row; i++) {
+ term.line[i] = xmalloc(col * sizeof(Glyph));
+ term.alt[i] = xmalloc(col * sizeof(Glyph));
+ }
+ if (col > term.col) {
+ bp = term.tabs + term.col;
+
+ memset(bp, 0, sizeof(*term.tabs) * (col - term.col));
+ while (--bp > term.tabs && !*bp)
+ /* nothing */ ;
+ for (bp += tabspaces; bp < term.tabs + col; bp += tabspaces)
+ *bp = 1;
+ }
+ /* update terminal size */
+ term.col = col;
+ term.row = row;
+ /* reset scrolling region */
+ tsetscroll(0, row-1);
+ /* make use of the LIMIT in tmoveto */
+ tmoveto(term.c.x, term.c.y);
+ /* Clearing both screens (it makes dirty all lines) */
+ c = term.c;
+ for (i = 0; i < 2; i++) {
+ if (mincol < col && 0 < minrow) {
+ tclearregion(mincol, 0, col - 1, minrow - 1);
+ }
+ if (0 < col && minrow < row) {
+ tclearregion(0, minrow, col - 1, row - 1);
+ }
+ tswapscreen();
+ tcursor(CURSOR_LOAD);
+ }
+ term.c = c;
+}
+
+void
+resettitle(void)
+{
+ xsettitle(NULL);
+}
+
+void
+drawregion(int x1, int y1, int x2, int y2)
+{
+ int y;
+ for (y = y1; y < y2; y++) {
+ if (!term.dirty[y])
+ continue;
+
+ term.dirty[y] = 0;
+ xdrawline(TLINE(y), x1, y, x2);
+ }
+}
+
+void
+draw(void)
+{
+ int cx = term.c.x;
+
+ if (!xstartdraw())
+ return;
+
+ /* adjust cursor position */
+ LIMIT(term.ocx, 0, term.col-1);
+ LIMIT(term.ocy, 0, term.row-1);
+ if (term.line[term.ocy][term.ocx].mode & ATTR_WDUMMY)
+ term.ocx--;
+ if (term.line[term.c.y][cx].mode & ATTR_WDUMMY)
+ cx--;
+
+ drawregion(0, 0, term.col, term.row);
+ if (term.scr == 0) {
+ xdrawcursor(cx, term.c.y, term.line[term.c.y][cx],
+ term.ocx, term.ocy, term.line[term.ocy][term.ocx]);
+ }
+ term.ocx = cx, term.ocy = term.c.y;
+ xfinishdraw();
+}
+
+void
+redraw(void)
+{
+ tfulldirt();
+ draw();
+}
diff --git a/suckless/st/st.h b/suckless/st/st.h
new file mode 100644
index 00000000..850c5f02
--- /dev/null
+++ b/suckless/st/st.h
@@ -0,0 +1,140 @@
+/* See LICENSE for license details. */
+
+#include
+#include
+
+/* Arbitrary size */
+#define HISTSIZE 2000
+
+/* macros */
+#define MIN(a, b) ((a) < (b) ? (a) : (b))
+#define MAX(a, b) ((a) < (b) ? (b) : (a))
+#define LEN(a) (sizeof(a) / sizeof(a)[0])
+#define BETWEEN(x, a, b) ((a) <= (x) && (x) <= (b))
+#define DIVCEIL(n, d) (((n) + ((d) - 1)) / (d))
+#define DEFAULT(a, b) (a) = (a) ? (a) : (b)
+#define LIMIT(x, a, b) (x) = (x) < (a) ? (a) : (x) > (b) ? (b) : (x)
+#define ATTRCMP(a, b) ((a).mode != (b).mode || (a).fg != (b).fg || \
+ (a).bg != (b).bg)
+#define TIMEDIFF(t1, t2) ((t1.tv_sec-t2.tv_sec)*1000 + \
+ (t1.tv_nsec-t2.tv_nsec)/1E6)
+#define MODBIT(x, set, bit) ((set) ? ((x) |= (bit)) : ((x) &= ~(bit)))
+
+#define TRUECOLOR(r,g,b) (1 << 24 | (r) << 16 | (g) << 8 | (b))
+#define IS_TRUECOL(x) (1 << 24 & (x))
+#define TLINE(y) ((y) < term.scr ? term.hist[((y) + term.histi - term.scr \
+ + HISTSIZE + 1) % HISTSIZE] : term.line[(y) - term.scr])
+
+enum glyph_attribute {
+ ATTR_NULL = 0,
+ ATTR_BOLD = 1 << 0,
+ ATTR_FAINT = 1 << 1,
+ ATTR_ITALIC = 1 << 2,
+ ATTR_UNDERLINE = 1 << 3,
+ ATTR_BLINK = 1 << 4,
+ ATTR_REVERSE = 1 << 5,
+ ATTR_INVISIBLE = 1 << 6,
+ ATTR_STRUCK = 1 << 7,
+ ATTR_WRAP = 1 << 8,
+ ATTR_WIDE = 1 << 9,
+ ATTR_WDUMMY = 1 << 10,
+ ATTR_BOLD_FAINT = ATTR_BOLD | ATTR_FAINT,
+};
+
+enum selection_mode {
+ SEL_IDLE = 0,
+ SEL_EMPTY = 1,
+ SEL_READY = 2
+};
+
+enum selection_type {
+ SEL_REGULAR = 1,
+ SEL_RECTANGULAR = 2
+};
+
+enum selection_snap {
+ SNAP_WORD = 1,
+ SNAP_LINE = 2
+};
+
+typedef unsigned char uchar;
+typedef unsigned int uint;
+typedef unsigned long ulong;
+typedef unsigned short ushort;
+
+typedef uint_least32_t Rune;
+
+#define Glyph Glyph_
+typedef struct {
+ Rune u; /* character code */
+ ushort mode; /* attribute flags */
+ uint32_t fg; /* foreground */
+ uint32_t bg; /* background */
+} Glyph;
+
+typedef Glyph *Line;
+
+typedef union {
+ int i;
+ uint ui;
+ float f;
+ const void *v;
+} Arg;
+
+typedef struct {
+ uint b;
+ uint mask;
+ void (*func)(const Arg *);
+ const Arg arg;
+} MouseKey;
+
+void die(const char *, ...);
+void redraw(void);
+void draw(void);
+
+void iso14755(const Arg *);
+void printscreen(const Arg *);
+void printsel(const Arg *);
+void sendbreak(const Arg *);
+void toggleprinter(const Arg *);
+
+int tattrset(int);
+void tnew(int, int);
+void tresize(int, int);
+void tsetdirtattr(int);
+void ttyhangup(void);
+int ttynew(char *, char *, char *, char **);
+size_t ttyread(void);
+void ttyresize(int, int);
+void ttywrite(const char *, size_t, int);
+
+void resettitle(void);
+
+void selclear(void);
+void selinit(void);
+void selstart(int, int, int);
+void selextend(int, int, int, int);
+int selected(int, int);
+char *getsel(void);
+
+size_t utf8encode(Rune, char *);
+
+void *xmalloc(size_t);
+void *xrealloc(void *, size_t);
+char *xstrdup(char *);
+
+void kscrolldown(const Arg *);
+void kscrollup(const Arg *);
+
+/* config.h globals */
+extern char *utmp;
+extern char *stty_args;
+extern char *vtiden;
+extern char *worddelimiters;
+extern int allowaltscreen;
+extern char *termname;
+extern unsigned int tabspaces;
+extern unsigned int alpha;
+extern unsigned int defaultfg;
+extern unsigned int defaultbg;
+extern MouseKey mkeys[];
diff --git a/suckless/st/st.info b/suckless/st/st.info
new file mode 100644
index 00000000..52fc6177
--- /dev/null
+++ b/suckless/st/st.info
@@ -0,0 +1,222 @@
+st| simpleterm,
+ acsc=+C\,D-A.B0E``aaffgghFiGjjkkllmmnnooppqqrrssttuuvvwwxxyyzz{{||}}~~,
+ am,
+ bce,
+ bel=^G,
+ blink=\E[5m,
+ bold=\E[1m,
+ cbt=\E[Z,
+ cvvis=\E[?25h,
+ civis=\E[?25l,
+ clear=\E[H\E[2J,
+ cnorm=\E[?12l\E[?25h,
+ colors#8,
+ cols#80,
+ cr=^M,
+ csr=\E[%i%p1%d;%p2%dr,
+ cub=\E[%p1%dD,
+ cub1=^H,
+ cud1=^J,
+ cud=\E[%p1%dB,
+ cuf1=\E[C,
+ cuf=\E[%p1%dC,
+ cup=\E[%i%p1%d;%p2%dH,
+ cuu1=\E[A,
+ cuu=\E[%p1%dA,
+ dch=\E[%p1%dP,
+ dch1=\E[P,
+ dim=\E[2m,
+ dl=\E[%p1%dM,
+ dl1=\E[M,
+ ech=\E[%p1%dX,
+ ed=\E[J,
+ el=\E[K,
+ el1=\E[1K,
+ enacs=\E)0,
+ flash=\E[?5h$<80/>\E[?5l,
+ fsl=^G,
+ home=\E[H,
+ hpa=\E[%i%p1%dG,
+ hs,
+ ht=^I,
+ hts=\EH,
+ ich=\E[%p1%d@,
+ il1=\E[L,
+ il=\E[%p1%dL,
+ ind=^J,
+ indn=\E[%p1%dS,
+ invis=\E[8m,
+ is2=\E[4l\E>\E[?1034l,
+ it#8,
+ kel=\E[1;2F,
+ ked=\E[1;5F,
+ ka1=\E[1~,
+ ka3=\E[5~,
+ kc1=\E[4~,
+ kc3=\E[6~,
+ kbs=\177,
+ kcbt=\E[Z,
+ kb2=\EOu,
+ kcub1=\EOD,
+ kcud1=\EOB,
+ kcuf1=\EOC,
+ kcuu1=\EOA,
+ kDC=\E[3;2~,
+ kent=\EOM,
+ kEND=\E[1;2F,
+ kIC=\E[2;2~,
+ kNXT=\E[6;2~,
+ kPRV=\E[5;2~,
+ kHOM=\E[1;2H,
+ kLFT=\E[1;2D,
+ kRIT=\E[1;2C,
+ kind=\E[1;2B,
+ kri=\E[1;2A,
+ kclr=\E[3;5~,
+ kdl1=\E[3;2~,
+ kdch1=\E[3~,
+ kich1=\E[2~,
+ kend=\E[4~,
+ kf1=\EOP,
+ kf2=\EOQ,
+ kf3=\EOR,
+ kf4=\EOS,
+ kf5=\E[15~,
+ kf6=\E[17~,
+ kf7=\E[18~,
+ kf8=\E[19~,
+ kf9=\E[20~,
+ kf10=\E[21~,
+ kf11=\E[23~,
+ kf12=\E[24~,
+ kf13=\E[1;2P,
+ kf14=\E[1;2Q,
+ kf15=\E[1;2R,
+ kf16=\E[1;2S,
+ kf17=\E[15;2~,
+ kf18=\E[17;2~,
+ kf19=\E[18;2~,
+ kf20=\E[19;2~,
+ kf21=\E[20;2~,
+ kf22=\E[21;2~,
+ kf23=\E[23;2~,
+ kf24=\E[24;2~,
+ kf25=\E[1;5P,
+ kf26=\E[1;5Q,
+ kf27=\E[1;5R,
+ kf28=\E[1;5S,
+ kf29=\E[15;5~,
+ kf30=\E[17;5~,
+ kf31=\E[18;5~,
+ kf32=\E[19;5~,
+ kf33=\E[20;5~,
+ kf34=\E[21;5~,
+ kf35=\E[23;5~,
+ kf36=\E[24;5~,
+ kf37=\E[1;6P,
+ kf38=\E[1;6Q,
+ kf39=\E[1;6R,
+ kf40=\E[1;6S,
+ kf41=\E[15;6~,
+ kf42=\E[17;6~,
+ kf43=\E[18;6~,
+ kf44=\E[19;6~,
+ kf45=\E[20;6~,
+ kf46=\E[21;6~,
+ kf47=\E[23;6~,
+ kf48=\E[24;6~,
+ kf49=\E[1;3P,
+ kf50=\E[1;3Q,
+ kf51=\E[1;3R,
+ kf52=\E[1;3S,
+ kf53=\E[15;3~,
+ kf54=\E[17;3~,
+ kf55=\E[18;3~,
+ kf56=\E[19;3~,
+ kf57=\E[20;3~,
+ kf58=\E[21;3~,
+ kf59=\E[23;3~,
+ kf60=\E[24;3~,
+ kf61=\E[1;4P,
+ kf62=\E[1;4Q,
+ kf63=\E[1;4R,
+ khome=\E[1~,
+ kil1=\E[2;5~,
+ krmir=\E[2;2~,
+ knp=\E[6~,
+ kmous=\E[M,
+ kpp=\E[5~,
+ lines#24,
+ mir,
+ msgr,
+ npc,
+ op=\E[39;49m,
+ pairs#64,
+ mc0=\E[i,
+ mc4=\E[4i,
+ mc5=\E[5i,
+ rc=\E8,
+ rev=\E[7m,
+ ri=\EM,
+ ritm=\E[23m,
+ rmacs=\E(B,
+ rmcup=\E[?1049l,
+ rmir=\E[4l,
+ rmkx=\E[?1l\E>,
+ rmso=\E[27m,
+ rmul=\E[24m,
+ rs1=\Ec,
+ rs2=\E[4l\E>\E[?1034l,
+ sc=\E7,
+ setab=\E[4%p1%dm,
+ setaf=\E[3%p1%dm,
+ setb=\E[4%?%p1%{1}%=%t4%e%p1%{3}%=%t6%e%p1%{4}%=%t1%e%p1%{6}%=%t3%e%p1%d%;m,
+ setf=\E[3%?%p1%{1}%=%t4%e%p1%{3}%=%t6%e%p1%{4}%=%t1%e%p1%{6}%=%t3%e%p1%d%;m,
+ sgr0=\E[0m,
+ sgr=%?%p9%t\E(0%e\E(B%;\E[0%?%p6%t;1%;%?%p2%t;4%;%?%p1%p3%|%t;7%;%?%p4%t;5%;%?%p7%t;8%;m,
+ sitm=\E[3m,
+ smacs=\E(0,
+ smcup=\E[?1049h,
+ smir=\E[4h,
+ smkx=\E[?1h\E=,
+ smso=\E[7m,
+ smul=\E[4m,
+ tbc=\E[3g,
+ tsl=\E]0;,
+ xenl,
+ vpa=\E[%i%p1%dd,
+# XTerm extensions
+ rmxx=\E[29m,
+ smxx=\E[9m,
+# tmux extensions, see TERMINFO EXTENSIONS in tmux(1)
+ Se,
+ Ss,
+ Tc,
+ Ms=\E]52;%p1%s;%p2%s\007,
+
+st-256color| simpleterm with 256 colors,
+ use=st,
+ ccc,
+ colors#256,
+ oc=\E]104\007,
+ pairs#32767,
+# Nicked from xterm-256color
+ initc=\E]4;%p1%d;rgb\:%p2%{255}%*%{1000}%/%2.2X/%p3%{255}%*%{1000}%/%2.2X/%p4%{255}%*%{1000}%/%2.2X\E\\,
+ setab=\E[%?%p1%{8}%<%t4%p1%d%e%p1%{16}%<%t10%p1%{8}%-%d%e48;5;%p1%d%;m,
+ setaf=\E[%?%p1%{8}%<%t3%p1%d%e%p1%{16}%<%t9%p1%{8}%-%d%e38;5;%p1%d%;m,
+
+st-meta| simpleterm with meta key,
+ use=st,
+ km,
+ rmm=\E[?1034l,
+ smm=\E[?1034h,
+ rs2=\E[4l\E>\E[?1034h,
+ is2=\E[4l\E>\E[?1034h,
+
+st-meta-256color| simpleterm with meta key and 256 colors,
+ use=st-256color,
+ km,
+ rmm=\E[?1034l,
+ smm=\E[?1034h,
+ rs2=\E[4l\E>\E[?1034h,
+ is2=\E[4l\E>\E[?1034h,
diff --git a/suckless/st/st.o b/suckless/st/st.o
new file mode 100644
index 00000000..c27f2497
Binary files /dev/null and b/suckless/st/st.o differ
diff --git a/suckless/st/win.h b/suckless/st/win.h
new file mode 100644
index 00000000..d277477a
--- /dev/null
+++ b/suckless/st/win.h
@@ -0,0 +1,42 @@
+/* See LICENSE for license details. */
+
+enum win_mode {
+ MODE_VISIBLE = 1 << 0,
+ MODE_FOCUSED = 1 << 1,
+ MODE_APPKEYPAD = 1 << 2,
+ MODE_MOUSEBTN = 1 << 3,
+ MODE_MOUSEMOTION = 1 << 4,
+ MODE_REVERSE = 1 << 5,
+ MODE_KBDLOCK = 1 << 6,
+ MODE_HIDE = 1 << 7,
+ MODE_APPCURSOR = 1 << 8,
+ MODE_MOUSESGR = 1 << 9,
+ MODE_8BIT = 1 << 10,
+ MODE_BLINK = 1 << 11,
+ MODE_FBLINK = 1 << 12,
+ MODE_FOCUS = 1 << 13,
+ MODE_MOUSEX10 = 1 << 14,
+ MODE_MOUSEMANY = 1 << 15,
+ MODE_BRCKTPASTE = 1 << 16,
+ MODE_NUMLOCK = 1 << 17,
+ MODE_MOUSE = MODE_MOUSEBTN|MODE_MOUSEMOTION|MODE_MOUSEX10\
+ |MODE_MOUSEMANY,
+};
+
+/* alpha */
+#define OPAQUE 0Xff
+#define USE_ARGB (alpha != OPAQUE && opt_embed == NULL)
+
+void xbell(void);
+void xclipcopy(void);
+void xdrawcursor(int, int, Glyph, int, int, Glyph);
+void xdrawline(Line, int, int, int);
+void xfinishdraw(void);
+void xloadcols(void);
+int xsetcolorname(int, const char *);
+void xsettitle(char *);
+int xsetcursor(int);
+void xsetmode(int, unsigned int);
+void xsetpointermotion(int);
+void xsetsel(char *);
+int xstartdraw(void);
diff --git a/suckless/st/x.c b/suckless/st/x.c
new file mode 100644
index 00000000..12dfc4c7
--- /dev/null
+++ b/suckless/st/x.c
@@ -0,0 +1,2090 @@
+/* See LICENSE for license details. */
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+#include
+
+static char *argv0;
+#include "arg.h"
+#include "st.h"
+#include "win.h"
+
+/* types used in config.h */
+typedef struct {
+ uint mod;
+ KeySym keysym;
+ void (*func)(const Arg *);
+ const Arg arg;
+} Shortcut;
+
+typedef struct {
+ uint b;
+ uint mask;
+ char *s;
+} MouseShortcut;
+
+typedef struct {
+ KeySym k;
+ uint mask;
+ char *s;
+ /* three-valued logic variables: 0 indifferent, 1 on, -1 off */
+ signed char appkey; /* application keypad */
+ signed char appcursor; /* application cursor */
+} Key;
+
+/* Xresources preferences */
+enum resource_type {
+ STRING = 0,
+ INTEGER = 1,
+ FLOAT = 2
+};
+
+typedef struct {
+ char *name;
+ enum resource_type type;
+ void *dst;
+} ResourcePref;
+
+/* X modifiers */
+#define XK_ANY_MOD UINT_MAX
+#define XK_NO_MOD 0
+#define XK_SWITCH_MOD (1<<13)
+
+/* function definitions used in config.h */
+static void clipcopy(const Arg *);
+static void clippaste(const Arg *);
+static void numlock(const Arg *);
+static void selpaste(const Arg *);
+static void zoom(const Arg *);
+static void zoomabs(const Arg *);
+static void zoomreset(const Arg *);
+
+/* config.h for applying patches and the configuration. */
+#include "config.h"
+
+/* XEMBED messages */
+#define XEMBED_FOCUS_IN 4
+#define XEMBED_FOCUS_OUT 5
+
+/* macros */
+#define IS_SET(flag) ((win.mode & (flag)) != 0)
+#define TRUERED(x) (((x) & 0xff0000) >> 8)
+#define TRUEGREEN(x) (((x) & 0xff00))
+#define TRUEBLUE(x) (((x) & 0xff) << 8)
+
+typedef XftDraw *Draw;
+typedef XftColor Color;
+typedef XftGlyphFontSpec GlyphFontSpec;
+
+/* Purely graphic info */
+typedef struct {
+ int tw, th; /* tty width and height */
+ int w, h; /* window width and height */
+ int ch; /* char height */
+ int cw; /* char width */
+ int cyo; /* char y offset */
+ int mode; /* window state/mode flags */
+ int cursor; /* cursor style */
+} TermWindow;
+
+typedef struct {
+ Display *dpy;
+ Colormap cmap;
+ Window win;
+ Drawable buf;
+ GlyphFontSpec *specbuf; /* font spec buffer used for rendering */
+ Atom xembed, wmdeletewin, netwmname, netwmpid;
+ XIM xim;
+ XIC xic;
+ Draw draw;
+ Visual *vis;
+ XSetWindowAttributes attrs;
+ int scr;
+ int isfixed; /* is fixed geometry? */
+ int depth; /* bit depth */
+ int l, t; /* left and top offset */
+ int gm; /* geometry mask */
+} XWindow;
+
+typedef struct {
+ Atom xtarget;
+ char *primary, *clipboard;
+ struct timespec tclick1;
+ struct timespec tclick2;
+} XSelection;
+
+/* Font structure */
+#define Font Font_
+typedef struct {
+ int height;
+ int width;
+ int ascent;
+ int descent;
+ int badslant;
+ int badweight;
+ short lbearing;
+ short rbearing;
+ XftFont *match;
+ FcFontSet *set;
+ FcPattern *pattern;
+} Font;
+
+/* Drawing Context */
+typedef struct {
+ Color *col;
+ size_t collen;
+ Font font, bfont, ifont, ibfont;
+ GC gc;
+} DC;
+
+static inline ushort sixd_to_16bit(int);
+static int xmakeglyphfontspecs(XftGlyphFontSpec *, const Glyph *, int, int, int);
+static void xdrawglyphfontspecs(const XftGlyphFontSpec *, Glyph, int, int, int);
+static void xdrawglyph(Glyph, int, int);
+static void xclear(int, int, int, int);
+static int xgeommasktogravity(int);
+static void xinit(int, int);
+static void cresize(int, int);
+static void xresize(int, int);
+static void xhints(void);
+static int xloadcolor(int, const char *, Color *);
+static int xloadfont(Font *, FcPattern *);
+static void xloadfonts(char *, double);
+static void xunloadfont(Font *);
+static void xunloadfonts(void);
+static void xsetenv(void);
+static void xseturgency(int);
+static int evcol(XEvent *);
+static int evrow(XEvent *);
+
+static void expose(XEvent *);
+static void visibility(XEvent *);
+static void unmap(XEvent *);
+static void kpress(XEvent *);
+static void cmessage(XEvent *);
+static void resize(XEvent *);
+static void focus(XEvent *);
+static void brelease(XEvent *);
+static void bpress(XEvent *);
+static void bmotion(XEvent *);
+static void propnotify(XEvent *);
+static void selnotify(XEvent *);
+static void selclear_(XEvent *);
+static void selrequest(XEvent *);
+static void setsel(char *, Time);
+static void mousesel(XEvent *, int);
+static void mousereport(XEvent *);
+static char *kmap(KeySym, uint);
+static int match(uint, uint);
+
+static void run(void);
+static void usage(void);
+
+static void (*handler[LASTEvent])(XEvent *) = {
+ [KeyPress] = kpress,
+ [ClientMessage] = cmessage,
+ [ConfigureNotify] = resize,
+ [VisibilityNotify] = visibility,
+ [UnmapNotify] = unmap,
+ [Expose] = expose,
+ [FocusIn] = focus,
+ [FocusOut] = focus,
+ [MotionNotify] = bmotion,
+ [ButtonPress] = bpress,
+ [ButtonRelease] = brelease,
+/*
+ * Uncomment if you want the selection to disappear when you select something
+ * different in another window.
+ */
+/* [SelectionClear] = selclear_, */
+ [SelectionNotify] = selnotify,
+/*
+ * PropertyNotify is only turned on when there is some INCR transfer happening
+ * for the selection retrieval.
+ */
+ [PropertyNotify] = propnotify,
+ [SelectionRequest] = selrequest,
+};
+
+/* Globals */
+static DC dc;
+static XWindow xw;
+static XSelection xsel;
+static TermWindow win;
+
+/* Font Ring Cache */
+enum {
+ FRC_NORMAL,
+ FRC_ITALIC,
+ FRC_BOLD,
+ FRC_ITALICBOLD
+};
+
+typedef struct {
+ XftFont *font;
+ int flags;
+ Rune unicodep;
+} Fontcache;
+
+/* Fontcache is an array now. A new font will be appended to the array. */
+static Fontcache frc[16];
+static int frclen = 0;
+static char *usedfont = NULL;
+static double usedfontsize = 0;
+static double defaultfontsize = 0;
+
+static char *opt_class = NULL;
+static char **opt_cmd = NULL;
+static char *opt_embed = NULL;
+static char *opt_font = NULL;
+static char *opt_io = NULL;
+static char *opt_line = NULL;
+static char *opt_name = NULL;
+static char *opt_title = NULL;
+
+static int oldbutton = 3; /* button event on startup: 3 = release */
+
+void
+clipcopy(const Arg *dummy)
+{
+ Atom clipboard;
+
+ free(xsel.clipboard);
+ xsel.clipboard = NULL;
+
+ if (xsel.primary != NULL) {
+ xsel.clipboard = xstrdup(xsel.primary);
+ clipboard = XInternAtom(xw.dpy, "CLIPBOARD", 0);
+ XSetSelectionOwner(xw.dpy, clipboard, xw.win, CurrentTime);
+ }
+}
+
+void
+clippaste(const Arg *dummy)
+{
+ Atom clipboard;
+
+ clipboard = XInternAtom(xw.dpy, "CLIPBOARD", 0);
+ XConvertSelection(xw.dpy, clipboard, xsel.xtarget, clipboard,
+ xw.win, CurrentTime);
+}
+
+void
+selpaste(const Arg *dummy)
+{
+ XConvertSelection(xw.dpy, XA_PRIMARY, xsel.xtarget, XA_PRIMARY,
+ xw.win, CurrentTime);
+}
+
+void
+numlock(const Arg *dummy)
+{
+ win.mode ^= MODE_NUMLOCK;
+}
+
+void
+zoom(const Arg *arg)
+{
+ Arg larg;
+
+ larg.f = usedfontsize + arg->f;
+ zoomabs(&larg);
+}
+
+void
+zoomabs(const Arg *arg)
+{
+ xunloadfonts();
+ xloadfonts(usedfont, arg->f);
+ cresize(0, 0);
+ redraw();
+ xhints();
+}
+
+void
+zoomreset(const Arg *arg)
+{
+ Arg larg;
+
+ if (defaultfontsize > 0) {
+ larg.f = defaultfontsize;
+ zoomabs(&larg);
+ }
+}
+
+int
+evcol(XEvent *e)
+{
+ int x = e->xbutton.x - borderpx;
+ LIMIT(x, 0, win.tw - 1);
+ return x / win.cw;
+}
+
+int
+evrow(XEvent *e)
+{
+ int y = e->xbutton.y - borderpx;
+ LIMIT(y, 0, win.th - 1);
+ return y / win.ch;
+}
+
+void
+mousesel(XEvent *e, int done)
+{
+ int type, seltype = SEL_REGULAR;
+ uint state = e->xbutton.state & ~(Button1Mask | forceselmod);
+
+ for (type = 1; type < LEN(selmasks); ++type) {
+ if (match(selmasks[type], state)) {
+ seltype = type;
+ break;
+ }
+ }
+ selextend(evcol(e), evrow(e), seltype, done);
+ if (done)
+ setsel(getsel(), e->xbutton.time);
+}
+
+void
+mousereport(XEvent *e)
+{
+ int len, x = evcol(e), y = evrow(e),
+ button = e->xbutton.button, state = e->xbutton.state;
+ char buf[40];
+ static int ox, oy;
+
+ /* from urxvt */
+ if (e->xbutton.type == MotionNotify) {
+ if (x == ox && y == oy)
+ return;
+ if (!IS_SET(MODE_MOUSEMOTION) && !IS_SET(MODE_MOUSEMANY))
+ return;
+ /* MOUSE_MOTION: no reporting if no button is pressed */
+ if (IS_SET(MODE_MOUSEMOTION) && oldbutton == 3)
+ return;
+
+ button = oldbutton + 32;
+ ox = x;
+ oy = y;
+ } else {
+ if (!IS_SET(MODE_MOUSESGR) && e->xbutton.type == ButtonRelease) {
+ button = 3;
+ } else {
+ button -= Button1;
+ if (button >= 3)
+ button += 64 - 3;
+ }
+ if (e->xbutton.type == ButtonPress) {
+ oldbutton = button;
+ ox = x;
+ oy = y;
+ } else if (e->xbutton.type == ButtonRelease) {
+ oldbutton = 3;
+ /* MODE_MOUSEX10: no button release reporting */
+ if (IS_SET(MODE_MOUSEX10))
+ return;
+ if (button == 64 || button == 65)
+ return;
+ }
+ }
+
+ if (!IS_SET(MODE_MOUSEX10)) {
+ button += ((state & ShiftMask ) ? 4 : 0)
+ + ((state & Mod4Mask ) ? 8 : 0)
+ + ((state & ControlMask) ? 16 : 0);
+ }
+
+ if (IS_SET(MODE_MOUSESGR)) {
+ len = snprintf(buf, sizeof(buf), "\033[<%d;%d;%d%c",
+ button, x+1, y+1,
+ e->xbutton.type == ButtonRelease ? 'm' : 'M');
+ } else if (x < 223 && y < 223) {
+ len = snprintf(buf, sizeof(buf), "\033[M%c%c%c",
+ 32+button, 32+x+1, 32+y+1);
+ } else {
+ return;
+ }
+
+ ttywrite(buf, len, 0);
+}
+
+void
+bpress(XEvent *e)
+{
+ struct timespec now;
+ MouseShortcut *ms;
+ MouseKey *mk;
+ int snap;
+
+ if (IS_SET(MODE_MOUSE) && !(e->xbutton.state & forceselmod)) {
+ mousereport(e);
+ return;
+ }
+
+ for (ms = mshortcuts; ms < mshortcuts + LEN(mshortcuts); ms++) {
+ if (e->xbutton.button == ms->b
+ && match(ms->mask, e->xbutton.state)) {
+ ttywrite(ms->s, strlen(ms->s), 1);
+ return;
+ }
+ }
+
+ for (mk = mkeys; mk < mkeys + LEN(mkeys); mk++) {
+ if (e->xbutton.button == mk->b
+ && match(mk->mask, e->xbutton.state)) {
+ mk->func(&mk->arg);
+ return;
+ }
+ }
+
+ if (e->xbutton.button == Button1) {
+ /*
+ * If the user clicks below predefined timeouts specific
+ * snapping behaviour is exposed.
+ */
+ clock_gettime(CLOCK_MONOTONIC, &now);
+ if (TIMEDIFF(now, xsel.tclick2) <= tripleclicktimeout) {
+ snap = SNAP_LINE;
+ } else if (TIMEDIFF(now, xsel.tclick1) <= doubleclicktimeout) {
+ snap = SNAP_WORD;
+ } else {
+ snap = 0;
+ }
+ xsel.tclick2 = xsel.tclick1;
+ xsel.tclick1 = now;
+
+ selstart(evcol(e), evrow(e), snap);
+ }
+}
+
+void
+propnotify(XEvent *e)
+{
+ XPropertyEvent *xpev;
+ Atom clipboard = XInternAtom(xw.dpy, "CLIPBOARD", 0);
+
+ xpev = &e->xproperty;
+ if (xpev->state == PropertyNewValue &&
+ (xpev->atom == XA_PRIMARY ||
+ xpev->atom == clipboard)) {
+ selnotify(e);
+ }
+}
+
+void
+selnotify(XEvent *e)
+{
+ ulong nitems, ofs, rem;
+ int format;
+ uchar *data, *last, *repl;
+ Atom type, incratom, property = None;
+
+ incratom = XInternAtom(xw.dpy, "INCR", 0);
+
+ ofs = 0;
+ if (e->type == SelectionNotify)
+ property = e->xselection.property;
+ else if (e->type == PropertyNotify)
+ property = e->xproperty.atom;
+
+ if (property == None)
+ return;
+
+ do {
+ if (XGetWindowProperty(xw.dpy, xw.win, property, ofs,
+ BUFSIZ/4, False, AnyPropertyType,
+ &type, &format, &nitems, &rem,
+ &data)) {
+ fprintf(stderr, "Clipboard allocation failed\n");
+ return;
+ }
+
+ if (e->type == PropertyNotify && nitems == 0 && rem == 0) {
+ /*
+ * If there is some PropertyNotify with no data, then
+ * this is the signal of the selection owner that all
+ * data has been transferred. We won't need to receive
+ * PropertyNotify events anymore.
+ */
+ MODBIT(xw.attrs.event_mask, 0, PropertyChangeMask);
+ XChangeWindowAttributes(xw.dpy, xw.win, CWEventMask,
+ &xw.attrs);
+ }
+
+ if (type == incratom) {
+ /*
+ * Activate the PropertyNotify events so we receive
+ * when the selection owner does send us the next
+ * chunk of data.
+ */
+ MODBIT(xw.attrs.event_mask, 1, PropertyChangeMask);
+ XChangeWindowAttributes(xw.dpy, xw.win, CWEventMask,
+ &xw.attrs);
+
+ /*
+ * Deleting the property is the transfer start signal.
+ */
+ XDeleteProperty(xw.dpy, xw.win, (int)property);
+ continue;
+ }
+
+ /*
+ * As seen in getsel:
+ * Line endings are inconsistent in the terminal and GUI world
+ * copy and pasting. When receiving some selection data,
+ * replace all '\n' with '\r'.
+ * FIXME: Fix the computer world.
+ */
+ repl = data;
+ last = data + nitems * format / 8;
+ while ((repl = memchr(repl, '\n', last - repl))) {
+ *repl++ = '\r';
+ }
+
+ if (IS_SET(MODE_BRCKTPASTE) && ofs == 0)
+ ttywrite("\033[200~", 6, 0);
+ ttywrite((char *)data, nitems * format / 8, 1);
+ if (IS_SET(MODE_BRCKTPASTE) && rem == 0)
+ ttywrite("\033[201~", 6, 0);
+ XFree(data);
+ /* number of 32-bit chunks returned */
+ ofs += nitems * format / 32;
+ } while (rem > 0);
+
+ /*
+ * Deleting the property again tells the selection owner to send the
+ * next data chunk in the property.
+ */
+ XDeleteProperty(xw.dpy, xw.win, (int)property);
+}
+
+void
+xclipcopy(void)
+{
+ clipcopy(NULL);
+}
+
+void
+selclear_(XEvent *e)
+{
+ selclear();
+}
+
+void
+selrequest(XEvent *e)
+{
+ XSelectionRequestEvent *xsre;
+ XSelectionEvent xev;
+ Atom xa_targets, string, clipboard;
+ char *seltext;
+
+ xsre = (XSelectionRequestEvent *) e;
+ xev.type = SelectionNotify;
+ xev.requestor = xsre->requestor;
+ xev.selection = xsre->selection;
+ xev.target = xsre->target;
+ xev.time = xsre->time;
+ if (xsre->property == None)
+ xsre->property = xsre->target;
+
+ /* reject */
+ xev.property = None;
+
+ xa_targets = XInternAtom(xw.dpy, "TARGETS", 0);
+ if (xsre->target == xa_targets) {
+ /* respond with the supported type */
+ string = xsel.xtarget;
+ XChangeProperty(xsre->display, xsre->requestor, xsre->property,
+ XA_ATOM, 32, PropModeReplace,
+ (uchar *) &string, 1);
+ xev.property = xsre->property;
+ } else if (xsre->target == xsel.xtarget || xsre->target == XA_STRING) {
+ /*
+ * xith XA_STRING non ascii characters may be incorrect in the
+ * requestor. It is not our problem, use utf8.
+ */
+ clipboard = XInternAtom(xw.dpy, "CLIPBOARD", 0);
+ if (xsre->selection == XA_PRIMARY) {
+ seltext = xsel.primary;
+ } else if (xsre->selection == clipboard) {
+ seltext = xsel.clipboard;
+ } else {
+ fprintf(stderr,
+ "Unhandled clipboard selection 0x%lx\n",
+ xsre->selection);
+ return;
+ }
+ if (seltext != NULL) {
+ XChangeProperty(xsre->display, xsre->requestor,
+ xsre->property, xsre->target,
+ 8, PropModeReplace,
+ (uchar *)seltext, strlen(seltext));
+ xev.property = xsre->property;
+ }
+ }
+
+ /* all done, send a notification to the listener */
+ if (!XSendEvent(xsre->display, xsre->requestor, 1, 0, (XEvent *) &xev))
+ fprintf(stderr, "Error sending SelectionNotify event\n");
+}
+
+void
+setsel(char *str, Time t)
+{
+ if (!str)
+ return;
+
+ free(xsel.primary);
+ xsel.primary = str;
+
+ XSetSelectionOwner(xw.dpy, XA_PRIMARY, xw.win, t);
+ if (XGetSelectionOwner(xw.dpy, XA_PRIMARY) != xw.win)
+ selclear();
+
+ xclipcopy();
+}
+
+void
+xsetsel(char *str)
+{
+ setsel(str, CurrentTime);
+}
+
+void
+brelease(XEvent *e)
+{
+ if (IS_SET(MODE_MOUSE) && !(e->xbutton.state & forceselmod)) {
+ mousereport(e);
+ return;
+ }
+
+ if (e->xbutton.button == Button2)
+ selpaste(NULL);
+ else if (e->xbutton.button == Button1)
+ mousesel(e, 1);
+}
+
+void
+bmotion(XEvent *e)
+{
+ if (IS_SET(MODE_MOUSE) && !(e->xbutton.state & forceselmod)) {
+ mousereport(e);
+ return;
+ }
+
+ mousesel(e, 0);
+}
+
+void
+cresize(int width, int height)
+{
+ int col, row;
+
+ if (width != 0)
+ win.w = width;
+ if (height != 0)
+ win.h = height;
+
+ col = (win.w - 2 * borderpx) / win.cw;
+ row = (win.h - 2 * borderpx) / win.ch;
+
+ tresize(col, row);
+ xresize(col, row);
+ ttyresize(win.tw, win.th);
+}
+
+void
+xresize(int col, int row)
+{
+ win.tw = MAX(1, col * win.cw);
+ win.th = MAX(1, row * win.ch);
+
+ XFreePixmap(xw.dpy, xw.buf);
+ xw.buf = XCreatePixmap(xw.dpy, xw.win, win.w, win.h,
+ xw.depth);
+ XftDrawChange(xw.draw, xw.buf);
+ xclear(0, 0, win.w, win.h);
+
+ /* resize to new width */
+ xw.specbuf = xrealloc(xw.specbuf, col * sizeof(GlyphFontSpec));
+}
+
+ushort
+sixd_to_16bit(int x)
+{
+ return x == 0 ? 0 : 0x3737 + 0x2828 * x;
+}
+
+int
+xloadcolor(int i, const char *name, Color *ncolor)
+{
+ XRenderColor color = { .alpha = 0xffff };
+
+ if (!name) {
+ if (BETWEEN(i, 16, 255)) { /* 256 color */
+ if (i < 6*6*6+16) { /* same colors as xterm */
+ color.red = sixd_to_16bit( ((i-16)/36)%6 );
+ color.green = sixd_to_16bit( ((i-16)/6) %6 );
+ color.blue = sixd_to_16bit( ((i-16)/1) %6 );
+ } else { /* greyscale */
+ color.red = 0x0808 + 0x0a0a * (i - (6*6*6+16));
+ color.green = color.blue = color.red;
+ }
+ return XftColorAllocValue(xw.dpy, xw.vis,
+ xw.cmap, &color, ncolor);
+ } else
+ name = colorname[i];
+ }
+
+ return XftColorAllocName(xw.dpy, xw.vis, xw.cmap, name, ncolor);
+}
+
+void
+xloadcols(void)
+{
+ int i;
+ static int loaded;
+ Color *cp;
+
+ dc.collen = MAX(LEN(colorname), 256);
+ dc.col = xmalloc(dc.collen * sizeof(Color));
+
+ if (loaded) {
+ for (cp = dc.col; cp < &dc.col[dc.collen]; ++cp)
+ XftColorFree(xw.dpy, xw.vis, xw.cmap, cp);
+ }
+
+ for (i = 0; i < dc.collen; i++)
+ if (!xloadcolor(i, NULL, &dc.col[i])) {
+ if (colorname[i])
+ die("Could not allocate color '%s'\n", colorname[i]);
+ else
+ die("Could not allocate color %d\n", i);
+ }
+
+ /* set alpha value of bg color */
+ if (USE_ARGB) {
+ dc.col[defaultbg].color.alpha = (0xffff * alpha) / OPAQUE;
+ dc.col[defaultbg].pixel &= 0x00111111;
+ dc.col[defaultbg].pixel |= alpha << 24;
+ }
+ loaded = 1;
+}
+
+int
+xsetcolorname(int x, const char *name)
+{
+ Color ncolor;
+
+ if (!BETWEEN(x, 0, dc.collen))
+ return 1;
+
+
+ if (!xloadcolor(x, name, &ncolor))
+ return 1;
+
+ XftColorFree(xw.dpy, xw.vis, xw.cmap, &dc.col[x]);
+ dc.col[x] = ncolor;
+
+ return 0;
+}
+
+void
+xtermclear(int col1, int row1, int col2, int row2)
+{
+ XftDrawRect(xw.draw,
+ &dc.col[IS_SET(MODE_REVERSE) ? defaultfg : defaultbg],
+ borderpx + col1 * win.cw,
+ borderpx + row1 * win.ch,
+ (col2-col1+1) * win.cw,
+ (row2-row1+1) * win.ch);
+}
+
+/*
+ * Absolute coordinates.
+ */
+void
+xclear(int x1, int y1, int x2, int y2)
+{
+ XftDrawRect(xw.draw,
+ &dc.col[IS_SET(MODE_REVERSE)? defaultfg : defaultbg],
+ x1, y1, x2-x1, y2-y1);
+}
+
+void
+xhints(void)
+{
+ XClassHint class = {opt_name ? opt_name : "st",
+ opt_class ? opt_class : "St"};
+ XWMHints wm = {.flags = InputHint, .input = 1};
+ XSizeHints *sizeh;
+
+ sizeh = XAllocSizeHints();
+
+ sizeh->flags = PSize | PResizeInc | PBaseSize;
+ sizeh->height = win.h;
+ sizeh->width = win.w;
+ sizeh->height_inc = win.ch;
+ sizeh->width_inc = win.cw;
+ sizeh->base_height = 2 * borderpx;
+ sizeh->base_width = 2 * borderpx;
+ if (xw.isfixed) {
+ sizeh->flags |= PMaxSize | PMinSize;
+ sizeh->min_width = sizeh->max_width = win.w;
+ sizeh->min_height = sizeh->max_height = win.h;
+ }
+ if (xw.gm & (XValue|YValue)) {
+ sizeh->flags |= USPosition | PWinGravity;
+ sizeh->x = xw.l;
+ sizeh->y = xw.t;
+ sizeh->win_gravity = xgeommasktogravity(xw.gm);
+ }
+
+ XSetWMProperties(xw.dpy, xw.win, NULL, NULL, NULL, 0, sizeh, &wm,
+ &class);
+ XFree(sizeh);
+}
+
+int
+xgeommasktogravity(int mask)
+{
+ switch (mask & (XNegative|YNegative)) {
+ case 0:
+ return NorthWestGravity;
+ case XNegative:
+ return NorthEastGravity;
+ case YNegative:
+ return SouthWestGravity;
+ }
+
+ return SouthEastGravity;
+}
+
+int
+xloadfont(Font *f, FcPattern *pattern)
+{
+ FcPattern *configured;
+ FcPattern *match;
+ FcResult result;
+ XGlyphInfo extents;
+ int wantattr, haveattr;
+
+ /*
+ * Manually configure instead of calling XftMatchFont
+ * so that we can use the configured pattern for
+ * "missing glyph" lookups.
+ */
+ configured = FcPatternDuplicate(pattern);
+ if (!configured)
+ return 1;
+
+ FcConfigSubstitute(NULL, configured, FcMatchPattern);
+ XftDefaultSubstitute(xw.dpy, xw.scr, configured);
+
+ match = FcFontMatch(NULL, configured, &result);
+ if (!match) {
+ FcPatternDestroy(configured);
+ return 1;
+ }
+
+ if (!(f->match = XftFontOpenPattern(xw.dpy, match))) {
+ FcPatternDestroy(configured);
+ FcPatternDestroy(match);
+ return 1;
+ }
+
+ if ((XftPatternGetInteger(pattern, "slant", 0, &wantattr) ==
+ XftResultMatch)) {
+ /*
+ * Check if xft was unable to find a font with the appropriate
+ * slant but gave us one anyway. Try to mitigate.
+ */
+ if ((XftPatternGetInteger(f->match->pattern, "slant", 0,
+ &haveattr) != XftResultMatch) || haveattr < wantattr) {
+ f->badslant = 1;
+ fputs("st: font slant does not match\n", stderr);
+ }
+ }
+
+ if ((XftPatternGetInteger(pattern, "weight", 0, &wantattr) ==
+ XftResultMatch)) {
+ if ((XftPatternGetInteger(f->match->pattern, "weight", 0,
+ &haveattr) != XftResultMatch) || haveattr != wantattr) {
+ f->badweight = 1;
+ fputs("st: font weight does not match\n", stderr);
+ }
+ }
+
+ XftTextExtentsUtf8(xw.dpy, f->match,
+ (const FcChar8 *) ascii_printable,
+ strlen(ascii_printable), &extents);
+
+ f->set = NULL;
+ f->pattern = configured;
+
+ f->ascent = f->match->ascent;
+ f->descent = f->match->descent;
+ f->lbearing = 0;
+ f->rbearing = f->match->max_advance_width;
+
+ f->height = f->ascent + f->descent;
+ f->width = DIVCEIL(extents.xOff, strlen(ascii_printable));
+
+ return 0;
+}
+
+void
+xloadfonts(char *fontstr, double fontsize)
+{
+ FcPattern *pattern;
+ double fontval;
+
+ if (fontstr[0] == '-') {
+ pattern = XftXlfdParse(fontstr, False, False);
+ } else {
+ pattern = FcNameParse((FcChar8 *)fontstr);
+ }
+
+ if (!pattern)
+ die("st: can't open font %s\n", fontstr);
+
+ if (fontsize > 1) {
+ FcPatternDel(pattern, FC_PIXEL_SIZE);
+ FcPatternDel(pattern, FC_SIZE);
+ FcPatternAddDouble(pattern, FC_PIXEL_SIZE, (double)fontsize);
+ usedfontsize = fontsize;
+ } else {
+ if (FcPatternGetDouble(pattern, FC_PIXEL_SIZE, 0, &fontval) ==
+ FcResultMatch) {
+ usedfontsize = fontval;
+ } else if (FcPatternGetDouble(pattern, FC_SIZE, 0, &fontval) ==
+ FcResultMatch) {
+ usedfontsize = -1;
+ } else {
+ /*
+ * Default font size is 12, if none given. This is to
+ * have a known usedfontsize value.
+ */
+ FcPatternAddDouble(pattern, FC_PIXEL_SIZE, 12);
+ usedfontsize = 12;
+ }
+ defaultfontsize = usedfontsize;
+ }
+
+ if (xloadfont(&dc.font, pattern))
+ die("st: can't open font %s\n", fontstr);
+
+ if (usedfontsize < 0) {
+ FcPatternGetDouble(dc.font.match->pattern,
+ FC_PIXEL_SIZE, 0, &fontval);
+ usedfontsize = fontval;
+ if (fontsize == 0)
+ defaultfontsize = fontval;
+ }
+
+ /* Setting character width and height. */
+ win.cw = ceilf(dc.font.width * cwscale);
+ win.ch = ceilf(dc.font.height * chscale);
+ win.cyo = ceilf(dc.font.height * (chscale - 1) / 2);
+
+ FcPatternDel(pattern, FC_SLANT);
+ FcPatternAddInteger(pattern, FC_SLANT, FC_SLANT_ITALIC);
+ if (xloadfont(&dc.ifont, pattern))
+ die("st: can't open font %s\n", fontstr);
+
+ FcPatternDel(pattern, FC_WEIGHT);
+ FcPatternAddInteger(pattern, FC_WEIGHT, FC_WEIGHT_BOLD);
+ if (xloadfont(&dc.ibfont, pattern))
+ die("st: can't open font %s\n", fontstr);
+
+ FcPatternDel(pattern, FC_SLANT);
+ FcPatternAddInteger(pattern, FC_SLANT, FC_SLANT_ROMAN);
+ if (xloadfont(&dc.bfont, pattern))
+ die("st: can't open font %s\n", fontstr);
+
+ FcPatternDestroy(pattern);
+}
+
+void
+xunloadfont(Font *f)
+{
+ XftFontClose(xw.dpy, f->match);
+ FcPatternDestroy(f->pattern);
+ if (f->set)
+ FcFontSetDestroy(f->set);
+}
+
+void
+xunloadfonts(void)
+{
+ /* Free the loaded fonts in the font cache. */
+ while (frclen > 0)
+ XftFontClose(xw.dpy, frc[--frclen].font);
+
+ xunloadfont(&dc.font);
+ xunloadfont(&dc.bfont);
+ xunloadfont(&dc.ifont);
+ xunloadfont(&dc.ibfont);
+}
+
+void
+xinit(int cols, int rows)
+{
+ XGCValues gcvalues;
+ Cursor cursor;
+ Window parent;
+ pid_t thispid = getpid();
+ XColor xmousefg, xmousebg;
+
+ xw.scr = XDefaultScreen(xw.dpy);
+ xw.depth = (USE_ARGB) ? 32: XDefaultDepth(xw.dpy, xw.scr);
+ if (!USE_ARGB)
+ xw.vis = XDefaultVisual(xw.dpy, xw.scr);
+ else {
+ XVisualInfo *vis;
+ XRenderPictFormat *fmt;
+ int nvi;
+ int i;
+
+ XVisualInfo tpl = {
+ .screen = xw.scr,
+ .depth = 32,
+ .class = TrueColor
+ };
+
+ vis = XGetVisualInfo(xw.dpy,
+ VisualScreenMask | VisualDepthMask | VisualClassMask,
+ &tpl, &nvi);
+ xw.vis = NULL;
+ for (i = 0; i < nvi; i++) {
+ fmt = XRenderFindVisualFormat(xw.dpy, vis[i].visual);
+ if (fmt->type == PictTypeDirect && fmt->direct.alphaMask) {
+ xw.vis = vis[i].visual;
+ break;
+ }
+ }
+
+ XFree(vis);
+
+ if (!xw.vis) {
+ fprintf(stderr, "Couldn't find ARGB visual.\n");
+ exit(1);
+ }
+ }
+
+ /* font */
+ if (!FcInit())
+ die("Could not init fontconfig.\n");
+
+ usedfont = (opt_font == NULL)? font : opt_font;
+ xloadfonts(usedfont, 0);
+
+ /* colors */
+ if (!USE_ARGB)
+ xw.cmap = XDefaultColormap(xw.dpy, xw.scr);
+ else
+ xw.cmap = XCreateColormap(xw.dpy, XRootWindow(xw.dpy, xw.scr),
+ xw.vis, None);
+ xloadcols();
+
+ /* adjust fixed window geometry */
+ win.w = 2 * borderpx + cols * win.cw;
+ win.h = 2 * borderpx + rows * win.ch;
+ if (xw.gm & XNegative)
+ xw.l += DisplayWidth(xw.dpy, xw.scr) - win.w - 2;
+ if (xw.gm & YNegative)
+ xw.t += DisplayHeight(xw.dpy, xw.scr) - win.h - 2;
+
+ /* Events */
+ xw.attrs.background_pixel = dc.col[defaultbg].pixel;
+ xw.attrs.border_pixel = dc.col[defaultbg].pixel;
+ xw.attrs.bit_gravity = NorthWestGravity;
+ xw.attrs.event_mask = FocusChangeMask | KeyPressMask
+ | ExposureMask | VisibilityChangeMask | StructureNotifyMask
+ | ButtonMotionMask | ButtonPressMask | ButtonReleaseMask;
+ xw.attrs.colormap = xw.cmap;
+
+ if (!(opt_embed && (parent = strtol(opt_embed, NULL, 0))))
+ parent = XRootWindow(xw.dpy, xw.scr);
+ xw.win = XCreateWindow(xw.dpy, parent, xw.l, xw.t,
+ win.w, win.h, 0, xw.depth, InputOutput,
+ xw.vis, CWBackPixel | CWBorderPixel | CWBitGravity
+ | CWEventMask | CWColormap, &xw.attrs);
+
+ memset(&gcvalues, 0, sizeof(gcvalues));
+ gcvalues.graphics_exposures = False;
+ xw.buf = XCreatePixmap(xw.dpy, xw.win, win.w, win.h, xw.depth);
+ dc.gc = XCreateGC(xw.dpy, (USE_ARGB) ? xw.buf: parent,
+ GCGraphicsExposures, &gcvalues);
+ XSetForeground(xw.dpy, dc.gc, dc.col[defaultbg].pixel);
+ XFillRectangle(xw.dpy, xw.buf, dc.gc, 0, 0, win.w, win.h);
+
+ /* font spec buffer */
+ xw.specbuf = xmalloc(cols * sizeof(GlyphFontSpec));
+
+ /* Xft rendering context */
+ xw.draw = XftDrawCreate(xw.dpy, xw.buf, xw.vis, xw.cmap);
+
+ /* input methods */
+ if ((xw.xim = XOpenIM(xw.dpy, NULL, NULL, NULL)) == NULL) {
+ XSetLocaleModifiers("@im=local");
+ if ((xw.xim = XOpenIM(xw.dpy, NULL, NULL, NULL)) == NULL) {
+ XSetLocaleModifiers("@im=");
+ if ((xw.xim = XOpenIM(xw.dpy,
+ NULL, NULL, NULL)) == NULL) {
+ die("XOpenIM failed. Could not open input"
+ " device.\n");
+ }
+ }
+ }
+ xw.xic = XCreateIC(xw.xim, XNInputStyle, XIMPreeditNothing
+ | XIMStatusNothing, XNClientWindow, xw.win,
+ XNFocusWindow, xw.win, NULL);
+ if (xw.xic == NULL)
+ die("XCreateIC failed. Could not obtain input method.\n");
+
+ /* white cursor, black outline */
+ cursor = XCreateFontCursor(xw.dpy, mouseshape);
+ XDefineCursor(xw.dpy, xw.win, cursor);
+
+ if (XParseColor(xw.dpy, xw.cmap, colorname[mousefg], &xmousefg) == 0) {
+ xmousefg.red = 0xffff;
+ xmousefg.green = 0xffff;
+ xmousefg.blue = 0xffff;
+ }
+
+ if (XParseColor(xw.dpy, xw.cmap, colorname[mousebg], &xmousebg) == 0) {
+ xmousebg.red = 0x0000;
+ xmousebg.green = 0x0000;
+ xmousebg.blue = 0x0000;
+ }
+
+ XRecolorCursor(xw.dpy, cursor, &xmousefg, &xmousebg);
+
+ xw.xembed = XInternAtom(xw.dpy, "_XEMBED", False);
+ xw.wmdeletewin = XInternAtom(xw.dpy, "WM_DELETE_WINDOW", False);
+ xw.netwmname = XInternAtom(xw.dpy, "_NET_WM_NAME", False);
+ XSetWMProtocols(xw.dpy, xw.win, &xw.wmdeletewin, 1);
+
+ xw.netwmpid = XInternAtom(xw.dpy, "_NET_WM_PID", False);
+ XChangeProperty(xw.dpy, xw.win, xw.netwmpid, XA_CARDINAL, 32,
+ PropModeReplace, (uchar *)&thispid, 1);
+
+ win.mode = MODE_NUMLOCK;
+ resettitle();
+ XMapWindow(xw.dpy, xw.win);
+ xhints();
+ XSync(xw.dpy, False);
+
+ clock_gettime(CLOCK_MONOTONIC, &xsel.tclick1);
+ clock_gettime(CLOCK_MONOTONIC, &xsel.tclick2);
+ xsel.primary = NULL;
+ xsel.clipboard = NULL;
+ xsel.xtarget = XInternAtom(xw.dpy, "UTF8_STRING", 0);
+ if (xsel.xtarget == None)
+ xsel.xtarget = XA_STRING;
+}
+
+int
+xmakeglyphfontspecs(XftGlyphFontSpec *specs, const Glyph *glyphs, int len, int x, int y)
+{
+ float winx = borderpx + x * win.cw, winy = borderpx + y * win.ch, xp, yp;
+ ushort mode, prevmode = USHRT_MAX;
+ Font *font = &dc.font;
+ int frcflags = FRC_NORMAL;
+ float runewidth = win.cw;
+ Rune rune;
+ FT_UInt glyphidx;
+ FcResult fcres;
+ FcPattern *fcpattern, *fontpattern;
+ FcFontSet *fcsets[] = { NULL };
+ FcCharSet *fccharset;
+ int i, f, numspecs = 0;
+
+ for (i = 0, xp = winx, yp = winy + font->ascent + win.cyo; i < len; ++i) {
+ /* Fetch rune and mode for current glyph. */
+ rune = glyphs[i].u;
+ mode = glyphs[i].mode;
+
+ /* Skip dummy wide-character spacing. */
+ if (mode == ATTR_WDUMMY)
+ continue;
+
+ /* Determine font for glyph if different from previous glyph. */
+ if (prevmode != mode) {
+ prevmode = mode;
+ font = &dc.font;
+ frcflags = FRC_NORMAL;
+ runewidth = win.cw * ((mode & ATTR_WIDE) ? 2.0f : 1.0f);
+ if ((mode & ATTR_ITALIC) && (mode & ATTR_BOLD)) {
+ font = &dc.ibfont;
+ frcflags = FRC_ITALICBOLD;
+ } else if (mode & ATTR_ITALIC) {
+ font = &dc.ifont;
+ frcflags = FRC_ITALIC;
+ } else if (mode & ATTR_BOLD) {
+ font = &dc.bfont;
+ frcflags = FRC_BOLD;
+ }
+ yp = winy + font->ascent + win.cyo;
+ }
+
+ /* Lookup character index with default font. */
+ glyphidx = XftCharIndex(xw.dpy, font->match, rune);
+ if (glyphidx) {
+ specs[numspecs].font = font->match;
+ specs[numspecs].glyph = glyphidx;
+ specs[numspecs].x = (short)xp;
+ specs[numspecs].y = (short)yp;
+ xp += runewidth;
+ numspecs++;
+ continue;
+ }
+
+ /* Fallback on font cache, search the font cache for match. */
+ for (f = 0; f < frclen; f++) {
+ glyphidx = XftCharIndex(xw.dpy, frc[f].font, rune);
+ /* Everything correct. */
+ if (glyphidx && frc[f].flags == frcflags)
+ break;
+ /* We got a default font for a not found glyph. */
+ if (!glyphidx && frc[f].flags == frcflags
+ && frc[f].unicodep == rune) {
+ break;
+ }
+ }
+
+ /* Nothing was found. Use fontconfig to find matching font. */
+ if (f >= frclen) {
+ if (!font->set)
+ font->set = FcFontSort(0, font->pattern,
+ 1, 0, &fcres);
+ fcsets[0] = font->set;
+
+ /*
+ * Nothing was found in the cache. Now use
+ * some dozen of Fontconfig calls to get the
+ * font for one single character.
+ *
+ * Xft and fontconfig are design failures.
+ */
+ fcpattern = FcPatternDuplicate(font->pattern);
+ fccharset = FcCharSetCreate();
+
+ FcCharSetAddChar(fccharset, rune);
+ FcPatternAddCharSet(fcpattern, FC_CHARSET,
+ fccharset);
+ FcPatternAddBool(fcpattern, FC_SCALABLE, 1);
+
+ FcConfigSubstitute(0, fcpattern,
+ FcMatchPattern);
+ FcDefaultSubstitute(fcpattern);
+
+ fontpattern = FcFontSetMatch(0, fcsets, 1,
+ fcpattern, &fcres);
+
+ /*
+ * Overwrite or create the new cache entry.
+ */
+ if (frclen >= LEN(frc)) {
+ frclen = LEN(frc) - 1;
+ XftFontClose(xw.dpy, frc[frclen].font);
+ frc[frclen].unicodep = 0;
+ }
+
+ frc[frclen].font = XftFontOpenPattern(xw.dpy,
+ fontpattern);
+ if (!frc[frclen].font)
+ die("XftFontOpenPattern failed seeking fallback font: %s\n",
+ strerror(errno));
+ frc[frclen].flags = frcflags;
+ frc[frclen].unicodep = rune;
+
+ glyphidx = XftCharIndex(xw.dpy, frc[frclen].font, rune);
+
+ f = frclen;
+ frclen++;
+
+ FcPatternDestroy(fcpattern);
+ FcCharSetDestroy(fccharset);
+ }
+
+ specs[numspecs].font = frc[f].font;
+ specs[numspecs].glyph = glyphidx;
+ specs[numspecs].x = (short)xp;
+ specs[numspecs].y = (short)yp;
+ xp += runewidth;
+ numspecs++;
+ }
+
+ return numspecs;
+}
+
+void
+xdrawglyphfontspecs(const XftGlyphFontSpec *specs, Glyph base, int len, int x, int y)
+{
+ int charlen = len * ((base.mode & ATTR_WIDE) ? 2 : 1);
+ int winx = borderpx + x * win.cw, winy = borderpx + y * win.ch,
+ width = charlen * win.cw;
+ Color *fg, *bg, *temp, revfg, revbg, truefg, truebg;
+ XRenderColor colfg, colbg;
+ XRectangle r;
+
+ /* Fallback on color display for attributes not supported by the font */
+ if (base.mode & ATTR_ITALIC && base.mode & ATTR_BOLD) {
+ if (dc.ibfont.badslant || dc.ibfont.badweight)
+ base.fg = defaultattr;
+ } else if ((base.mode & ATTR_ITALIC && dc.ifont.badslant) ||
+ (base.mode & ATTR_BOLD && dc.bfont.badweight)) {
+ base.fg = defaultattr;
+ }
+
+ if (IS_TRUECOL(base.fg)) {
+ colfg.alpha = 0xffff;
+ colfg.red = TRUERED(base.fg);
+ colfg.green = TRUEGREEN(base.fg);
+ colfg.blue = TRUEBLUE(base.fg);
+ XftColorAllocValue(xw.dpy, xw.vis, xw.cmap, &colfg, &truefg);
+ fg = &truefg;
+ } else {
+ fg = &dc.col[base.fg];
+ }
+
+ if (IS_TRUECOL(base.bg)) {
+ colbg.alpha = 0xffff;
+ colbg.green = TRUEGREEN(base.bg);
+ colbg.red = TRUERED(base.bg);
+ colbg.blue = TRUEBLUE(base.bg);
+ XftColorAllocValue(xw.dpy, xw.vis, xw.cmap, &colbg, &truebg);
+ bg = &truebg;
+ } else {
+ bg = &dc.col[base.bg];
+ }
+
+ /* Change basic system colors [0-7] to bright system colors [8-15] */
+ if ((base.mode & ATTR_BOLD_FAINT) == ATTR_BOLD && BETWEEN(base.fg, 0, 7))
+ fg = &dc.col[base.fg];
+
+ if (IS_SET(MODE_REVERSE)) {
+ if (fg == &dc.col[defaultfg]) {
+ fg = &dc.col[defaultbg];
+ } else {
+ colfg.red = ~fg->color.red;
+ colfg.green = ~fg->color.green;
+ colfg.blue = ~fg->color.blue;
+ colfg.alpha = fg->color.alpha;
+ XftColorAllocValue(xw.dpy, xw.vis, xw.cmap, &colfg,
+ &revfg);
+ fg = &revfg;
+ }
+
+ if (bg == &dc.col[defaultbg]) {
+ bg = &dc.col[defaultfg];
+ } else {
+ colbg.red = ~bg->color.red;
+ colbg.green = ~bg->color.green;
+ colbg.blue = ~bg->color.blue;
+ colbg.alpha = bg->color.alpha;
+ XftColorAllocValue(xw.dpy, xw.vis, xw.cmap, &colbg,
+ &revbg);
+ bg = &revbg;
+ }
+ }
+
+ if ((base.mode & ATTR_BOLD_FAINT) == ATTR_FAINT) {
+ colfg.red = fg->color.red / 2;
+ colfg.green = fg->color.green / 2;
+ colfg.blue = fg->color.blue / 2;
+ colfg.alpha = fg->color.alpha;
+ XftColorAllocValue(xw.dpy, xw.vis, xw.cmap, &colfg, &revfg);
+ fg = &revfg;
+ }
+
+ if (base.mode & ATTR_REVERSE) {
+ temp = fg;
+ fg = bg;
+ bg = temp;
+ }
+
+ if (base.mode & ATTR_BLINK && win.mode & MODE_BLINK)
+ fg = bg;
+
+ if (base.mode & ATTR_INVISIBLE)
+ fg = bg;
+
+ /* Intelligent cleaning up of the borders. */
+ if (x == 0) {
+ xclear(0, (y == 0)? 0 : winy, borderpx,
+ winy + win.ch +
+ ((winy + win.ch >= borderpx + win.th)? win.h : 0));
+ }
+ if (winx + width >= borderpx + win.tw) {
+ xclear(winx + width, (y == 0)? 0 : winy, win.w,
+ ((winy + win.ch >= borderpx + win.th)? win.h : (winy + win.ch)));
+ }
+ if (y == 0)
+ xclear(winx, 0, winx + width, borderpx);
+ if (winy + win.ch >= borderpx + win.th)
+ xclear(winx, winy + win.ch, winx + width, win.h);
+
+ /* Clean up the region we want to draw to. */
+ XftDrawRect(xw.draw, bg, winx, winy, width, win.ch);
+
+ /* Set the clip region because Xft is sometimes dirty. */
+ r.x = 0;
+ r.y = 0;
+ r.height = win.ch;
+ r.width = width;
+ XftDrawSetClipRectangles(xw.draw, winx, winy, &r, 1);
+
+ /* Render the glyphs. */
+ XftDrawGlyphFontSpec(xw.draw, fg, specs, len);
+
+ /* Render underline and strikethrough. */
+ if (base.mode & ATTR_UNDERLINE) {
+ XftDrawRect(xw.draw, fg, winx, winy + win.cyo + dc.font.ascent + 1,
+ width, 1);
+ }
+
+ if (base.mode & ATTR_STRUCK) {
+ XftDrawRect(xw.draw, fg, winx, winy + win.cyo + 2 * dc.font.ascent / 3,
+ width, 1);
+ }
+
+ /* Reset clip to none. */
+ XftDrawSetClip(xw.draw, 0);
+}
+
+void
+xdrawglyph(Glyph g, int x, int y)
+{
+ int numspecs;
+ XftGlyphFontSpec spec;
+
+ numspecs = xmakeglyphfontspecs(&spec, &g, 1, x, y);
+ xdrawglyphfontspecs(&spec, g, numspecs, x, y);
+}
+
+void
+xdrawcursor(int cx, int cy, Glyph g, int ox, int oy, Glyph og)
+{
+ Color drawcol;
+
+ /* remove the old cursor */
+ if (selected(ox, oy))
+ og.mode ^= ATTR_REVERSE;
+ xdrawglyph(og, ox, oy);
+
+ if (IS_SET(MODE_HIDE))
+ return;
+
+ /*
+ * Select the right color for the right mode.
+ */
+ g.mode &= ATTR_BOLD|ATTR_ITALIC|ATTR_UNDERLINE|ATTR_STRUCK|ATTR_WIDE;
+
+ if (IS_SET(MODE_REVERSE)) {
+ g.mode |= ATTR_REVERSE;
+ g.bg = defaultfg;
+ if (selected(cx, cy)) {
+ drawcol = dc.col[defaultcs];
+ g.fg = defaultrcs;
+ } else {
+ drawcol = dc.col[defaultrcs];
+ g.fg = defaultcs;
+ }
+ } else {
+ if (selected(cx, cy)) {
+ g.fg = defaultfg;
+ g.bg = defaultrcs;
+ } else {
+ g.fg = defaultbg;
+ g.bg = defaultcs;
+ }
+ drawcol = dc.col[g.bg];
+ }
+
+ /* draw the new one */
+ if (IS_SET(MODE_FOCUSED)) {
+ switch (win.cursor) {
+ case 7: /* st extension: snowman (U+2603) */
+ g.u = 0x2603;
+ case 0: /* Blinking Block */
+ case 1: /* Blinking Block (Default) */
+ case 2: /* Steady Block */
+ xdrawglyph(g, cx, cy);
+ break;
+ case 3: /* Blinking Underline */
+ case 4: /* Steady Underline */
+ XftDrawRect(xw.draw, &drawcol,
+ borderpx + cx * win.cw,
+ borderpx + (cy + 1) * win.ch - \
+ cursorthickness,
+ win.cw, cursorthickness);
+ break;
+ case 5: /* Blinking bar */
+ case 6: /* Steady bar */
+ XftDrawRect(xw.draw, &drawcol,
+ borderpx + cx * win.cw,
+ borderpx + cy * win.ch,
+ cursorthickness, win.ch);
+ break;
+ }
+ } else {
+ XftDrawRect(xw.draw, &drawcol,
+ borderpx + cx * win.cw,
+ borderpx + cy * win.ch,
+ win.cw - 1, 1);
+ XftDrawRect(xw.draw, &drawcol,
+ borderpx + cx * win.cw,
+ borderpx + cy * win.ch,
+ 1, win.ch - 1);
+ XftDrawRect(xw.draw, &drawcol,
+ borderpx + (cx + 1) * win.cw - 1,
+ borderpx + cy * win.ch,
+ 1, win.ch - 1);
+ XftDrawRect(xw.draw, &drawcol,
+ borderpx + cx * win.cw,
+ borderpx + (cy + 1) * win.ch - 1,
+ win.cw, 1);
+ }
+}
+
+void
+xsetenv(void)
+{
+ char buf[sizeof(long) * 8 + 1];
+
+ snprintf(buf, sizeof(buf), "%lu", xw.win);
+ setenv("WINDOWID", buf, 1);
+}
+
+void
+xsettitle(char *p)
+{
+ XTextProperty prop;
+ DEFAULT(p, opt_title);
+
+ Xutf8TextListToTextProperty(xw.dpy, &p, 1, XUTF8StringStyle,
+ &prop);
+ XSetWMName(xw.dpy, xw.win, &prop);
+ XSetTextProperty(xw.dpy, xw.win, &prop, xw.netwmname);
+ XFree(prop.value);
+}
+
+int
+xstartdraw(void)
+{
+ return IS_SET(MODE_VISIBLE);
+}
+
+void
+xdrawline(Line line, int x1, int y1, int x2)
+{
+ int i, x, ox, numspecs;
+ Glyph base, new;
+ XftGlyphFontSpec *specs = xw.specbuf;
+
+ numspecs = xmakeglyphfontspecs(specs, &line[x1], x2 - x1, x1, y1);
+ i = ox = 0;
+ for (x = x1; x < x2 && i < numspecs; x++) {
+ new = line[x];
+ if (new.mode == ATTR_WDUMMY)
+ continue;
+ if (selected(x, y1))
+ new.mode ^= ATTR_REVERSE;
+ if (i > 0 && ATTRCMP(base, new)) {
+ xdrawglyphfontspecs(specs, base, i, ox, y1);
+ specs += i;
+ numspecs -= i;
+ i = 0;
+ }
+ if (i == 0) {
+ ox = x;
+ base = new;
+ }
+ i++;
+ }
+ if (i > 0)
+ xdrawglyphfontspecs(specs, base, i, ox, y1);
+}
+
+void
+xfinishdraw(void)
+{
+ XCopyArea(xw.dpy, xw.buf, xw.win, dc.gc, 0, 0, win.w,
+ win.h, 0, 0);
+ XSetForeground(xw.dpy, dc.gc,
+ dc.col[IS_SET(MODE_REVERSE)?
+ defaultfg : defaultbg].pixel);
+}
+
+void
+expose(XEvent *ev)
+{
+ redraw();
+}
+
+void
+visibility(XEvent *ev)
+{
+ XVisibilityEvent *e = &ev->xvisibility;
+
+ MODBIT(win.mode, e->state != VisibilityFullyObscured, MODE_VISIBLE);
+}
+
+void
+unmap(XEvent *ev)
+{
+ win.mode &= ~MODE_VISIBLE;
+}
+
+void
+xsetpointermotion(int set)
+{
+ MODBIT(xw.attrs.event_mask, set, PointerMotionMask);
+ XChangeWindowAttributes(xw.dpy, xw.win, CWEventMask, &xw.attrs);
+}
+
+void
+xsetmode(int set, unsigned int flags)
+{
+ int mode = win.mode;
+ MODBIT(win.mode, set, flags);
+ if ((win.mode & MODE_REVERSE) != (mode & MODE_REVERSE))
+ redraw();
+}
+
+int
+xsetcursor(int cursor)
+{
+ DEFAULT(cursor, 1);
+ if (!BETWEEN(cursor, 0, 6))
+ return 1;
+ win.cursor = cursor;
+ return 0;
+}
+
+void
+xseturgency(int add)
+{
+ XWMHints *h = XGetWMHints(xw.dpy, xw.win);
+
+ MODBIT(h->flags, add, XUrgencyHint);
+ XSetWMHints(xw.dpy, xw.win, h);
+ XFree(h);
+}
+
+void
+xbell(void)
+{
+ if (!(IS_SET(MODE_FOCUSED)))
+ xseturgency(1);
+ if (bellvolume)
+ XkbBell(xw.dpy, xw.win, bellvolume, (Atom)NULL);
+}
+
+void
+focus(XEvent *ev)
+{
+ XFocusChangeEvent *e = &ev->xfocus;
+
+ if (e->mode == NotifyGrab)
+ return;
+
+ if (ev->type == FocusIn) {
+ XSetICFocus(xw.xic);
+ win.mode |= MODE_FOCUSED;
+ xseturgency(0);
+ if (IS_SET(MODE_FOCUS))
+ ttywrite("\033[I", 3, 0);
+ } else {
+ XUnsetICFocus(xw.xic);
+ win.mode &= ~MODE_FOCUSED;
+ if (IS_SET(MODE_FOCUS))
+ ttywrite("\033[O", 3, 0);
+ }
+}
+
+int
+match(uint mask, uint state)
+{
+ return mask == XK_ANY_MOD || mask == (state & ~ignoremod);
+}
+
+char*
+kmap(KeySym k, uint state)
+{
+ Key *kp;
+ int i;
+
+ /* Check for mapped keys out of X11 function keys. */
+ for (i = 0; i < LEN(mappedkeys); i++) {
+ if (mappedkeys[i] == k)
+ break;
+ }
+ if (i == LEN(mappedkeys)) {
+ if ((k & 0xFFFF) < 0xFD00)
+ return NULL;
+ }
+
+ for (kp = key; kp < key + LEN(key); kp++) {
+ if (kp->k != k)
+ continue;
+
+ if (!match(kp->mask, state))
+ continue;
+
+ if (IS_SET(MODE_APPKEYPAD) ? kp->appkey < 0 : kp->appkey > 0)
+ continue;
+ if (IS_SET(MODE_NUMLOCK) && kp->appkey == 2)
+ continue;
+
+ if (IS_SET(MODE_APPCURSOR) ? kp->appcursor < 0 : kp->appcursor > 0)
+ continue;
+
+ return kp->s;
+ }
+
+ return NULL;
+}
+
+void
+kpress(XEvent *ev)
+{
+ XKeyEvent *e = &ev->xkey;
+ KeySym ksym;
+ char buf[32], *customkey;
+ int len;
+ Rune c;
+ Status status;
+ Shortcut *bp;
+
+ if (IS_SET(MODE_KBDLOCK))
+ return;
+
+ len = XmbLookupString(xw.xic, e, buf, sizeof buf, &ksym, &status);
+ /* 1. shortcuts */
+ for (bp = shortcuts; bp < shortcuts + LEN(shortcuts); bp++) {
+ if (ksym == bp->keysym && match(bp->mod, e->state)) {
+ bp->func(&(bp->arg));
+ return;
+ }
+ }
+
+ /* 2. custom keys from config.h */
+ if ((customkey = kmap(ksym, e->state))) {
+ ttywrite(customkey, strlen(customkey), 1);
+ return;
+ }
+
+ /* 3. composed string from input method */
+ if (len == 0)
+ return;
+ if (len == 1 && e->state & Mod1Mask) {
+ if (IS_SET(MODE_8BIT)) {
+ if (*buf < 0177) {
+ c = *buf | 0x80;
+ len = utf8encode(c, buf);
+ }
+ } else {
+ buf[1] = buf[0];
+ buf[0] = '\033';
+ len = 2;
+ }
+ }
+ ttywrite(buf, len, 1);
+}
+
+
+void
+cmessage(XEvent *e)
+{
+ /*
+ * See xembed specs
+ * http://standards.freedesktop.org/xembed-spec/xembed-spec-latest.html
+ */
+ if (e->xclient.message_type == xw.xembed && e->xclient.format == 32) {
+ if (e->xclient.data.l[1] == XEMBED_FOCUS_IN) {
+ win.mode |= MODE_FOCUSED;
+ xseturgency(0);
+ } else if (e->xclient.data.l[1] == XEMBED_FOCUS_OUT) {
+ win.mode &= ~MODE_FOCUSED;
+ }
+ } else if (e->xclient.data.l[0] == xw.wmdeletewin) {
+ ttyhangup();
+ exit(0);
+ }
+}
+
+void
+resize(XEvent *e)
+{
+ if (e->xconfigure.width == win.w && e->xconfigure.height == win.h)
+ return;
+
+ cresize(e->xconfigure.width, e->xconfigure.height);
+}
+
+void
+run(void)
+{
+ XEvent ev;
+ int w = win.w, h = win.h;
+ fd_set rfd;
+ int xfd = XConnectionNumber(xw.dpy), xev, blinkset = 0, dodraw = 0;
+ int ttyfd;
+ struct timespec drawtimeout, *tv = NULL, now, last, lastblink;
+ long deltatime;
+
+ /* Waiting for window mapping */
+ do {
+ XNextEvent(xw.dpy, &ev);
+ /*
+ * This XFilterEvent call is required because of XOpenIM. It
+ * does filter out the key event and some client message for
+ * the input method too.
+ */
+ if (XFilterEvent(&ev, None))
+ continue;
+ if (ev.type == ConfigureNotify) {
+ w = ev.xconfigure.width;
+ h = ev.xconfigure.height;
+ }
+ } while (ev.type != MapNotify);
+
+ ttyfd = ttynew(opt_line, shell, opt_io, opt_cmd);
+ cresize(w, h);
+
+ clock_gettime(CLOCK_MONOTONIC, &last);
+ lastblink = last;
+
+ for (xev = actionfps;;) {
+ FD_ZERO(&rfd);
+ FD_SET(ttyfd, &rfd);
+ FD_SET(xfd, &rfd);
+
+ if (pselect(MAX(xfd, ttyfd)+1, &rfd, NULL, NULL, tv, NULL) < 0) {
+ if (errno == EINTR)
+ continue;
+ die("select failed: %s\n", strerror(errno));
+ }
+ if (FD_ISSET(ttyfd, &rfd)) {
+ ttyread();
+ if (blinktimeout) {
+ blinkset = tattrset(ATTR_BLINK);
+ if (!blinkset)
+ MODBIT(win.mode, 0, MODE_BLINK);
+ }
+ }
+
+ if (FD_ISSET(xfd, &rfd))
+ xev = actionfps;
+
+ clock_gettime(CLOCK_MONOTONIC, &now);
+ drawtimeout.tv_sec = 0;
+ drawtimeout.tv_nsec = (1000 * 1E6)/ xfps;
+ tv = &drawtimeout;
+
+ dodraw = 0;
+ if (blinktimeout && TIMEDIFF(now, lastblink) > blinktimeout) {
+ tsetdirtattr(ATTR_BLINK);
+ win.mode ^= MODE_BLINK;
+ lastblink = now;
+ dodraw = 1;
+ }
+ deltatime = TIMEDIFF(now, last);
+ if (deltatime > 1000 / (xev ? xfps : actionfps)) {
+ dodraw = 1;
+ last = now;
+ }
+
+ if (dodraw) {
+ while (XPending(xw.dpy)) {
+ XNextEvent(xw.dpy, &ev);
+ if (XFilterEvent(&ev, None))
+ continue;
+ if (handler[ev.type])
+ (handler[ev.type])(&ev);
+ }
+
+ draw();
+ XFlush(xw.dpy);
+
+ if (xev && !FD_ISSET(xfd, &rfd))
+ xev--;
+ if (!FD_ISSET(ttyfd, &rfd) && !FD_ISSET(xfd, &rfd)) {
+ if (blinkset) {
+ if (TIMEDIFF(now, lastblink) \
+ > blinktimeout) {
+ drawtimeout.tv_nsec = 1000;
+ } else {
+ drawtimeout.tv_nsec = (1E6 * \
+ (blinktimeout - \
+ TIMEDIFF(now,
+ lastblink)));
+ }
+ drawtimeout.tv_sec = \
+ drawtimeout.tv_nsec / 1E9;
+ drawtimeout.tv_nsec %= (long)1E9;
+ } else {
+ tv = NULL;
+ }
+ }
+ }
+ }
+}
+
+int
+resource_load(XrmDatabase db, char *name, enum resource_type rtype, void *dst)
+{
+ char **sdst = dst;
+ int *idst = dst;
+ float *fdst = dst;
+
+ char fullname[256];
+ char fullclass[256];
+ char *type;
+ XrmValue ret;
+
+ snprintf(fullname, sizeof(fullname), "%s.%s",
+ opt_name ? opt_name : "st", name);
+ snprintf(fullclass, sizeof(fullclass), "%s.%s",
+ opt_class ? opt_class : "St", name);
+ fullname[sizeof(fullname) - 1] = fullclass[sizeof(fullclass) - 1] = '\0';
+
+ XrmGetResource(db, fullname, fullclass, &type, &ret);
+ if (ret.addr == NULL || strncmp("String", type, 64))
+ return 1;
+
+ switch (rtype) {
+ case STRING:
+ *sdst = ret.addr;
+ break;
+ case INTEGER:
+ *idst = strtoul(ret.addr, NULL, 10);
+ break;
+ case FLOAT:
+ *fdst = strtof(ret.addr, NULL);
+ break;
+ }
+ return 0;
+}
+
+void
+config_init(void)
+{
+ char *resm;
+ XrmDatabase db;
+ ResourcePref *p;
+
+ XrmInitialize();
+ resm = XResourceManagerString(xw.dpy);
+ if (!resm)
+ return;
+
+ db = XrmGetStringDatabase(resm);
+ for (p = resources; p < resources + LEN(resources); p++)
+ resource_load(db, p->name, p->type, p->dst);
+}
+
+void
+usage(void)
+{
+ die("usage: %s [-aiv] [-c class] [-f font] [-g geometry]"
+ " [-n name] [-o file]\n"
+ " [-T title] [-t title] [-w windowid]"
+ " [[-e] command [args ...]]\n"
+ " %s [-aiv] [-c class] [-f font] [-g geometry]"
+ " [-n name] [-o file]\n"
+ " [-T title] [-t title] [-w windowid] -l line"
+ " [stty_args ...]\n", argv0, argv0);
+}
+
+int
+main(int argc, char *argv[])
+{
+ xw.l = xw.t = 0;
+ xw.isfixed = False;
+ win.cursor = cursorshape;
+
+ ARGBEGIN {
+ case 'a':
+ allowaltscreen = 0;
+ break;
+ case 'c':
+ opt_class = EARGF(usage());
+ break;
+ case 'e':
+ if (argc > 0)
+ --argc, ++argv;
+ goto run;
+ case 'f':
+ opt_font = EARGF(usage());
+ break;
+ case 'g':
+ xw.gm = XParseGeometry(EARGF(usage()),
+ &xw.l, &xw.t, &cols, &rows);
+ break;
+ case 'i':
+ xw.isfixed = 1;
+ break;
+ case 'o':
+ opt_io = EARGF(usage());
+ break;
+ case 'l':
+ opt_line = EARGF(usage());
+ break;
+ case 'n':
+ opt_name = EARGF(usage());
+ break;
+ case 't':
+ case 'T':
+ opt_title = EARGF(usage());
+ break;
+ case 'w':
+ opt_embed = EARGF(usage());
+ break;
+ case 'v':
+ die("%s " VERSION " (c) 2010-2016 st engineers\n", argv0);
+ break;
+ default:
+ usage();
+ } ARGEND;
+
+run:
+ if (argc > 0) /* eat all remaining arguments */
+ opt_cmd = argv;
+
+ if (!opt_title)
+ opt_title = (opt_line || !opt_cmd) ? "st" : opt_cmd[0];
+
+ setlocale(LC_CTYPE, "");
+ XSetLocaleModifiers("");
+
+ if(!(xw.dpy = XOpenDisplay(NULL)))
+ die("Can't open display\n");
+
+ config_init();
+ cols = MAX(cols, 1);
+ rows = MAX(rows, 1);
+ tnew(cols, rows);
+ xinit(cols, rows);
+ xsetenv();
+ selinit();
+ run();
+
+ return 0;
+}
diff --git a/suckless/st/x.o b/suckless/st/x.o
new file mode 100644
index 00000000..b52b9aaa
Binary files /dev/null and b/suckless/st/x.o differ