diff --git a/hosts/desktop/default.nix b/hosts/desktop/default.nix
index 322799c..358bb6f 100644
--- a/hosts/desktop/default.nix
+++ b/hosts/desktop/default.nix
@@ -42,8 +42,9 @@
services.udev.packages = [unstable.utsushi];
- hyprland.enable = true; # Window Manager
+ # hyprland.enable = true;
kde.enable = true;
+ awesome.enable = true;
environment = {
systemPackages = with pkgs; [
diff --git a/modules/desktops/awesome.nix b/modules/desktops/awesome.nix
new file mode 100644
index 0000000..cdb87bd
--- /dev/null
+++ b/modules/desktops/awesome.nix
@@ -0,0 +1,59 @@
+#
+# AwesomeWM Configuration
+# Enable with "awesome.enable = true;"
+#
+{
+ config,
+ lib,
+ pkgs,
+ vars,
+ ...
+}:
+with lib; {
+ options = {
+ awesome = {
+ enable = mkOption {
+ type = types.bool;
+ default = false;
+ };
+ };
+ };
+
+ config = mkIf (config.awesome.enable) {
+ services = {
+ xserver = {
+ enable = true;
+ layout = "de";
+ libinput.enable = true;
+ modules = [pkgs.xf86_input_wacom];
+ wacom.enable = true;
+ windowManager.awesome.enable = true;
+ screenSection = ''
+ Option "metamodes" "DP-4: 2560x1440_144 +1920+0, DP-0: 1920x1080_75 +0+0"
+ '';
+ };
+ };
+
+ home-manager.users.${vars.user} = {
+ home.file.".config/awesome/" = {
+ source = ../../rsc/config/awesome;
+ recursive = true;
+ };
+ home.file.".config/picom/"={
+ source = ../../rsc/config/picom;
+ recursive = true;
+ };
+ };
+ environment = {
+ systemPackages = with pkgs;
+ [
+ picom
+ rofi
+ redshift
+ xbindkeys
+ clipmenu
+ ]
+ ++ (with unstable; []);
+ };
+ };
+}
diff --git a/modules/desktops/default.nix b/modules/desktops/default.nix
index c5d69b9..7b2a303 100644
--- a/modules/desktops/default.nix
+++ b/modules/desktops/default.nix
@@ -2,4 +2,5 @@
./hyprland.nix
./kde.nix
./gnome.nix
+ ./awesome.nix
]
diff --git a/rsc/config/awesome/autostart.lua b/rsc/config/awesome/autostart.lua
new file mode 100644
index 0000000..7ef8a8e
--- /dev/null
+++ b/rsc/config/awesome/autostart.lua
@@ -0,0 +1,30 @@
+local autostart = {}
+
+function autostart.exec(awful)
+ awful.spawn.with_shell("picom --backend glx --xrender-sync-fence")
+ -- awful.spawn.with_shell("nitrogen --restore")
+ awful.spawn.with_shell("redshift -x &&redshift -O 3500")
+ awful.spawn.with_shell("/usr/lib/polkit-kde-authentication-agent-1")
+ awful.spawn.with_shell("flatpak run io.github.spacingbat3.webcord --start-minimized")
+ awful.spawn.with_shell("playerctld daemon")
+ awful.spawn.with_shell("xbindkeys")
+ -- awful.spawn.with_shell("easyeffects --gapplication-service")
+ awful.spawn.with_shell("$HOME/.config/nanoKontroller/initSink.sh")
+ awful.spawn.with_shell("$HOME/.config/nanoKontroller/audio1.sh")
+ awful.spawn.with_shell(
+ "python3 $HOME/.local/bin/nanoKontroller --config $HOME/.config/nanoKontroller/nanoKontroller.ini")
+ -- awful.spawn.with_shell("flatpak run com.spotify.Client")
+ -- awful.spawn.with_shell("/usr/bin/emacs --daemon")
+ awful.spawn.with_shell("razer-cli --dpi 400")
+ awful.spawn.with_shell("electron-mail")
+ awful.spawn.with_shell("clipmenud")
+ awful.spawn.with_shell("nextcloud --background")
+ awful.spawn.with_shell("openrgb --server --profile default")
+ -- awful.spawn.with_shell("autocutsel -fork -selection CLIPBOARD")
+ -- awful.spawn.with_shell("autocutsel -fork -selection PRIMARY")
+ awful.spawn.with_shell("xmousepasteblock")
+ awful.spawn.with_shell("xinput map-to-output 20 DVI-I")
+ awful.spawn.with_shell("fcitx5 -d")
+end
+
+return autostart
diff --git a/rsc/config/awesome/colors.lua b/rsc/config/awesome/colors.lua
new file mode 100644
index 0000000..7a2a784
--- /dev/null
+++ b/rsc/config/awesome/colors.lua
@@ -0,0 +1,29 @@
+local catpuccin = {}
+
+catpuccin.rosewater = "#f5e0dc"
+catpuccin.flamingo = "#f2cdcd"
+catpuccin.pink = "#f5c2e7"
+catpuccin.mauve = "#cba6f7"
+catpuccin.red = "#f38ba8"
+catpuccin.maroon = "#eba0ac"
+catpuccin.peach = "#fab387"
+catpuccin.yellow = "#f9e2af"
+catpuccin.green = "#a6e3a1"
+catpuccin.teal = "#94e2d5"
+catpuccin.sky = "#89dceb"
+catpuccin.sapphire = "#74c7ec"
+catpuccin.blue = "#89b4fa"
+catpuccin.lavender = "#b4befe"
+catpuccin.text = "#cdd6f4"
+catpuccin.subtext1 = "#bac2de"
+catpuccin.subtext0 = "#a6adc8"
+catpuccin.overlay2 = "#9399b2"
+catpuccin.overlay1 = "#7f849c"
+catpuccin.overlay0 = "#6c7086"
+catpuccin.surface2 = "#585b70"
+catpuccin.surface1 = "#45475a"
+catpuccin.surface0 = "#313244"
+catpuccin.base = "#1e1e2e"
+catpuccin.mantle = "#181825"
+catpuccin.crust = "#11111b"
+return catpuccin
diff --git a/rsc/config/awesome/icons/ac.png b/rsc/config/awesome/icons/ac.png
new file mode 100755
index 0000000..96efcb4
Binary files /dev/null and b/rsc/config/awesome/icons/ac.png differ
diff --git a/rsc/config/awesome/icons/battery.png b/rsc/config/awesome/icons/battery.png
new file mode 100755
index 0000000..8c85596
Binary files /dev/null and b/rsc/config/awesome/icons/battery.png differ
diff --git a/rsc/config/awesome/icons/battery_empty.png b/rsc/config/awesome/icons/battery_empty.png
new file mode 100755
index 0000000..00821c9
Binary files /dev/null and b/rsc/config/awesome/icons/battery_empty.png differ
diff --git a/rsc/config/awesome/icons/battery_low.png b/rsc/config/awesome/icons/battery_low.png
new file mode 100755
index 0000000..29f3fc5
Binary files /dev/null and b/rsc/config/awesome/icons/battery_low.png differ
diff --git a/rsc/config/awesome/icons/centerfair.png b/rsc/config/awesome/icons/centerfair.png
new file mode 100755
index 0000000..c4f64b0
Binary files /dev/null and b/rsc/config/awesome/icons/centerfair.png differ
diff --git a/rsc/config/awesome/icons/centerwork.png b/rsc/config/awesome/icons/centerwork.png
new file mode 100755
index 0000000..9071516
Binary files /dev/null and b/rsc/config/awesome/icons/centerwork.png differ
diff --git a/rsc/config/awesome/icons/centerworkh.png b/rsc/config/awesome/icons/centerworkh.png
new file mode 100755
index 0000000..b1d1379
Binary files /dev/null and b/rsc/config/awesome/icons/centerworkh.png differ
diff --git a/rsc/config/awesome/icons/cpu.png b/rsc/config/awesome/icons/cpu.png
new file mode 100755
index 0000000..35af1c3
Binary files /dev/null and b/rsc/config/awesome/icons/cpu.png differ
diff --git a/rsc/config/awesome/icons/custom/bat.png b/rsc/config/awesome/icons/custom/bat.png
new file mode 100644
index 0000000..15f8aad
Binary files /dev/null and b/rsc/config/awesome/icons/custom/bat.png differ
diff --git a/rsc/config/awesome/icons/custom/calendar.png b/rsc/config/awesome/icons/custom/calendar.png
new file mode 100644
index 0000000..c425305
Binary files /dev/null and b/rsc/config/awesome/icons/custom/calendar.png differ
diff --git a/rsc/config/awesome/icons/custom/chip.png b/rsc/config/awesome/icons/custom/chip.png
new file mode 100644
index 0000000..a09bd1a
Binary files /dev/null and b/rsc/config/awesome/icons/custom/chip.png differ
diff --git a/rsc/config/awesome/icons/custom/clock.png b/rsc/config/awesome/icons/custom/clock.png
new file mode 100644
index 0000000..758a1b0
Binary files /dev/null and b/rsc/config/awesome/icons/custom/clock.png differ
diff --git a/rsc/config/awesome/icons/custom/cpu.png b/rsc/config/awesome/icons/custom/cpu.png
new file mode 100644
index 0000000..f8b072d
Binary files /dev/null and b/rsc/config/awesome/icons/custom/cpu.png differ
diff --git a/rsc/config/awesome/icons/custom/cpu.svg b/rsc/config/awesome/icons/custom/cpu.svg
new file mode 100644
index 0000000..3bbb824
--- /dev/null
+++ b/rsc/config/awesome/icons/custom/cpu.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/rsc/config/awesome/icons/custom/mem.png b/rsc/config/awesome/icons/custom/mem.png
new file mode 100644
index 0000000..1500bfb
Binary files /dev/null and b/rsc/config/awesome/icons/custom/mem.png differ
diff --git a/rsc/config/awesome/icons/custom/mem.svg b/rsc/config/awesome/icons/custom/mem.svg
new file mode 100644
index 0000000..96d5bb6
--- /dev/null
+++ b/rsc/config/awesome/icons/custom/mem.svg
@@ -0,0 +1 @@
+
\ No newline at end of file
diff --git a/rsc/config/awesome/icons/dwindle.png b/rsc/config/awesome/icons/dwindle.png
new file mode 100755
index 0000000..649ea99
Binary files /dev/null and b/rsc/config/awesome/icons/dwindle.png differ
diff --git a/rsc/config/awesome/icons/fairh.png b/rsc/config/awesome/icons/fairh.png
new file mode 100755
index 0000000..62d3d99
Binary files /dev/null and b/rsc/config/awesome/icons/fairh.png differ
diff --git a/rsc/config/awesome/icons/fairv.png b/rsc/config/awesome/icons/fairv.png
new file mode 100755
index 0000000..131dd0f
Binary files /dev/null and b/rsc/config/awesome/icons/fairv.png differ
diff --git a/rsc/config/awesome/icons/floating.png b/rsc/config/awesome/icons/floating.png
new file mode 100755
index 0000000..d25c47b
Binary files /dev/null and b/rsc/config/awesome/icons/floating.png differ
diff --git a/rsc/config/awesome/icons/hdd.png b/rsc/config/awesome/icons/hdd.png
new file mode 100755
index 0000000..0fb6833
Binary files /dev/null and b/rsc/config/awesome/icons/hdd.png differ
diff --git a/rsc/config/awesome/icons/magnifier.png b/rsc/config/awesome/icons/magnifier.png
new file mode 100755
index 0000000..60d3e0d
Binary files /dev/null and b/rsc/config/awesome/icons/magnifier.png differ
diff --git a/rsc/config/awesome/icons/mail.png b/rsc/config/awesome/icons/mail.png
new file mode 100755
index 0000000..474e602
Binary files /dev/null and b/rsc/config/awesome/icons/mail.png differ
diff --git a/rsc/config/awesome/icons/mail_on.png b/rsc/config/awesome/icons/mail_on.png
new file mode 100755
index 0000000..19106d7
Binary files /dev/null and b/rsc/config/awesome/icons/mail_on.png differ
diff --git a/rsc/config/awesome/icons/max.png b/rsc/config/awesome/icons/max.png
new file mode 100755
index 0000000..5b0a5b7
Binary files /dev/null and b/rsc/config/awesome/icons/max.png differ
diff --git a/rsc/config/awesome/icons/mem.png b/rsc/config/awesome/icons/mem.png
new file mode 100755
index 0000000..7860a1a
Binary files /dev/null and b/rsc/config/awesome/icons/mem.png differ
diff --git a/rsc/config/awesome/icons/net.png b/rsc/config/awesome/icons/net.png
new file mode 100755
index 0000000..bc42fdc
Binary files /dev/null and b/rsc/config/awesome/icons/net.png differ
diff --git a/rsc/config/awesome/icons/net_wired.png b/rsc/config/awesome/icons/net_wired.png
new file mode 100755
index 0000000..e8cc2bd
Binary files /dev/null and b/rsc/config/awesome/icons/net_wired.png differ
diff --git a/rsc/config/awesome/icons/note.png b/rsc/config/awesome/icons/note.png
new file mode 100755
index 0000000..baa29c3
Binary files /dev/null and b/rsc/config/awesome/icons/note.png differ
diff --git a/rsc/config/awesome/icons/note_on.png b/rsc/config/awesome/icons/note_on.png
new file mode 100755
index 0000000..1a7ab94
Binary files /dev/null and b/rsc/config/awesome/icons/note_on.png differ
diff --git a/rsc/config/awesome/icons/scissors.png b/rsc/config/awesome/icons/scissors.png
new file mode 100755
index 0000000..f8c700d
Binary files /dev/null and b/rsc/config/awesome/icons/scissors.png differ
diff --git a/rsc/config/awesome/icons/spiral.png b/rsc/config/awesome/icons/spiral.png
new file mode 100755
index 0000000..d9ee0f6
Binary files /dev/null and b/rsc/config/awesome/icons/spiral.png differ
diff --git a/rsc/config/awesome/icons/square_sel.png b/rsc/config/awesome/icons/square_sel.png
new file mode 100755
index 0000000..1102a9f
Binary files /dev/null and b/rsc/config/awesome/icons/square_sel.png differ
diff --git a/rsc/config/awesome/icons/square_unsel.png b/rsc/config/awesome/icons/square_unsel.png
new file mode 100755
index 0000000..7386b85
Binary files /dev/null and b/rsc/config/awesome/icons/square_unsel.png differ
diff --git a/rsc/config/awesome/icons/submenu.png b/rsc/config/awesome/icons/submenu.png
new file mode 100755
index 0000000..b55ebce
Binary files /dev/null and b/rsc/config/awesome/icons/submenu.png differ
diff --git a/rsc/config/awesome/icons/task.png b/rsc/config/awesome/icons/task.png
new file mode 100755
index 0000000..9701b68
Binary files /dev/null and b/rsc/config/awesome/icons/task.png differ
diff --git a/rsc/config/awesome/icons/temp.png b/rsc/config/awesome/icons/temp.png
new file mode 100755
index 0000000..6793a9f
Binary files /dev/null and b/rsc/config/awesome/icons/temp.png differ
diff --git a/rsc/config/awesome/icons/termfair.png b/rsc/config/awesome/icons/termfair.png
new file mode 100755
index 0000000..3e06023
Binary files /dev/null and b/rsc/config/awesome/icons/termfair.png differ
diff --git a/rsc/config/awesome/icons/tile.png b/rsc/config/awesome/icons/tile.png
new file mode 100755
index 0000000..922c05c
Binary files /dev/null and b/rsc/config/awesome/icons/tile.png differ
diff --git a/rsc/config/awesome/icons/tilebottom.png b/rsc/config/awesome/icons/tilebottom.png
new file mode 100755
index 0000000..6ec9cb8
Binary files /dev/null and b/rsc/config/awesome/icons/tilebottom.png differ
diff --git a/rsc/config/awesome/icons/tileleft.png b/rsc/config/awesome/icons/tileleft.png
new file mode 100755
index 0000000..8e3d2d6
Binary files /dev/null and b/rsc/config/awesome/icons/tileleft.png differ
diff --git a/rsc/config/awesome/icons/tiletop.png b/rsc/config/awesome/icons/tiletop.png
new file mode 100755
index 0000000..3da75a1
Binary files /dev/null and b/rsc/config/awesome/icons/tiletop.png differ
diff --git a/rsc/config/awesome/icons/titlebar/close_focus.png b/rsc/config/awesome/icons/titlebar/close_focus.png
new file mode 100755
index 0000000..b2051b0
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/close_focus.png differ
diff --git a/rsc/config/awesome/icons/titlebar/close_normal.png b/rsc/config/awesome/icons/titlebar/close_normal.png
new file mode 100755
index 0000000..da6028c
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/close_normal.png differ
diff --git a/rsc/config/awesome/icons/titlebar/floating_focus_active.png b/rsc/config/awesome/icons/titlebar/floating_focus_active.png
new file mode 100755
index 0000000..5fe84c0
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/floating_focus_active.png differ
diff --git a/rsc/config/awesome/icons/titlebar/floating_focus_inactive.png b/rsc/config/awesome/icons/titlebar/floating_focus_inactive.png
new file mode 100755
index 0000000..47f19f6
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/floating_focus_inactive.png differ
diff --git a/rsc/config/awesome/icons/titlebar/floating_normal_active.png b/rsc/config/awesome/icons/titlebar/floating_normal_active.png
new file mode 100755
index 0000000..576fa36
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/floating_normal_active.png differ
diff --git a/rsc/config/awesome/icons/titlebar/floating_normal_inactive.png b/rsc/config/awesome/icons/titlebar/floating_normal_inactive.png
new file mode 100755
index 0000000..4adc5e9
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/floating_normal_inactive.png differ
diff --git a/rsc/config/awesome/icons/titlebar/maximized_focus_active.png b/rsc/config/awesome/icons/titlebar/maximized_focus_active.png
new file mode 100755
index 0000000..7d9a11a
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/maximized_focus_active.png differ
diff --git a/rsc/config/awesome/icons/titlebar/maximized_focus_inactive.png b/rsc/config/awesome/icons/titlebar/maximized_focus_inactive.png
new file mode 100755
index 0000000..bce1d00
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/maximized_focus_inactive.png differ
diff --git a/rsc/config/awesome/icons/titlebar/maximized_normal_active.png b/rsc/config/awesome/icons/titlebar/maximized_normal_active.png
new file mode 100755
index 0000000..9f24945
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/maximized_normal_active.png differ
diff --git a/rsc/config/awesome/icons/titlebar/maximized_normal_inactive.png b/rsc/config/awesome/icons/titlebar/maximized_normal_inactive.png
new file mode 100755
index 0000000..2e56d32
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/maximized_normal_inactive.png differ
diff --git a/rsc/config/awesome/icons/titlebar/ontop_focus_active.png b/rsc/config/awesome/icons/titlebar/ontop_focus_active.png
new file mode 100755
index 0000000..41a69e2
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/ontop_focus_active.png differ
diff --git a/rsc/config/awesome/icons/titlebar/ontop_focus_inactive.png b/rsc/config/awesome/icons/titlebar/ontop_focus_inactive.png
new file mode 100755
index 0000000..2f3a2be
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/ontop_focus_inactive.png differ
diff --git a/rsc/config/awesome/icons/titlebar/ontop_normal_active.png b/rsc/config/awesome/icons/titlebar/ontop_normal_active.png
new file mode 100755
index 0000000..0f937b7
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/ontop_normal_active.png differ
diff --git a/rsc/config/awesome/icons/titlebar/ontop_normal_inactive.png b/rsc/config/awesome/icons/titlebar/ontop_normal_inactive.png
new file mode 100755
index 0000000..a9a3206
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/ontop_normal_inactive.png differ
diff --git a/rsc/config/awesome/icons/titlebar/sticky_focus_active.png b/rsc/config/awesome/icons/titlebar/sticky_focus_active.png
new file mode 100755
index 0000000..a9bc8a2
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/sticky_focus_active.png differ
diff --git a/rsc/config/awesome/icons/titlebar/sticky_focus_inactive.png b/rsc/config/awesome/icons/titlebar/sticky_focus_inactive.png
new file mode 100755
index 0000000..5493d8e
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/sticky_focus_inactive.png differ
diff --git a/rsc/config/awesome/icons/titlebar/sticky_normal_active.png b/rsc/config/awesome/icons/titlebar/sticky_normal_active.png
new file mode 100755
index 0000000..1e150f5
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/sticky_normal_active.png differ
diff --git a/rsc/config/awesome/icons/titlebar/sticky_normal_inactive.png b/rsc/config/awesome/icons/titlebar/sticky_normal_inactive.png
new file mode 100755
index 0000000..7e6c99b
Binary files /dev/null and b/rsc/config/awesome/icons/titlebar/sticky_normal_inactive.png differ
diff --git a/rsc/config/awesome/icons/vol.png b/rsc/config/awesome/icons/vol.png
new file mode 100755
index 0000000..bbf33d4
Binary files /dev/null and b/rsc/config/awesome/icons/vol.png differ
diff --git a/rsc/config/awesome/icons/vol_low.png b/rsc/config/awesome/icons/vol_low.png
new file mode 100755
index 0000000..aa3ce4d
Binary files /dev/null and b/rsc/config/awesome/icons/vol_low.png differ
diff --git a/rsc/config/awesome/icons/vol_mute.png b/rsc/config/awesome/icons/vol_mute.png
new file mode 100755
index 0000000..e855fd2
Binary files /dev/null and b/rsc/config/awesome/icons/vol_mute.png differ
diff --git a/rsc/config/awesome/icons/vol_no.png b/rsc/config/awesome/icons/vol_no.png
new file mode 100755
index 0000000..bbe917b
Binary files /dev/null and b/rsc/config/awesome/icons/vol_no.png differ
diff --git a/rsc/config/awesome/lain/.github/ISSUE_TEMPLATE/bug_report.md b/rsc/config/awesome/lain/.github/ISSUE_TEMPLATE/bug_report.md
new file mode 100644
index 0000000..d4297ce
--- /dev/null
+++ b/rsc/config/awesome/lain/.github/ISSUE_TEMPLATE/bug_report.md
@@ -0,0 +1,33 @@
+---
+name: Bugs
+about: Bugs and unintended behavior
+
+---
+
+**Please read the [wiki](https://github.com/lcpz/lain/wiki) and browse the [Issues section](https://github.com/lcpz/lain/issues) first.**
+
+If you can't find a solution, then please provide:
+
+* output of `awesome -v` and `lua -v`
+* expected behavior and actual behavior
+* steps to reproduce the problem
+* X error log
+
+### How to provide X error log
+
+There are two ways:
+
+* (Physically) Restart X like this:
+ ```shell
+ startx -- -keeptty -nolisten tcp > $HOME/.xorg.log 2>&1
+ ```
+ the error log will be output into `$HOME/.xorg.log`.
+
+* (Virtually) Use [Xephyr](https://wikipedia.org/wiki/Xephyr):
+ ```shell
+ # set screen size as you like
+ Xephyr :1 -screen 1280x800 2> stdout.txt & DISPLAY=:1 awesome
+ ```
+ the error log will be output in the file `stdout.txt`.
+
+Before reporting, read the log and see if you can solve it yourself.
diff --git a/rsc/config/awesome/lain/.github/workflows/main.yml b/rsc/config/awesome/lain/.github/workflows/main.yml
new file mode 100644
index 0000000..3193e0b
--- /dev/null
+++ b/rsc/config/awesome/lain/.github/workflows/main.yml
@@ -0,0 +1,36 @@
+name: Lain
+
+on: [push]
+
+permissions:
+ contents: read
+
+jobs:
+ linting:
+ runs-on: ubuntu-latest
+ steps:
+ - name: Checkout
+ uses: actions/checkout@v3
+
+ - name: Luacheck linter
+ uses: lunarmodules/luacheck@v0
+
+ rockspec:
+ runs-on: ubuntu-latest
+ steps:
+ - uses: actions/checkout@v3
+ - uses: leafo/gh-actions-lua@v9
+ - uses: leafo/gh-actions-luarocks@v4
+
+ - name: Update Rockspec
+ shell: bash
+ env:
+ LUAROCKS_API_KEY: ${{ secrets.LUAROCKS_API_KEY }}
+ run: |
+ if [[ $(git diff *.rockspec | grep "+version") ]]
+ then
+ echo "Rockspec changed, uploading to LuaRocks.org"
+ luarocks upload *.rockspec --api-key "$LUAROCKS_API_KEY"
+ else
+ echo "Rockspec unchanged, nothing to do"
+ fi
diff --git a/rsc/config/awesome/lain/.gitmodules b/rsc/config/awesome/lain/.gitmodules
new file mode 100644
index 0000000..d95bcb5
--- /dev/null
+++ b/rsc/config/awesome/lain/.gitmodules
@@ -0,0 +1,3 @@
+[submodule "lain.wiki"]
+ path = wiki
+ url = https://github.com/lcpz/lain.wiki.git
diff --git a/rsc/config/awesome/lain/.luacheckrc b/rsc/config/awesome/lain/.luacheckrc
new file mode 100644
index 0000000..9944223
--- /dev/null
+++ b/rsc/config/awesome/lain/.luacheckrc
@@ -0,0 +1,27 @@
+-- Only allow symbols available in all Lua versions
+std = "min"
+
+allow_defined = true
+max_line_length = false
+cache = true
+
+-- Global objects defined by the C code
+read_globals = {
+ "awesome",
+ "mousegrabber",
+ "table.unpack",
+ "unpack",
+ "utf8"
+}
+
+globals = {
+ "client",
+ "mouse",
+ "root",
+ "screen"
+}
+
+-- https://luacheck.readthedocs.io/en/stable/warnings.html
+ignore = {
+ "131"
+}
diff --git a/rsc/config/awesome/lain/LICENSE b/rsc/config/awesome/lain/LICENSE
new file mode 100644
index 0000000..23cb790
--- /dev/null
+++ b/rsc/config/awesome/lain/LICENSE
@@ -0,0 +1,339 @@
+ GNU GENERAL PUBLIC LICENSE
+ Version 2, June 1991
+
+ Copyright (C) 1989, 1991 Free Software Foundation, Inc.,
+ 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+ Preamble
+
+ The licenses for most software are designed to take away your
+freedom to share and change it. By contrast, the GNU General Public
+License is intended to guarantee your freedom to share and change free
+software--to make sure the software is free for all its users. This
+General Public License applies to most of the Free Software
+Foundation's software and to any other program whose authors commit to
+using it. (Some other Free Software Foundation software is covered by
+the GNU Lesser General Public License instead.) You can apply it to
+your programs, too.
+
+ When we speak of free software, we are referring to freedom, not
+price. Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+this service if you wish), that you receive source code or can get it
+if you want it, that you can change the software or use pieces of it
+in new free programs; and that you know you can do these things.
+
+ To protect your rights, we need to make restrictions that forbid
+anyone to deny you these rights or to ask you to surrender the rights.
+These restrictions translate to certain responsibilities for you if you
+distribute copies of the software, or if you modify it.
+
+ For example, if you distribute copies of such a program, whether
+gratis or for a fee, you must give the recipients all the rights that
+you have. You must make sure that they, too, receive or can get the
+source code. And you must show them these terms so they know their
+rights.
+
+ We protect your rights with two steps: (1) copyright the software, and
+(2) offer you this license which gives you legal permission to copy,
+distribute and/or modify the software.
+
+ Also, for each author's protection and ours, we want to make certain
+that everyone understands that there is no warranty for this free
+software. If the software is modified by someone else and passed on, we
+want its recipients to know that what they have is not the original, so
+that any problems introduced by others will not reflect on the original
+authors' reputations.
+
+ Finally, any free program is threatened constantly by software
+patents. We wish to avoid the danger that redistributors of a free
+program will individually obtain patent licenses, in effect making the
+program proprietary. To prevent this, we have made it clear that any
+patent must be licensed for everyone's free use or not licensed at all.
+
+ The precise terms and conditions for copying, distribution and
+modification follow.
+
+ GNU GENERAL PUBLIC LICENSE
+ TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
+
+ 0. This License applies to any program or other work which contains
+a notice placed by the copyright holder saying it may be distributed
+under the terms of this General Public License. The "Program", below,
+refers to any such program or work, and a "work based on the Program"
+means either the Program or any derivative work under copyright law:
+that is to say, a work containing the Program or a portion of it,
+either verbatim or with modifications and/or translated into another
+language. (Hereinafter, translation is included without limitation in
+the term "modification".) Each licensee is addressed as "you".
+
+Activities other than copying, distribution and modification are not
+covered by this License; they are outside its scope. The act of
+running the Program is not restricted, and the output from the Program
+is covered only if its contents constitute a work based on the
+Program (independent of having been made by running the Program).
+Whether that is true depends on what the Program does.
+
+ 1. You may copy and distribute verbatim copies of the Program's
+source code as you receive it, in any medium, provided that you
+conspicuously and appropriately publish on each copy an appropriate
+copyright notice and disclaimer of warranty; keep intact all the
+notices that refer to this License and to the absence of any warranty;
+and give any other recipients of the Program a copy of this License
+along with the Program.
+
+You may charge a fee for the physical act of transferring a copy, and
+you may at your option offer warranty protection in exchange for a fee.
+
+ 2. You may modify your copy or copies of the Program or any portion
+of it, thus forming a work based on the Program, and copy and
+distribute such modifications or work under the terms of Section 1
+above, provided that you also meet all of these conditions:
+
+ a) You must cause the modified files to carry prominent notices
+ stating that you changed the files and the date of any change.
+
+ b) You must cause any work that you distribute or publish, that in
+ whole or in part contains or is derived from the Program or any
+ part thereof, to be licensed as a whole at no charge to all third
+ parties under the terms of this License.
+
+ c) If the modified program normally reads commands interactively
+ when run, you must cause it, when started running for such
+ interactive use in the most ordinary way, to print or display an
+ announcement including an appropriate copyright notice and a
+ notice that there is no warranty (or else, saying that you provide
+ a warranty) and that users may redistribute the program under
+ these conditions, and telling the user how to view a copy of this
+ License. (Exception: if the Program itself is interactive but
+ does not normally print such an announcement, your work based on
+ the Program is not required to print an announcement.)
+
+These requirements apply to the modified work as a whole. If
+identifiable sections of that work are not derived from the Program,
+and can be reasonably considered independent and separate works in
+themselves, then this License, and its terms, do not apply to those
+sections when you distribute them as separate works. But when you
+distribute the same sections as part of a whole which is a work based
+on the Program, the distribution of the whole must be on the terms of
+this License, whose permissions for other licensees extend to the
+entire whole, and thus to each and every part regardless of who wrote it.
+
+Thus, it is not the intent of this section to claim rights or contest
+your rights to work written entirely by you; rather, the intent is to
+exercise the right to control the distribution of derivative or
+collective works based on the Program.
+
+In addition, mere aggregation of another work not based on the Program
+with the Program (or with a work based on the Program) on a volume of
+a storage or distribution medium does not bring the other work under
+the scope of this License.
+
+ 3. You may copy and distribute the Program (or a work based on it,
+under Section 2) in object code or executable form under the terms of
+Sections 1 and 2 above provided that you also do one of the following:
+
+ a) Accompany it with the complete corresponding machine-readable
+ source code, which must be distributed under the terms of Sections
+ 1 and 2 above on a medium customarily used for software interchange; or,
+
+ b) Accompany it with a written offer, valid for at least three
+ years, to give any third party, for a charge no more than your
+ cost of physically performing source distribution, a complete
+ machine-readable copy of the corresponding source code, to be
+ distributed under the terms of Sections 1 and 2 above on a medium
+ customarily used for software interchange; or,
+
+ c) Accompany it with the information you received as to the offer
+ to distribute corresponding source code. (This alternative is
+ allowed only for noncommercial distribution and only if you
+ received the program in object code or executable form with such
+ an offer, in accord with Subsection b above.)
+
+The source code for a work means the preferred form of the work for
+making modifications to it. For an executable work, complete source
+code means all the source code for all modules it contains, plus any
+associated interface definition files, plus the scripts used to
+control compilation and installation of the executable. However, as a
+special exception, the source code distributed need not include
+anything that is normally distributed (in either source or binary
+form) with the major components (compiler, kernel, and so on) of the
+operating system on which the executable runs, unless that component
+itself accompanies the executable.
+
+If distribution of executable or object code is made by offering
+access to copy from a designated place, then offering equivalent
+access to copy the source code from the same place counts as
+distribution of the source code, even though third parties are not
+compelled to copy the source along with the object code.
+
+ 4. You may not copy, modify, sublicense, or distribute the Program
+except as expressly provided under this License. Any attempt
+otherwise to copy, modify, sublicense or distribute the Program is
+void, and will automatically terminate your rights under this License.
+However, parties who have received copies, or rights, from you under
+this License will not have their licenses terminated so long as such
+parties remain in full compliance.
+
+ 5. You are not required to accept this License, since you have not
+signed it. However, nothing else grants you permission to modify or
+distribute the Program or its derivative works. These actions are
+prohibited by law if you do not accept this License. Therefore, by
+modifying or distributing the Program (or any work based on the
+Program), you indicate your acceptance of this License to do so, and
+all its terms and conditions for copying, distributing or modifying
+the Program or works based on it.
+
+ 6. Each time you redistribute the Program (or any work based on the
+Program), the recipient automatically receives a license from the
+original licensor to copy, distribute or modify the Program subject to
+these terms and conditions. You may not impose any further
+restrictions on the recipients' exercise of the rights granted herein.
+You are not responsible for enforcing compliance by third parties to
+this License.
+
+ 7. If, as a consequence of a court judgment or allegation of patent
+infringement or for any other reason (not limited to patent issues),
+conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License. If you cannot
+distribute so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you
+may not distribute the Program at all. For example, if a patent
+license would not permit royalty-free redistribution of the Program by
+all those who receive copies directly or indirectly through you, then
+the only way you could satisfy both it and this License would be to
+refrain entirely from distribution of the Program.
+
+If any portion of this section is held invalid or unenforceable under
+any particular circumstance, the balance of the section is intended to
+apply and the section as a whole is intended to apply in other
+circumstances.
+
+It is not the purpose of this section to induce you to infringe any
+patents or other property right claims or to contest validity of any
+such claims; this section has the sole purpose of protecting the
+integrity of the free software distribution system, which is
+implemented by public license practices. Many people have made
+generous contributions to the wide range of software distributed
+through that system in reliance on consistent application of that
+system; it is up to the author/donor to decide if he or she is willing
+to distribute software through any other system and a licensee cannot
+impose that choice.
+
+This section is intended to make thoroughly clear what is believed to
+be a consequence of the rest of this License.
+
+ 8. If the distribution and/or use of the Program is restricted in
+certain countries either by patents or by copyrighted interfaces, the
+original copyright holder who places the Program under this License
+may add an explicit geographical distribution limitation excluding
+those countries, so that distribution is permitted only in or among
+countries not thus excluded. In such case, this License incorporates
+the limitation as if written in the body of this License.
+
+ 9. The Free Software Foundation may publish revised and/or new versions
+of the General Public License from time to time. Such new versions will
+be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+Each version is given a distinguishing version number. If the Program
+specifies a version number of this License which applies to it and "any
+later version", you have the option of following the terms and conditions
+either of that version or of any later version published by the Free
+Software Foundation. If the Program does not specify a version number of
+this License, you may choose any version ever published by the Free Software
+Foundation.
+
+ 10. If you wish to incorporate parts of the Program into other free
+programs whose distribution conditions are different, write to the author
+to ask for permission. For software which is copyrighted by the Free
+Software Foundation, write to the Free Software Foundation; we sometimes
+make exceptions for this. Our decision will be guided by the two goals
+of preserving the free status of all derivatives of our free software and
+of promoting the sharing and reuse of software generally.
+
+ NO WARRANTY
+
+ 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
+FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
+OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
+PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
+OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
+MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
+TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
+PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
+REPAIR OR CORRECTION.
+
+ 12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
+REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
+INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
+OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
+TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
+YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
+PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
+POSSIBILITY OF SUCH DAMAGES.
+
+ END OF TERMS AND CONDITIONS
+
+ How to Apply These Terms to Your New Programs
+
+ If you develop a new program, and you want it to be of the greatest
+possible use to the public, the best way to achieve this is to make it
+free software which everyone can redistribute and change under these terms.
+
+ To do so, attach the following notices to the program. It is safest
+to attach them to the start of each source file to most effectively
+convey the exclusion of warranty; and each file should have at least
+the "copyright" line and a pointer to where the full notice is found.
+
+ {description}
+ Copyright (C) {year} {fullname}
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License along
+ with this program; if not, write to the Free Software Foundation, Inc.,
+ 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+
+Also add information on how to contact you by electronic and paper mail.
+
+If the program is interactive, make it output a short notice like this
+when it starts in an interactive mode:
+
+ Gnomovision version 69, Copyright (C) year name of author
+ Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'.
+ This is free software, and you are welcome to redistribute it
+ under certain conditions; type `show c' for details.
+
+The hypothetical commands `show w' and `show c' should show the appropriate
+parts of the General Public License. Of course, the commands you use may
+be called something other than `show w' and `show c'; they could even be
+mouse-clicks or menu items--whatever suits your program.
+
+You should also get your employer (if you work as a programmer) or your
+school, if any, to sign a "copyright disclaimer" for the program, if
+necessary. Here is a sample; alter the names:
+
+ Yoyodyne, Inc., hereby disclaims all copyright interest in the program
+ `Gnomovision' (which makes passes at compilers) written by James Hacker.
+
+ {signature of Ty Coon}, 1 April 1989
+ Ty Coon, President of Vice
+
+This General Public License does not permit incorporating your program into
+proprietary programs. If your program is a subroutine library, you may
+consider it more useful to permit linking proprietary applications with the
+library. If this is what you want to do, use the GNU Lesser General
+Public License instead of this License.
diff --git a/rsc/config/awesome/lain/README.rst b/rsc/config/awesome/lain/README.rst
new file mode 100644
index 0000000..616c7cd
--- /dev/null
+++ b/rsc/config/awesome/lain/README.rst
@@ -0,0 +1,40 @@
+Lain
+====
+
+.. image:: https://github.com/lcpz/lain/actions/workflows/main.yml/badge.svg
+
+-------------------------------------------------
+Layouts, widgets and utilities for Awesome WM 4.x
+-------------------------------------------------
+
+:Author: Luca CPZ
+:Version: git
+:License: GNU-GPL2_
+:Source: https://github.com/lcpz/lain
+
+Description
+-----------
+
+Successor of awesome-vain_, this module provides alternative layouts, asynchronous widgets and utility functions for Awesome_.
+
+Contributions
+-------------
+
+Constructive criticism and suggestions are welcome.
+
+If you want to create a pull request, make sure that:
+
+- Your code fits with the general style of the module. In particular, you should use the same indentation pattern that the code uses, and also avoid adding space at the ends of lines.
+
+- Your code its easy to understand, maintainable, and modularized. You should also avoid code duplication wherever possible by adding functions to or using lain.helpers_. If something is unclear, or you can not write it in such a way that it will be clear, explain it with a comment.
+
+- You test your changes before submitting to make sure that your code works and does not break other parts of the module.
+
+- You update ``wiki`` submodule with a thorough section, if necessary.
+
+Contributed widgets have to be put in ``widget/contrib``.
+
+.. _GNU-GPL2: http://www.gnu.org/licenses/gpl-2.0.html
+.. _awesome-vain: https://github.com/vain/awesome-vain
+.. _Awesome: https://github.com/awesomeWM/awesome
+.. _lain.helpers: https://github.com/lcpz/lain/blob/master/helpers.lua
diff --git a/rsc/config/awesome/lain/helpers.lua b/rsc/config/awesome/lain/helpers.lua
new file mode 100644
index 0000000..ef9e08b
--- /dev/null
+++ b/rsc/config/awesome/lain/helpers.lua
@@ -0,0 +1,203 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2013, Luca CPZ
+
+--]]
+
+local spawn = require("awful.spawn")
+local timer = require("gears.timer")
+local debug = require("debug")
+local io = { lines = io.lines,
+ open = io.open }
+local pairs = pairs
+local rawget = rawget
+local tsort = table.sort
+local unpack = unpack or table.unpack -- lua 5.1 retro-compatibility
+
+-- Lain helper functions for internal use
+-- lain.helpers
+local helpers = {}
+
+helpers.lain_dir = debug.getinfo(1, 'S').source:match[[^@(.*/).*$]]
+helpers.icons_dir = helpers.lain_dir .. 'icons/'
+helpers.scripts_dir = helpers.lain_dir .. 'scripts/'
+
+-- {{{ Modules loader
+
+function helpers.wrequire(t, k)
+ return rawget(t, k) or require(t._NAME .. '.' .. k)
+end
+
+-- }}}
+
+-- {{{ File operations
+
+-- check if the file exists and is readable
+function helpers.file_exists(path)
+ local file = io.open(path, "rb")
+ if file then file:close() end
+ return file ~= nil
+end
+
+-- get a table with all lines from a file
+function helpers.lines_from(path)
+ local lines = {}
+ for line in io.lines(path) do
+ lines[#lines + 1] = line
+ end
+ return lines
+end
+
+-- get a table with all lines from a file matching regexp
+function helpers.lines_match(regexp, path)
+ local lines = {}
+ for line in io.lines(path) do
+ if string.match(line, regexp) then
+ lines[#lines + 1] = line
+ end
+ end
+ return lines
+end
+
+-- get first line of a file
+function helpers.first_line(path)
+ local file, first = io.open(path, "rb"), nil
+ if file then
+ first = file:read("*l")
+ file:close()
+ end
+ return first
+end
+
+-- get first non empty line from a file
+function helpers.first_nonempty_line(path)
+ for line in io.lines(path) do
+ if #line then return line end
+ end
+ return nil
+end
+
+-- }}}
+
+-- {{{ Timer maker
+
+helpers.timer_table = {}
+
+function helpers.newtimer(name, timeout, fun, nostart, stoppable)
+ if not name or #name == 0 then return end
+ name = (stoppable and name) or timeout
+ if not helpers.timer_table[name] then
+ helpers.timer_table[name] = timer({ timeout = timeout })
+ helpers.timer_table[name]:start()
+ end
+ helpers.timer_table[name]:connect_signal("timeout", fun)
+ if not nostart then
+ helpers.timer_table[name]:emit_signal("timeout")
+ end
+ return stoppable and helpers.timer_table[name]
+end
+
+-- }}}
+
+-- {{{ Pipe operations
+
+-- run a command and execute a function on its output (asynchronous pipe)
+-- @param cmd the input command
+-- @param callback function to execute on cmd output
+-- @return cmd PID
+function helpers.async(cmd, callback)
+ return spawn.easy_async(cmd,
+ function (stdout, _, _, exit_code)
+ callback(stdout, exit_code)
+ end)
+end
+
+-- like above, but call spawn.easy_async with a shell
+function helpers.async_with_shell(cmd, callback)
+ return spawn.easy_async_with_shell(cmd,
+ function (stdout, _, _, exit_code)
+ callback(stdout, exit_code)
+ end)
+end
+
+-- run a command and execute a function on its output line by line
+function helpers.line_callback(cmd, callback)
+ return spawn.with_line_callback(cmd, {
+ stdout = function (line)
+ callback(line)
+ end,
+ })
+end
+
+-- }}}
+
+-- {{{ A map utility
+
+helpers.map_table = {}
+
+function helpers.set_map(element, value)
+ helpers.map_table[element] = value
+end
+
+function helpers.get_map(element)
+ return helpers.map_table[element]
+end
+
+-- }}}
+
+-- {{{ Misc
+
+-- check if an element exist on a table
+function helpers.element_in_table(element, tbl)
+ for _, i in pairs(tbl) do
+ if i == element then
+ return true
+ end
+ end
+ return false
+end
+
+-- iterate over table of records sorted by keys
+function helpers.spairs(t)
+ -- collect the keys
+ local keys = {}
+ for k in pairs(t) do keys[#keys+1] = k end
+
+ tsort(keys)
+
+ -- return the iterator function
+ local i = 0
+ return function()
+ i = i + 1
+ if keys[i] then
+ return keys[i], t[keys[i]]
+ end
+ end
+end
+
+-- create the partition of singletons of a given set
+-- example: the trivial partition set of {a, b, c}, is {{a}, {b}, {c}}
+function helpers.trivial_partition_set(set)
+ local ss = {}
+ for _,e in pairs(set) do
+ ss[#ss+1] = {e}
+ end
+ return ss
+end
+
+-- create the powerset of a given set
+function helpers.powerset(s)
+ if not s then return {} end
+ local t = {{}}
+ for i = 1, #s do
+ for j = 1, #t do
+ t[#t+1] = {s[i],unpack(t[j])}
+ end
+ end
+ return t
+end
+
+-- }}}
+
+return helpers
diff --git a/rsc/config/awesome/lain/icons/cal/black/1.png b/rsc/config/awesome/lain/icons/cal/black/1.png
new file mode 100644
index 0000000..d2fb62e
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/1.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/10.png b/rsc/config/awesome/lain/icons/cal/black/10.png
new file mode 100644
index 0000000..507b079
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/10.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/11.png b/rsc/config/awesome/lain/icons/cal/black/11.png
new file mode 100644
index 0000000..336141b
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/11.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/12.png b/rsc/config/awesome/lain/icons/cal/black/12.png
new file mode 100644
index 0000000..c589729
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/12.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/13.png b/rsc/config/awesome/lain/icons/cal/black/13.png
new file mode 100644
index 0000000..377518b
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/13.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/14.png b/rsc/config/awesome/lain/icons/cal/black/14.png
new file mode 100644
index 0000000..6f4a9fe
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/14.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/15.png b/rsc/config/awesome/lain/icons/cal/black/15.png
new file mode 100644
index 0000000..1a271c1
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/15.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/16.png b/rsc/config/awesome/lain/icons/cal/black/16.png
new file mode 100644
index 0000000..5e65835
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/16.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/17.png b/rsc/config/awesome/lain/icons/cal/black/17.png
new file mode 100644
index 0000000..f3fa0a9
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/17.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/18.png b/rsc/config/awesome/lain/icons/cal/black/18.png
new file mode 100644
index 0000000..7acb37a
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/18.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/19.png b/rsc/config/awesome/lain/icons/cal/black/19.png
new file mode 100644
index 0000000..a557957
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/19.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/2.png b/rsc/config/awesome/lain/icons/cal/black/2.png
new file mode 100644
index 0000000..17b33e0
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/2.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/20.png b/rsc/config/awesome/lain/icons/cal/black/20.png
new file mode 100644
index 0000000..558d111
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/20.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/21.png b/rsc/config/awesome/lain/icons/cal/black/21.png
new file mode 100644
index 0000000..0bbedc8
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/21.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/22.png b/rsc/config/awesome/lain/icons/cal/black/22.png
new file mode 100644
index 0000000..762d262
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/22.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/23.png b/rsc/config/awesome/lain/icons/cal/black/23.png
new file mode 100644
index 0000000..a39dcee
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/23.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/24.png b/rsc/config/awesome/lain/icons/cal/black/24.png
new file mode 100644
index 0000000..c00dbca
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/24.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/25.png b/rsc/config/awesome/lain/icons/cal/black/25.png
new file mode 100644
index 0000000..dc9243c
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/25.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/26.png b/rsc/config/awesome/lain/icons/cal/black/26.png
new file mode 100644
index 0000000..50bb182
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/26.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/27.png b/rsc/config/awesome/lain/icons/cal/black/27.png
new file mode 100644
index 0000000..0fbf9fc
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/27.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/28.png b/rsc/config/awesome/lain/icons/cal/black/28.png
new file mode 100644
index 0000000..def6ab2
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/28.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/29.png b/rsc/config/awesome/lain/icons/cal/black/29.png
new file mode 100644
index 0000000..531923c
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/29.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/3.png b/rsc/config/awesome/lain/icons/cal/black/3.png
new file mode 100644
index 0000000..98b552d
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/3.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/30.png b/rsc/config/awesome/lain/icons/cal/black/30.png
new file mode 100644
index 0000000..ca58151
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/30.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/31.png b/rsc/config/awesome/lain/icons/cal/black/31.png
new file mode 100644
index 0000000..6e8da21
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/31.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/4.png b/rsc/config/awesome/lain/icons/cal/black/4.png
new file mode 100644
index 0000000..4335979
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/4.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/5.png b/rsc/config/awesome/lain/icons/cal/black/5.png
new file mode 100644
index 0000000..576ec11
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/5.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/6.png b/rsc/config/awesome/lain/icons/cal/black/6.png
new file mode 100644
index 0000000..56fa8ab
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/6.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/7.png b/rsc/config/awesome/lain/icons/cal/black/7.png
new file mode 100644
index 0000000..7c90b3a
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/7.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/8.png b/rsc/config/awesome/lain/icons/cal/black/8.png
new file mode 100644
index 0000000..9d1f28e
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/8.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/black/9.png b/rsc/config/awesome/lain/icons/cal/black/9.png
new file mode 100644
index 0000000..00d0933
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/black/9.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/1.png b/rsc/config/awesome/lain/icons/cal/white/1.png
new file mode 100644
index 0000000..a0faa20
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/1.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/10.png b/rsc/config/awesome/lain/icons/cal/white/10.png
new file mode 100644
index 0000000..7d9343b
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/10.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/11.png b/rsc/config/awesome/lain/icons/cal/white/11.png
new file mode 100644
index 0000000..7af5e99
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/11.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/12.png b/rsc/config/awesome/lain/icons/cal/white/12.png
new file mode 100644
index 0000000..b164f85
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/12.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/13.png b/rsc/config/awesome/lain/icons/cal/white/13.png
new file mode 100644
index 0000000..fef74f3
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/13.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/14.png b/rsc/config/awesome/lain/icons/cal/white/14.png
new file mode 100644
index 0000000..d747a6b
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/14.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/15.png b/rsc/config/awesome/lain/icons/cal/white/15.png
new file mode 100644
index 0000000..64418a6
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/15.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/16.png b/rsc/config/awesome/lain/icons/cal/white/16.png
new file mode 100644
index 0000000..8b86700
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/16.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/17.png b/rsc/config/awesome/lain/icons/cal/white/17.png
new file mode 100644
index 0000000..033b5ff
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/17.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/18.png b/rsc/config/awesome/lain/icons/cal/white/18.png
new file mode 100644
index 0000000..0cf1c24
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/18.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/19.png b/rsc/config/awesome/lain/icons/cal/white/19.png
new file mode 100644
index 0000000..bfd3530
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/19.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/2.png b/rsc/config/awesome/lain/icons/cal/white/2.png
new file mode 100644
index 0000000..e7f3fa4
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/2.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/20.png b/rsc/config/awesome/lain/icons/cal/white/20.png
new file mode 100644
index 0000000..9a5a1fb
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/20.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/21.png b/rsc/config/awesome/lain/icons/cal/white/21.png
new file mode 100644
index 0000000..266ab9f
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/21.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/22.png b/rsc/config/awesome/lain/icons/cal/white/22.png
new file mode 100644
index 0000000..f486289
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/22.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/23.png b/rsc/config/awesome/lain/icons/cal/white/23.png
new file mode 100644
index 0000000..244dceb
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/23.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/24.png b/rsc/config/awesome/lain/icons/cal/white/24.png
new file mode 100644
index 0000000..0ce1c75
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/24.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/25.png b/rsc/config/awesome/lain/icons/cal/white/25.png
new file mode 100644
index 0000000..48d279c
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/25.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/26.png b/rsc/config/awesome/lain/icons/cal/white/26.png
new file mode 100644
index 0000000..7535855
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/26.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/27.png b/rsc/config/awesome/lain/icons/cal/white/27.png
new file mode 100644
index 0000000..2aa9074
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/27.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/28.png b/rsc/config/awesome/lain/icons/cal/white/28.png
new file mode 100644
index 0000000..0201976
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/28.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/29.png b/rsc/config/awesome/lain/icons/cal/white/29.png
new file mode 100644
index 0000000..9305b9b
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/29.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/3.png b/rsc/config/awesome/lain/icons/cal/white/3.png
new file mode 100644
index 0000000..f1eb5de
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/3.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/30.png b/rsc/config/awesome/lain/icons/cal/white/30.png
new file mode 100644
index 0000000..1ba61aa
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/30.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/31.png b/rsc/config/awesome/lain/icons/cal/white/31.png
new file mode 100644
index 0000000..e9a873b
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/31.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/4.png b/rsc/config/awesome/lain/icons/cal/white/4.png
new file mode 100644
index 0000000..ee1ed6a
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/4.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/5.png b/rsc/config/awesome/lain/icons/cal/white/5.png
new file mode 100644
index 0000000..466aa71
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/5.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/6.png b/rsc/config/awesome/lain/icons/cal/white/6.png
new file mode 100644
index 0000000..0a7bf4d
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/6.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/7.png b/rsc/config/awesome/lain/icons/cal/white/7.png
new file mode 100644
index 0000000..e971951
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/7.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/8.png b/rsc/config/awesome/lain/icons/cal/white/8.png
new file mode 100644
index 0000000..cb03d0b
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/8.png differ
diff --git a/rsc/config/awesome/lain/icons/cal/white/9.png b/rsc/config/awesome/lain/icons/cal/white/9.png
new file mode 100644
index 0000000..fca554a
Binary files /dev/null and b/rsc/config/awesome/lain/icons/cal/white/9.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/default/cascade.png b/rsc/config/awesome/lain/icons/layout/default/cascade.png
new file mode 100644
index 0000000..292a057
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/default/cascade.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/default/cascadetile.png b/rsc/config/awesome/lain/icons/layout/default/cascadetile.png
new file mode 100644
index 0000000..ba30f43
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/default/cascadetile.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/default/cascadetilew.png b/rsc/config/awesome/lain/icons/layout/default/cascadetilew.png
new file mode 100644
index 0000000..d15eb70
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/default/cascadetilew.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/default/cascadew.png b/rsc/config/awesome/lain/icons/layout/default/cascadew.png
new file mode 100644
index 0000000..da64bd6
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/default/cascadew.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/default/centerfair.png b/rsc/config/awesome/lain/icons/layout/default/centerfair.png
new file mode 100644
index 0000000..188c243
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/default/centerfair.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/default/centerfairw.png b/rsc/config/awesome/lain/icons/layout/default/centerfairw.png
new file mode 100644
index 0000000..ed4bcf5
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/default/centerfairw.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/default/centerwork.png b/rsc/config/awesome/lain/icons/layout/default/centerwork.png
new file mode 100644
index 0000000..51e06bc
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/default/centerwork.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/default/centerworkh.png b/rsc/config/awesome/lain/icons/layout/default/centerworkh.png
new file mode 100644
index 0000000..c59092f
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/default/centerworkh.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/default/centerworkhw.png b/rsc/config/awesome/lain/icons/layout/default/centerworkhw.png
new file mode 100644
index 0000000..7820f8c
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/default/centerworkhw.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/default/centerworkw.png b/rsc/config/awesome/lain/icons/layout/default/centerworkw.png
new file mode 100644
index 0000000..85e6996
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/default/centerworkw.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/default/termfair.png b/rsc/config/awesome/lain/icons/layout/default/termfair.png
new file mode 100644
index 0000000..06226c1
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/default/termfair.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/default/termfairw.png b/rsc/config/awesome/lain/icons/layout/default/termfairw.png
new file mode 100644
index 0000000..0a8b576
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/default/termfairw.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/zenburn/cascade.png b/rsc/config/awesome/lain/icons/layout/zenburn/cascade.png
new file mode 100644
index 0000000..fbe4fac
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/zenburn/cascade.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/zenburn/cascadetile.png b/rsc/config/awesome/lain/icons/layout/zenburn/cascadetile.png
new file mode 100644
index 0000000..2e03a80
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/zenburn/cascadetile.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/zenburn/centerfair.png b/rsc/config/awesome/lain/icons/layout/zenburn/centerfair.png
new file mode 100644
index 0000000..75dc993
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/zenburn/centerfair.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/zenburn/centerwork.png b/rsc/config/awesome/lain/icons/layout/zenburn/centerwork.png
new file mode 100644
index 0000000..af7a863
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/zenburn/centerwork.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/zenburn/centerworkh.png b/rsc/config/awesome/lain/icons/layout/zenburn/centerworkh.png
new file mode 100644
index 0000000..88019b3
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/zenburn/centerworkh.png differ
diff --git a/rsc/config/awesome/lain/icons/layout/zenburn/termfair.png b/rsc/config/awesome/lain/icons/layout/zenburn/termfair.png
new file mode 100644
index 0000000..f7640b5
Binary files /dev/null and b/rsc/config/awesome/lain/icons/layout/zenburn/termfair.png differ
diff --git a/rsc/config/awesome/lain/icons/mail.png b/rsc/config/awesome/lain/icons/mail.png
new file mode 100644
index 0000000..9c0c7a3
Binary files /dev/null and b/rsc/config/awesome/lain/icons/mail.png differ
diff --git a/rsc/config/awesome/lain/icons/no_net.png b/rsc/config/awesome/lain/icons/no_net.png
new file mode 100644
index 0000000..3613372
Binary files /dev/null and b/rsc/config/awesome/lain/icons/no_net.png differ
diff --git a/rsc/config/awesome/lain/icons/openweathermap/01d.png b/rsc/config/awesome/lain/icons/openweathermap/01d.png
new file mode 100644
index 0000000..569965e
Binary files /dev/null and b/rsc/config/awesome/lain/icons/openweathermap/01d.png differ
diff --git a/rsc/config/awesome/lain/icons/openweathermap/01n.png b/rsc/config/awesome/lain/icons/openweathermap/01n.png
new file mode 100644
index 0000000..ce5b135
Binary files /dev/null and b/rsc/config/awesome/lain/icons/openweathermap/01n.png differ
diff --git a/rsc/config/awesome/lain/icons/openweathermap/02d.png b/rsc/config/awesome/lain/icons/openweathermap/02d.png
new file mode 100644
index 0000000..2ba9799
Binary files /dev/null and b/rsc/config/awesome/lain/icons/openweathermap/02d.png differ
diff --git a/rsc/config/awesome/lain/icons/openweathermap/02n.png b/rsc/config/awesome/lain/icons/openweathermap/02n.png
new file mode 100644
index 0000000..12e4283
Binary files /dev/null and b/rsc/config/awesome/lain/icons/openweathermap/02n.png differ
diff --git a/rsc/config/awesome/lain/icons/openweathermap/03d.png b/rsc/config/awesome/lain/icons/openweathermap/03d.png
new file mode 100644
index 0000000..1cf0e9d
Binary files /dev/null and b/rsc/config/awesome/lain/icons/openweathermap/03d.png differ
diff --git a/rsc/config/awesome/lain/icons/openweathermap/03n.png b/rsc/config/awesome/lain/icons/openweathermap/03n.png
new file mode 100644
index 0000000..89a42b8
Binary files /dev/null and b/rsc/config/awesome/lain/icons/openweathermap/03n.png differ
diff --git a/rsc/config/awesome/lain/icons/openweathermap/04d.png b/rsc/config/awesome/lain/icons/openweathermap/04d.png
new file mode 100644
index 0000000..e7fb67f
Binary files /dev/null and b/rsc/config/awesome/lain/icons/openweathermap/04d.png differ
diff --git a/rsc/config/awesome/lain/icons/openweathermap/04n.png b/rsc/config/awesome/lain/icons/openweathermap/04n.png
new file mode 120000
index 0000000..b9a83df
--- /dev/null
+++ b/rsc/config/awesome/lain/icons/openweathermap/04n.png
@@ -0,0 +1 @@
+04d.png
\ No newline at end of file
diff --git a/rsc/config/awesome/lain/icons/openweathermap/09d.png b/rsc/config/awesome/lain/icons/openweathermap/09d.png
new file mode 100644
index 0000000..cfa066a
Binary files /dev/null and b/rsc/config/awesome/lain/icons/openweathermap/09d.png differ
diff --git a/rsc/config/awesome/lain/icons/openweathermap/09n.png b/rsc/config/awesome/lain/icons/openweathermap/09n.png
new file mode 120000
index 0000000..cca1f5d
--- /dev/null
+++ b/rsc/config/awesome/lain/icons/openweathermap/09n.png
@@ -0,0 +1 @@
+09d.png
\ No newline at end of file
diff --git a/rsc/config/awesome/lain/icons/openweathermap/10d.png b/rsc/config/awesome/lain/icons/openweathermap/10d.png
new file mode 100644
index 0000000..712d0c8
Binary files /dev/null and b/rsc/config/awesome/lain/icons/openweathermap/10d.png differ
diff --git a/rsc/config/awesome/lain/icons/openweathermap/10n.png b/rsc/config/awesome/lain/icons/openweathermap/10n.png
new file mode 120000
index 0000000..6e01227
--- /dev/null
+++ b/rsc/config/awesome/lain/icons/openweathermap/10n.png
@@ -0,0 +1 @@
+10d.png
\ No newline at end of file
diff --git a/rsc/config/awesome/lain/icons/openweathermap/11d.png b/rsc/config/awesome/lain/icons/openweathermap/11d.png
new file mode 100644
index 0000000..3b62f7c
Binary files /dev/null and b/rsc/config/awesome/lain/icons/openweathermap/11d.png differ
diff --git a/rsc/config/awesome/lain/icons/openweathermap/11n.png b/rsc/config/awesome/lain/icons/openweathermap/11n.png
new file mode 120000
index 0000000..b227917
--- /dev/null
+++ b/rsc/config/awesome/lain/icons/openweathermap/11n.png
@@ -0,0 +1 @@
+11d.png
\ No newline at end of file
diff --git a/rsc/config/awesome/lain/icons/openweathermap/13d.png b/rsc/config/awesome/lain/icons/openweathermap/13d.png
new file mode 100644
index 0000000..e265b01
Binary files /dev/null and b/rsc/config/awesome/lain/icons/openweathermap/13d.png differ
diff --git a/rsc/config/awesome/lain/icons/openweathermap/13n.png b/rsc/config/awesome/lain/icons/openweathermap/13n.png
new file mode 120000
index 0000000..94e5a52
--- /dev/null
+++ b/rsc/config/awesome/lain/icons/openweathermap/13n.png
@@ -0,0 +1 @@
+13d.png
\ No newline at end of file
diff --git a/rsc/config/awesome/lain/icons/openweathermap/50d.png b/rsc/config/awesome/lain/icons/openweathermap/50d.png
new file mode 100644
index 0000000..905ace3
Binary files /dev/null and b/rsc/config/awesome/lain/icons/openweathermap/50d.png differ
diff --git a/rsc/config/awesome/lain/icons/openweathermap/50n.png b/rsc/config/awesome/lain/icons/openweathermap/50n.png
new file mode 120000
index 0000000..e3ba961
--- /dev/null
+++ b/rsc/config/awesome/lain/icons/openweathermap/50n.png
@@ -0,0 +1 @@
+50d.png
\ No newline at end of file
diff --git a/rsc/config/awesome/lain/icons/openweathermap/README.md b/rsc/config/awesome/lain/icons/openweathermap/README.md
new file mode 100644
index 0000000..f908fbd
--- /dev/null
+++ b/rsc/config/awesome/lain/icons/openweathermap/README.md
@@ -0,0 +1,3 @@
+[Plain Weather Icons](http://merlinthered.deviantart.com/art/plain-weather-icons-157162192), created by [MerlinTheRed](http://merlinthered.deviantart.com/).
+
+
diff --git a/rsc/config/awesome/lain/icons/openweathermap/na.png b/rsc/config/awesome/lain/icons/openweathermap/na.png
new file mode 100644
index 0000000..1cc5132
Binary files /dev/null and b/rsc/config/awesome/lain/icons/openweathermap/na.png differ
diff --git a/rsc/config/awesome/lain/icons/taskwarrior.png b/rsc/config/awesome/lain/icons/taskwarrior.png
new file mode 100644
index 0000000..c64fe86
Binary files /dev/null and b/rsc/config/awesome/lain/icons/taskwarrior.png differ
diff --git a/rsc/config/awesome/lain/init.lua b/rsc/config/awesome/lain/init.lua
new file mode 100644
index 0000000..b59d5dd
--- /dev/null
+++ b/rsc/config/awesome/lain/init.lua
@@ -0,0 +1,15 @@
+--[[
+
+ Lain
+ Layouts, widgets and utilities for Awesome WM
+
+ Licensed under GNU General Public License v2
+ * (c) 2013, Luca CPZ
+
+--]]
+
+return {
+ layout = require("lain.layout"),
+ util = require("lain.util"),
+ widget = require("lain.widget")
+}
diff --git a/rsc/config/awesome/lain/lain-scm-1.rockspec b/rsc/config/awesome/lain/lain-scm-1.rockspec
new file mode 100644
index 0000000..57fd1f5
--- /dev/null
+++ b/rsc/config/awesome/lain/lain-scm-1.rockspec
@@ -0,0 +1,57 @@
+rockspec_format = "3.0"
+package = "lain"
+version = "scm-1"
+source = {
+ url = "git+https://github.com/lcpz/lain.git"
+}
+description = {
+ summary = "Layout, widgets and utilities for Awesome WM",
+ detailed = "Alternative layouts, asynchronous widgets and utility functions for Awesome WM. Non-Lua dependency: curl (for IMAP, MPD and weather widgets).",
+ homepage = "https://github.com/lcpz/lain",
+ issues_url = "https://github.com/lcpz/lain/issues",
+ maintainer = "Luca Cpz",
+ license = "GPL2"
+}
+dependencies = {
+ "lua >= 5.3",
+ "dkjson >= 2.6-1"
+}
+supported_platforms = { "linux" }
+build = {
+ type = "builtin",
+ modules = {
+ ["lain"] = "init.lua",
+ ["lain.helpers"] = "helpers.lua",
+ ["lain.layout"] = "layout/init.lua",
+ ["lain.layout.cascade"] = "layout/cascade.lua",
+ ["lain.layout.centerwork"] = "layout/centerwork.lua",
+ ["lain.layout.termfair"] = "layout/termfair.lua",
+ ["lain.util"] = "util/init.lua",
+ -- ["lain.util.dkjson"] = "util/dkjson.lua", -- RESOLVED BY DEPENDENCY TO dkjson
+ ["lain.util.markup"] = "util/markup.lua",
+ ["lain.util.menu_iterator"] = "util/menu_iterator.lua",
+ ["lain.util.quake"] = "util/quake.lua",
+ ["lain.util.separators"] = "util/separators.lua",
+ ["lain.widget"] = "widget/init.lua",
+ ["lain.widget.contrib"] = "widget/contrib/init.lua",
+ ["lain.widget.contrib.moc"] = "widget/contrib/moc.lua",
+ ["lain.widget.contrib.redshift"] = "widget/contrib/redshift.lua",
+ ["lain.widget.contrib.task"] = "widget/contrib/task.lua",
+ ["lain.widget.contrib.tp_smapi"] = "widget/contrib/tp_smapi.lua",
+ ["lain.widget.alsa"] = "widget/alsa.lua",
+ ["lain.widget.alsabar"] = "widget/alsabar.lua",
+ ["lain.widget.bat"] = "widget/bat.lua",
+ ["lain.widget.cal"] = "widget/cal.lua",
+ ["lain.widget.cpu"] = "widget/cpu.lua",
+ ["lain.widget.fs"] = "widget/fs.lua",
+ ["lain.widget.imap"] = "widget/imap.lua",
+ ["lain.widget.mem"] = "widget/mem.lua",
+ ["lain.widget.mpd"] = "widget/mpd.lua",
+ ["lain.widget.net"] = "widget/net.lua",
+ ["lain.widget.pulse"] = "widget/pulse.lua",
+ ["lain.widget.pulsebar"] = "widget/pulsebar.lua",
+ ["lain.widget.sysload"] = "widget/sysload.lua",
+ ["lain.widget.temp"] = "widget/temp.lua",
+ ["lain.widget.weather"] = "widget/weather.lua"
+ }
+}
diff --git a/rsc/config/awesome/lain/layout/cascade.lua b/rsc/config/awesome/lain/layout/cascade.lua
new file mode 100644
index 0000000..cbc3877
--- /dev/null
+++ b/rsc/config/awesome/lain/layout/cascade.lua
@@ -0,0 +1,172 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2014, projektile
+ * (c) 2013, Luca CPZ
+ * (c) 2010-2012, Peter Hofmann
+
+--]]
+
+local floor = math.floor
+local screen = screen
+
+local cascade = {
+ name = "cascade",
+ nmaster = 0,
+ offset_x = 32,
+ offset_y = 8,
+ tile = {
+ name = "cascadetile",
+ nmaster = 0,
+ ncol = 0,
+ mwfact = 0,
+ offset_x = 5,
+ offset_y = 32,
+ extra_padding = 0
+ }
+}
+
+local function do_cascade(p, tiling)
+ local t = p.tag or screen[p.screen].selected_tag
+ local wa = p.workarea
+ local cls = p.clients
+
+ if #cls == 0 then return end
+
+ if not tiling then
+ -- Cascade windows.
+
+ local num_c
+ if cascade.nmaster > 0 then
+ num_c = cascade.nmaster
+ else
+ num_c = t.master_count
+ end
+
+ -- Opening a new window will usually force all existing windows to
+ -- get resized. This wastes a lot of CPU time. So let's set a lower
+ -- bound to "how_many": This wastes a little screen space but you'll
+ -- get a much better user experience.
+ local how_many = (#cls >= num_c and #cls) or num_c
+
+ local current_offset_x = cascade.offset_x * (how_many - 1)
+ local current_offset_y = cascade.offset_y * (how_many - 1)
+
+ -- Iterate.
+ for i = 1,#cls,1 do
+ local c = cls[i]
+ local g = {}
+
+ g.x = wa.x + (how_many - i) * cascade.offset_x
+ g.y = wa.y + (i - 1) * cascade.offset_y
+ g.width = wa.width - current_offset_x
+ g.height = wa.height - current_offset_y
+
+ if g.width < 1 then g.width = 1 end
+ if g.height < 1 then g.height = 1 end
+
+ p.geometries[c] = g
+ end
+ else
+ -- Layout with one fixed column meant for a master window. Its
+ -- width is calculated according to mwfact. Other clients are
+ -- cascaded or "tabbed" in a slave column on the right.
+
+ -- (1) (2) (3) (4)
+ -- +----------+---+ +----------+---+ +----------+---+ +----------+---+
+ -- | | | | | 3 | | | 4 | | +---+|
+ -- | | | -> | | | -> | +---++ -> | +---+|+
+ -- | 1 | 2 | | 1 +---++ | 1 | 3 || | 1 +---+|+|
+ -- | | | | | 2 || | +---++| | +---+|+ |
+ -- | | | | | || | | 2 | | | | 2 |+ |
+ -- +----------+---+ +---------+---++ +--------+---+-+ +------+---+---+
+
+ local mwfact
+ if cascade.tile.mwfact > 0 then
+ mwfact = cascade.tile.mwfact
+ else
+ mwfact = t.master_width_factor
+ end
+
+ -- Make slave windows overlap main window? Do this if ncol is 1.
+ local overlap_main
+ if cascade.tile.ncol > 0 then
+ overlap_main = cascade.tile.ncol
+ else
+ overlap_main = t.column_count
+ end
+
+ -- Minimum space for slave windows? See cascade.tile.lua.
+ local num_c
+ if cascade.tile.nmaster > 0 then
+ num_c = cascade.tile.nmaster
+ else
+ num_c = t.master_count
+ end
+
+ local how_many = (#cls - 1 >= num_c and (#cls - 1)) or num_c
+
+ local current_offset_x = cascade.tile.offset_x * (how_many - 1)
+ local current_offset_y = cascade.tile.offset_y * (how_many - 1)
+
+ if #cls <= 0 then return end
+
+ -- Main column, fixed width and height.
+ local c = cls[1]
+ local g = {}
+ -- Rounding is necessary to prevent the rendered size of slavewid
+ -- from being 1 pixel off when the result is not an integer.
+ local mainwid = floor(wa.width * mwfact)
+ local slavewid = wa.width - mainwid
+
+ if overlap_main == 1 then
+ g.width = wa.width
+
+ -- The size of the main window may be reduced a little bit.
+ -- This allows you to see if there are any windows below the
+ -- main window.
+ -- This only makes sense, though, if the main window is
+ -- overlapping everything else.
+ g.width = g.width - cascade.tile.extra_padding
+ else
+ g.width = mainwid
+ end
+
+ g.height = wa.height
+ g.x = wa.x
+ g.y = wa.y
+
+ if g.width < 1 then g.width = 1 end
+ if g.height < 1 then g.height = 1 end
+
+ p.geometries[c] = g
+
+ -- Remaining clients stacked in slave column, new ones on top.
+ if #cls <= 1 then return end
+ for i = 2,#cls do
+ c = cls[i]
+ g = {}
+
+ g.width = slavewid - current_offset_x
+ g.height = wa.height - current_offset_y
+
+ g.x = wa.x + mainwid + (how_many - (i - 1)) * cascade.tile.offset_x
+ g.y = wa.y + (i - 2) * cascade.tile.offset_y
+
+ if g.width < 1 then g.width = 1 end
+ if g.height < 1 then g.height = 1 end
+
+ p.geometries[c] = g
+ end
+ end
+end
+
+function cascade.tile.arrange(p)
+ return do_cascade(p, true)
+end
+
+function cascade.arrange(p)
+ return do_cascade(p, false)
+end
+
+return cascade
diff --git a/rsc/config/awesome/lain/layout/centerwork.lua b/rsc/config/awesome/lain/layout/centerwork.lua
new file mode 100644
index 0000000..c105676
--- /dev/null
+++ b/rsc/config/awesome/lain/layout/centerwork.lua
@@ -0,0 +1,276 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2018, Eugene Pakhomov
+ * (c) 2016, Henrik Antonsson
+ * (c) 2015, Joerg Jaspert
+ * (c) 2014, projektile
+ * (c) 2013, Luca CPZ
+ * (c) 2010-2012, Peter Hofmann
+
+--]]
+
+local floor = math.floor
+local max = math.max
+local mouse = mouse
+local mousegrabber = mousegrabber
+local screen = screen
+
+local centerwork = {
+ name = "centerwork",
+ horizontal = { name = "centerworkh" }
+}
+
+local function arrange(p, layout)
+ local t = p.tag or screen[p.screen].selected_tag
+ local wa = p.workarea
+ local cls = p.clients
+
+ if #cls == 0 then return end
+
+ local g = {}
+
+ -- Main column, fixed width and height
+ local mwfact = t.master_width_factor
+ local mainhei = floor(wa.height * mwfact)
+ local mainwid = floor(wa.width * mwfact)
+ local slavewid = wa.width - mainwid
+ local slaveLwid = floor(slavewid / 2)
+ local slaveRwid = slavewid - slaveLwid
+ local slavehei = wa.height - mainhei
+ local slaveThei = floor(slavehei / 2)
+ local slaveBhei = slavehei - slaveThei
+ local nbrFirstSlaves = floor(#cls / 2)
+ local nbrSecondSlaves = floor((#cls - 1) / 2)
+
+ local slaveFirstDim, slaveSecondDim = 0, 0
+
+ if layout.name == "centerwork" then -- vertical
+ if nbrFirstSlaves > 0 then slaveFirstDim = floor(wa.height / nbrFirstSlaves) end
+ if nbrSecondSlaves > 0 then slaveSecondDim = floor(wa.height / nbrSecondSlaves) end
+
+ g.height = wa.height
+ g.width = mainwid
+
+ g.x = wa.x + slaveLwid
+ g.y = wa.y
+ else -- horizontal
+ if nbrFirstSlaves > 0 then slaveFirstDim = floor(wa.width / nbrFirstSlaves) end
+ if nbrSecondSlaves > 0 then slaveSecondDim = floor(wa.width / nbrSecondSlaves) end
+
+ g.height = mainhei
+ g.width = wa.width
+
+ g.x = wa.x
+ g.y = wa.y + slaveThei
+ end
+
+ g.width = max(g.width, 1)
+ g.height = max(g.height, 1)
+
+ p.geometries[cls[1]] = g
+
+ -- Auxiliary clients
+ if #cls <= 1 then return end
+ for i = 2, #cls do
+ g = {}
+ local idxChecker, dimToAssign
+
+ local rowIndex = floor(i/2)
+
+ if layout.name == "centerwork" then
+ if i % 2 == 0 then -- left slave
+ g.x = wa.x
+ g.y = wa.y + (rowIndex - 1) * slaveFirstDim
+ g.width = slaveLwid
+
+ idxChecker, dimToAssign = nbrFirstSlaves, slaveFirstDim
+ else -- right slave
+ g.x = wa.x + slaveLwid + mainwid
+ g.y = wa.y + (rowIndex - 1) * slaveSecondDim
+ g.width = slaveRwid
+
+ idxChecker, dimToAssign = nbrSecondSlaves, slaveSecondDim
+ end
+
+ -- if last slave in row, use remaining space for it
+ if rowIndex == idxChecker then
+ g.height = wa.y + wa.height - g.y
+ else
+ g.height = dimToAssign
+ end
+ else
+ if i % 2 == 0 then -- top slave
+ g.x = wa.x + (rowIndex - 1) * slaveFirstDim
+ g.y = wa.y
+ g.height = slaveThei
+
+ idxChecker, dimToAssign = nbrFirstSlaves, slaveFirstDim
+ else -- bottom slave
+ g.x = wa.x + (rowIndex - 1) * slaveSecondDim
+ g.y = wa.y + slaveThei + mainhei
+ g.height = slaveBhei
+
+ idxChecker, dimToAssign = nbrSecondSlaves, slaveSecondDim
+ end
+
+ -- if last slave in row, use remaining space for it
+ if rowIndex == idxChecker then
+ g.width = wa.x + wa.width - g.x
+ else
+ g.width = dimToAssign
+ end
+ end
+
+ g.width = max(g.width, 1)
+ g.height = max(g.height, 1)
+
+ p.geometries[cls[i]] = g
+ end
+end
+
+local function mouse_resize_handler(c, _, _, _, orientation)
+ local wa = c.screen.workarea
+ local mwfact = c.screen.selected_tag.master_width_factor
+ local g = c:geometry()
+ local offset = 0
+ local cursor = "cross"
+
+ local corner_coords
+
+ if orientation == 'vertical' then
+ if g.height + 15 >= wa.height then
+ offset = g.height * .5
+ cursor = "sb_h_double_arrow"
+ elseif g.y + g.height + 15 <= wa.y + wa.height then
+ offset = g.height
+ end
+ corner_coords = { x = wa.x + wa.width * (1 - mwfact) / 2, y = g.y + offset }
+ else
+ if g.width + 15 >= wa.width then
+ offset = g.width * .5
+ cursor = "sb_v_double_arrow"
+ elseif g.x + g.width + 15 <= wa.x + wa.width then
+ offset = g.width
+ end
+ corner_coords = { y = wa.y + wa.height * (1 - mwfact) / 2, x = g.x + offset }
+ end
+
+ mouse.coords(corner_coords)
+
+ local prev_coords = {}
+
+ mousegrabber.run(function(m)
+ if not c.valid then return false end
+ for _, v in ipairs(m.buttons) do
+ if v then
+ prev_coords = { x = m.x, y = m.y }
+ local new_mwfact
+ if orientation == 'vertical' then
+ new_mwfact = 1 - (m.x - wa.x) / wa.width * 2
+ else
+ new_mwfact = 1 - (m.y - wa.y) / wa.height * 2
+ end
+ c.screen.selected_tag.master_width_factor = math.min(math.max(new_mwfact, 0.01), 0.99)
+ return true
+ end
+ end
+ return prev_coords.x == m.x and prev_coords.y == m.y
+ end, cursor)
+end
+
+function centerwork.arrange(p)
+ return arrange(p, centerwork)
+end
+
+function centerwork.horizontal.arrange(p)
+ return arrange(p, centerwork.horizontal)
+end
+
+function centerwork.mouse_resize_handler(c, corner, x, y)
+ return mouse_resize_handler(c, corner, x, y, 'vertical')
+end
+
+function centerwork.horizontal.mouse_resize_handler(c, corner, x, y)
+ return mouse_resize_handler(c, corner, x, y, 'horizontal')
+end
+
+
+--[[
+Make focus.byidx and swap.byidx behave more consistently with other layouts.
+--]]
+
+local awful = require("awful")
+local gears = require("gears")
+local client = client
+
+local function compare_position(a, b)
+ if a.x == b.x then
+ return a.y < b.y
+ else
+ return a.x < b.x
+ end
+end
+
+local function clients_by_position()
+ local this = client.focus
+ if this then
+ local sorted = {}
+ for _, c in ipairs(client.focus.first_tag:clients()) do
+ if not c.minimized then sorted[#sorted+1] = c end
+ end
+ table.sort(sorted, compare_position)
+
+ local idx = 0
+ for i, that in ipairs(sorted) do
+ if this.window == that.window then
+ idx = i
+ end
+ end
+
+ if idx > 0 then
+ return { sorted = sorted, idx = idx }
+ end
+ end
+ return {}
+end
+
+local function in_centerwork()
+ return client.focus and client.focus.first_tag.layout.name == "centerwork"
+end
+
+centerwork.focus = {}
+
+
+--[[
+Drop in replacements for awful.client.focus.byidx and awful.client.swap.byidx
+that behaves consistently with other layouts.
+--]]
+
+function centerwork.focus.byidx(i)
+ if in_centerwork() then
+ local cls = clients_by_position()
+ if cls.idx then
+ local target = cls.sorted[gears.math.cycle(#cls.sorted, cls.idx + i)]
+ awful.client.focus.byidx(0, target)
+ end
+ else
+ awful.client.focus.byidx(i)
+ end
+end
+
+centerwork.swap = {}
+
+function centerwork.swap.byidx(i)
+ if in_centerwork() then
+ local cls = clients_by_position()
+ if cls.idx then
+ local target = cls.sorted[gears.math.cycle(#cls.sorted, cls.idx + i)]
+ client.focus:swap(target)
+ end
+ else
+ awful.client.swap.byidx(i)
+ end
+end
+
+return centerwork
diff --git a/rsc/config/awesome/lain/layout/init.lua b/rsc/config/awesome/lain/layout/init.lua
new file mode 100644
index 0000000..6478b06
--- /dev/null
+++ b/rsc/config/awesome/lain/layout/init.lua
@@ -0,0 +1,19 @@
+--[[
+
+ Lain
+ Layouts, widgets and utilities for Awesome WM
+
+ Layouts section
+
+ Licensed under GNU General Public License v2
+ * (c) 2013, Luca CPZ
+ * (c) 2010-2012, Peter Hofmann
+
+--]]
+
+local wrequire = require("lain.helpers").wrequire
+local setmetatable = setmetatable
+
+local layout = { _NAME = "lain.layout" }
+
+return setmetatable(layout, { __index = wrequire })
diff --git a/rsc/config/awesome/lain/layout/termfair.lua b/rsc/config/awesome/lain/layout/termfair.lua
new file mode 100644
index 0000000..cf018ef
--- /dev/null
+++ b/rsc/config/awesome/lain/layout/termfair.lua
@@ -0,0 +1,282 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2014, projektile
+ * (c) 2013, Luca CPZ
+ * (c) 2010, Nicolas Estibals
+ * (c) 2010-2012, Peter Hofmann
+
+--]]
+
+local math = math
+local screen = screen
+local tonumber = tonumber
+
+local termfair = { name = "termfair" }
+termfair.center = { name = "centerfair" }
+termfair.stable = { name = "stablefair" }
+
+local function do_fair(p, orientation)
+ local t = p.tag or screen[p.screen].selected_tag
+ local wa = p.workarea
+ local cls = p.clients
+
+ if #cls == 0 then return end
+
+ -- How many vertical columns? Read from nmaster on the tag.
+ local num_x = tonumber(termfair.nmaster) or t.master_count
+ local ncol = tonumber(termfair.ncol) or t.column_count
+ if num_x <= 2 then num_x = 2 end
+ if ncol <= 1 then ncol = 1 end
+ local width = math.floor(wa.width/num_x)
+
+ if orientation == "west" then
+ -- Layout with fixed number of vertical columns (read from nmaster).
+ -- New windows align from left to right. When a row is full, a new
+ -- one above it is created. Like this:
+
+ -- (1) (2) (3)
+ -- +---+---+---+ +---+---+---+ +---+---+---+
+ -- | | | | | | | | | | | |
+ -- | 1 | | | -> | 1 | 2 | | -> | 1 | 2 | 3 | ->
+ -- | | | | | | | | | | | |
+ -- +---+---+---+ +---+---+---+ +---+---+---+
+
+ -- (4) (5) (6)
+ -- +---+---+---+ +---+---+---+ +---+---+---+
+ -- | 1 | | | | 1 | 2 | | | 1 | 2 | 3 |
+ -- +---+---+---+ -> +---+---+---+ -> +---+---+---+
+ -- | 2 | 3 | 4 | | 3 | 4 | 5 | | 4 | 5 | 6 |
+ -- +---+---+---+ +---+---+---+ +---+---+---+
+
+ local num_y = math.max(math.ceil(#cls / num_x), ncol)
+ local height = math.floor(wa.height/num_y)
+ local cur_num_x = num_x
+ local at_x = 0
+ local at_y = 0
+
+ local remaining_clients = #cls
+
+ -- We start the first row. Left-align by limiting the number of
+ -- available slots.
+ if remaining_clients < num_x then
+ cur_num_x = remaining_clients
+ end
+
+ -- Iterate in reversed order.
+ for i = #cls,1,-1 do
+ -- Get x and y position.
+ local c = cls[i]
+ local this_x = cur_num_x - at_x - 1
+ local this_y = num_y - at_y - 1
+
+ -- Calculate geometry.
+ local g = {}
+ if this_x == (num_x - 1) then
+ g.width = wa.width - (num_x - 1)*width
+ else
+ g.width = width
+ end
+
+ if this_y == (num_y - 1) then
+ g.height = wa.height - (num_y - 1)*height
+ else
+ g.height = height
+ end
+
+ g.x = wa.x + this_x*width
+ g.y = wa.y + this_y*height
+
+ if g.width < 1 then g.width = 1 end
+ if g.height < 1 then g.height = 1 end
+
+ p.geometries[c] = g
+
+ remaining_clients = remaining_clients - 1
+
+ -- Next grid position.
+ at_x = at_x + 1
+ if at_x == num_x then
+ -- Row full, create a new one above it.
+ at_x = 0
+ at_y = at_y + 1
+
+ -- We start a new row. Left-align.
+ if remaining_clients < num_x then
+ cur_num_x = remaining_clients
+ end
+ end
+ end
+ elseif orientation == "stable" then
+ -- Layout with fixed number of vertical columns (read from nmaster).
+ -- New windows align from left to right. When a row is full, a new
+ -- one below it is created. Like this:
+
+ -- (1) (2) (3)
+ -- +---+---+---+ +---+---+---+ +---+---+---+
+ -- | | | | | | | | | | | |
+ -- | 1 | | | -> | 1 | 2 | | -> | 1 | 2 | 3 | ->
+ -- | | | | | | | | | | | |
+ -- +---+---+---+ +---+---+---+ +---+---+---+
+
+ -- (4) (5) (6)
+ -- +---+---+---+ +---+---+---+ +---+---+---+
+ -- | 1 | 2 | 3 | | 1 | 2 | 3 | | 1 | 2 | 3 |
+ -- +---+---+---+ +---+---+---+ +---+---+---+
+ -- | 4 | | | | 4 | 5 | | | 4 | 5 | 6 |
+ -- +---+---+---+ -> +---+---+---+ -> +---+---+---+
+
+ local num_y = math.max(math.ceil(#cls / num_x), ncol)
+ local height = math.floor(wa.height/num_y)
+
+ for i = #cls,1,-1 do
+ -- Get x and y position.
+ local c = cls[i]
+ local this_x = (i - 1) % num_x
+ local this_y = math.floor((i - this_x - 1) / num_x)
+
+ -- Calculate geometry.
+ local g = {}
+ if this_x == (num_x - 1) then
+ g.width = wa.width - (num_x - 1)*width
+ else
+ g.width = width
+ end
+
+ if this_y == (num_y - 1) then
+ g.height = wa.height - (num_y - 1)*height
+ else
+ g.height = height
+ end
+
+ g.x = wa.x + this_x*width
+ g.y = wa.y + this_y*height
+
+ if g.width < 1 then g.width = 1 end
+ if g.height < 1 then g.height = 1 end
+
+ p.geometries[c] = g
+ end
+ elseif orientation == "center" then
+ -- Layout with fixed number of vertical columns (read from nmaster).
+ -- Cols are centerded until there is nmaster columns, then windows
+ -- are stacked in the slave columns, with at most ncol clients per
+ -- column if possible.
+
+ -- with nmaster=3 and ncol=1 you'll have
+ -- (1) (2) (3)
+ -- +---+---+---+ +-+---+---+-+ +---+---+---+
+ -- | | | | | | | | | | | | |
+ -- | | 1 | | -> | | 1 | 2 | | -> | 1 | 2 | 3 | ->
+ -- | | | | | | | | | | | | |
+ -- +---+---+---+ +-+---+---+-+ +---+---+---+
+
+ -- (4) (5)
+ -- +---+---+---+ +---+---+---+
+ -- | | | 3 | | | 2 | 4 |
+ -- + 1 + 2 +---+ -> + 1 +---+---+
+ -- | | | 4 | | | 3 | 5 |
+ -- +---+---+---+ +---+---+---+
+
+ if #cls < num_x then
+ -- Less clients than the number of columns, let's center it!
+ local offset_x = wa.x + (wa.width - #cls*width) / 2
+ for i = 1, #cls do
+ local g = { y = wa.y }
+ g.width = width
+ g.height = wa.height
+ if g.width < 1 then g.width = 1 end
+ if g.height < 1 then g.height = 1 end
+ g.x = offset_x + (i - 1) * width
+ p.geometries[cls[i]] = g
+ end
+ else
+ -- More clients than the number of columns, let's arrange it!
+ -- Master client deserves a special treatement
+ local g = {}
+ g.width = wa.width - (num_x - 1)*width
+ g.height = wa.height
+ if g.width < 1 then g.width = 1 end
+ if g.height < 1 then g.height = 1 end
+ g.x = wa.x
+ g.y = wa.y
+ p.geometries[cls[1]] = g
+
+ -- Treat the other clients
+
+ -- Compute distribution of clients among columns
+ local num_y = {}
+ local remaining_clients = #cls-1
+ local ncol_min = math.ceil(remaining_clients/(num_x-1))
+
+ if ncol >= ncol_min then
+ for i = (num_x-1), 1, -1 do
+ if (remaining_clients-i+1) < ncol then
+ num_y[i] = remaining_clients-i + 1
+ else
+ num_y[i] = ncol
+ end
+ remaining_clients = remaining_clients - num_y[i]
+ end
+ else
+ local rem = remaining_clients % (num_x-1)
+ if rem == 0 then
+ for i = 1, num_x-1 do
+ num_y[i] = ncol_min
+ end
+ else
+ for i = 1, num_x-1 do
+ num_y[i] = ncol_min - 1
+ end
+ for i = 0, rem-1 do
+ num_y[num_x-1-i] = num_y[num_x-1-i] + 1
+ end
+ end
+ end
+
+ -- Compute geometry of the other clients
+ local nclient = 2 -- we start with the 2nd client
+ local wx = g.x + g.width
+ for i = 1, (num_x-1) do
+ local height = math.floor(wa.height / num_y[i])
+ local wy = wa.y
+ for _ = 0, (num_y[i]-2) do
+ g = {}
+ g.x = wx
+ g.y = wy
+ g.height = height
+ g.width = width
+ if g.width < 1 then g.width = 1 end
+ if g.height < 1 then g.height = 1 end
+ p.geometries[cls[nclient]] = g
+ nclient = nclient + 1
+ wy = wy + height
+ end
+ g = {}
+ g.x = wx
+ g.y = wy
+ g.height = wa.height - (num_y[i] - 1)*height
+ g.width = width
+ if g.width < 1 then g.width = 1 end
+ if g.height < 1 then g.height = 1 end
+ p.geometries[cls[nclient]] = g
+ nclient = nclient + 1
+ wx = wx + width
+ end
+ end
+ end
+end
+
+function termfair.center.arrange(p)
+ return do_fair(p, "center")
+end
+
+function termfair.stable.arrange(p)
+ return do_fair(p, "stable")
+end
+
+function termfair.arrange(p)
+ return do_fair(p, "west")
+end
+
+return termfair
diff --git a/rsc/config/awesome/lain/util/dkjson.lua b/rsc/config/awesome/lain/util/dkjson.lua
new file mode 100644
index 0000000..61cccb9
--- /dev/null
+++ b/rsc/config/awesome/lain/util/dkjson.lua
@@ -0,0 +1,747 @@
+-- Module options:
+local always_use_lpeg = false
+local register_global_module_table = false
+local global_module_name = 'json'
+
+--[==[
+
+David Kolf's JSON module for Lua 5.1 - 5.4
+
+Version 2.6
+
+
+For the documentation see the corresponding readme.txt or visit
+.
+
+You can contact the author by sending an e-mail to 'david' at the
+domain 'dkolf.de'.
+
+
+Copyright (C) 2010-2021 David Heiko Kolf
+
+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.
+
+--]==]
+
+-- global dependencies:
+local pairs, type, tostring, tonumber, getmetatable, setmetatable =
+ pairs, type, tostring, tonumber, getmetatable, setmetatable
+local error, require, pcall, select = error, require, pcall, select
+local floor, huge = math.floor, math.huge
+local strrep, gsub, strsub, strbyte, strchar, strfind, strlen, strformat =
+ string.rep, string.gsub, string.sub, string.byte, string.char,
+ string.find, string.len, string.format
+local strmatch = string.match
+local concat = table.concat
+
+local json = { version = "dkjson 2.6" }
+
+local jsonlpeg = {}
+
+if register_global_module_table then
+ if always_use_lpeg then
+ _G[global_module_name] = jsonlpeg
+ else
+ _G[global_module_name] = json
+ end
+end
+
+_ENV = nil -- blocking globals in Lua 5.2 and later
+
+pcall (function()
+ -- Enable access to blocked metatables.
+ -- Don't worry, this module doesn't change anything in them.
+ local debmeta = require "debug".getmetatable
+ if debmeta then getmetatable = debmeta end
+end)
+
+json.null = setmetatable ({}, {
+ __tojson = function () return "null" end
+})
+
+local function isarray (tbl)
+ local max, n, arraylen = 0, 0, 0
+ for k,v in pairs (tbl) do
+ if k == 'n' and type(v) == 'number' then
+ arraylen = v
+ if v > max then
+ max = v
+ end
+ else
+ if type(k) ~= 'number' or k < 1 or floor(k) ~= k then
+ return false
+ end
+ if k > max then
+ max = k
+ end
+ n = n + 1
+ end
+ end
+ if max > 10 and max > arraylen and max > n * 2 then
+ return false -- don't create an array with too many holes
+ end
+ return true, max
+end
+
+local escapecodes = {
+ ["\""] = "\\\"", ["\\"] = "\\\\", ["\b"] = "\\b", ["\f"] = "\\f",
+ ["\n"] = "\\n", ["\r"] = "\\r", ["\t"] = "\\t"
+}
+
+local function escapeutf8 (uchar)
+ local value = escapecodes[uchar]
+ if value then
+ return value
+ end
+ local a, b, c, d = strbyte (uchar, 1, 4)
+ a, b, c, d = a or 0, b or 0, c or 0, d or 0
+ if a <= 0x7f then
+ value = a
+ elseif 0xc0 <= a and a <= 0xdf and b >= 0x80 then
+ value = (a - 0xc0) * 0x40 + b - 0x80
+ elseif 0xe0 <= a and a <= 0xef and b >= 0x80 and c >= 0x80 then
+ value = ((a - 0xe0) * 0x40 + b - 0x80) * 0x40 + c - 0x80
+ elseif 0xf0 <= a and a <= 0xf7 and b >= 0x80 and c >= 0x80 and d >= 0x80 then
+ value = (((a - 0xf0) * 0x40 + b - 0x80) * 0x40 + c - 0x80) * 0x40 + d - 0x80
+ else
+ return ""
+ end
+ if value <= 0xffff then
+ return strformat ("\\u%.4x", value)
+ elseif value <= 0x10ffff then
+ -- encode as UTF-16 surrogate pair
+ value = value - 0x10000
+ local highsur, lowsur = 0xD800 + floor (value/0x400), 0xDC00 + (value % 0x400)
+ return strformat ("\\u%.4x\\u%.4x", highsur, lowsur)
+ else
+ return ""
+ end
+end
+
+local function fsub (str, pattern, repl)
+ -- gsub always builds a new string in a buffer, even when no match
+ -- exists. First using find should be more efficient when most strings
+ -- don't contain the pattern.
+ if strfind (str, pattern) then
+ return gsub (str, pattern, repl)
+ else
+ return str
+ end
+end
+
+local function quotestring (value)
+ -- based on the regexp "escapable" in https://github.com/douglascrockford/JSON-js
+ value = fsub (value, "[%z\1-\31\"\\\127]", escapeutf8)
+ if strfind (value, "[\194\216\220\225\226\239]") then
+ value = fsub (value, "\194[\128-\159\173]", escapeutf8)
+ value = fsub (value, "\216[\128-\132]", escapeutf8)
+ value = fsub (value, "\220\143", escapeutf8)
+ value = fsub (value, "\225\158[\180\181]", escapeutf8)
+ value = fsub (value, "\226\128[\140-\143\168-\175]", escapeutf8)
+ value = fsub (value, "\226\129[\160-\175]", escapeutf8)
+ value = fsub (value, "\239\187\191", escapeutf8)
+ value = fsub (value, "\239\191[\176-\191]", escapeutf8)
+ end
+ return "\"" .. value .. "\""
+end
+json.quotestring = quotestring
+
+local function replace(str, o, n)
+ local i, j = strfind (str, o, 1, true)
+ if i then
+ return strsub(str, 1, i-1) .. n .. strsub(str, j+1, -1)
+ else
+ return str
+ end
+end
+
+-- locale independent num2str and str2num functions
+local decpoint, numfilter
+
+local function updatedecpoint ()
+ decpoint = strmatch(tostring(0.5), "([^05+])")
+ -- build a filter that can be used to remove group separators
+ numfilter = "[^0-9%-%+eE" .. gsub(decpoint, "[%^%$%(%)%%%.%[%]%*%+%-%?]", "%%%0") .. "]+"
+end
+
+updatedecpoint()
+
+local function num2str (num)
+ return replace(fsub(tostring(num), numfilter, ""), decpoint, ".")
+end
+
+local function str2num (str)
+ local num = tonumber(replace(str, ".", decpoint))
+ if not num then
+ updatedecpoint()
+ num = tonumber(replace(str, ".", decpoint))
+ end
+ return num
+end
+
+local function addnewline2 (level, buffer, buflen)
+ buffer[buflen+1] = "\n"
+ buffer[buflen+2] = strrep (" ", level)
+ buflen = buflen + 2
+ return buflen
+end
+
+function json.addnewline (state)
+ if state.indent then
+ state.bufferlen = addnewline2 (state.level or 0,
+ state.buffer, state.bufferlen or #(state.buffer))
+ end
+end
+
+local encode2 -- forward declaration
+
+local function addpair (key, value, prev, indent, level, buffer, buflen, tables, globalorder, state)
+ local kt = type (key)
+ if kt ~= 'string' and kt ~= 'number' then
+ return nil, "type '" .. kt .. "' is not supported as a key by JSON."
+ end
+ if prev then
+ buflen = buflen + 1
+ buffer[buflen] = ","
+ end
+ if indent then
+ buflen = addnewline2 (level, buffer, buflen)
+ end
+ buffer[buflen+1] = quotestring (key)
+ buffer[buflen+2] = ":"
+ return encode2 (value, indent, level, buffer, buflen + 2, tables, globalorder, state)
+end
+
+local function appendcustom(res, buffer, state)
+ local buflen = state.bufferlen
+ if type (res) == 'string' then
+ buflen = buflen + 1
+ buffer[buflen] = res
+ end
+ return buflen
+end
+
+local function exception(reason, value, state, buffer, buflen, defaultmessage)
+ defaultmessage = defaultmessage or reason
+ local handler = state.exception
+ if not handler then
+ return nil, defaultmessage
+ else
+ state.bufferlen = buflen
+ local ret, msg = handler (reason, value, state, defaultmessage)
+ if not ret then return nil, msg or defaultmessage end
+ return appendcustom(ret, buffer, state)
+ end
+end
+
+function json.encodeexception(_reason, _value, _state, defaultmessage)
+ return quotestring("<" .. defaultmessage .. ">")
+end
+
+encode2 = function (value, indent, level, buffer, buflen, tables, globalorder, state)
+ local valtype = type (value)
+ local valmeta = getmetatable (value)
+ valmeta = type (valmeta) == 'table' and valmeta -- only tables
+ local valtojson = valmeta and valmeta.__tojson
+ if valtojson then
+ if tables[value] then
+ return exception('reference cycle', value, state, buffer, buflen)
+ end
+ tables[value] = true
+ state.bufferlen = buflen
+ local ret, msg = valtojson (value, state)
+ if not ret then return exception('custom encoder failed', value, state, buffer, buflen, msg) end
+ tables[value] = nil
+ buflen = appendcustom(ret, buffer, state)
+ elseif value == nil then
+ buflen = buflen + 1
+ buffer[buflen] = "null"
+ elseif valtype == 'number' then
+ local s
+ if value ~= value or value >= huge or -value >= huge then
+ -- This is the behaviour of the original JSON implementation.
+ s = "null"
+ else
+ s = num2str (value)
+ end
+ buflen = buflen + 1
+ buffer[buflen] = s
+ elseif valtype == 'boolean' then
+ buflen = buflen + 1
+ buffer[buflen] = value and "true" or "false"
+ elseif valtype == 'string' then
+ buflen = buflen + 1
+ buffer[buflen] = quotestring (value)
+ elseif valtype == 'table' then
+ if tables[value] then
+ return exception('reference cycle', value, state, buffer, buflen)
+ end
+ tables[value] = true
+ level = level + 1
+ local isa, n = isarray (value)
+ if n == 0 and valmeta and valmeta.__jsontype == 'object' then
+ isa = false
+ end
+ local msg
+ if isa then -- JSON array
+ buflen = buflen + 1
+ buffer[buflen] = "["
+ for i = 1, n do
+ buflen, msg = encode2 (value[i], indent, level, buffer, buflen, tables, globalorder, state)
+ if not buflen then return nil, msg end
+ if i < n then
+ buflen = buflen + 1
+ buffer[buflen] = ","
+ end
+ end
+ buflen = buflen + 1
+ buffer[buflen] = "]"
+ else -- JSON object
+ local prev = false
+ buflen = buflen + 1
+ buffer[buflen] = "{"
+ local order = valmeta and valmeta.__jsonorder or globalorder
+ if order then
+ local used = {}
+ n = #order
+ for i = 1, n do
+ local k = order[i]
+ local v = value[k]
+ if v ~= nil then
+ used[k] = true
+ buflen, _msg = addpair (k, v, prev, indent, level, buffer, buflen, tables, globalorder, state)
+ prev = true -- add a seperator before the next element
+ end
+ end
+ for k,v in pairs (value) do
+ if not used[k] then
+ buflen, msg = addpair (k, v, prev, indent, level, buffer, buflen, tables, globalorder, state)
+ if not buflen then return nil, msg end
+ prev = true -- add a seperator before the next element
+ end
+ end
+ else -- unordered
+ for k,v in pairs (value) do
+ buflen, msg = addpair (k, v, prev, indent, level, buffer, buflen, tables, globalorder, state)
+ if not buflen then return nil, msg end
+ prev = true -- add a seperator before the next element
+ end
+ end
+ if indent then
+ buflen = addnewline2 (level - 1, buffer, buflen)
+ end
+ buflen = buflen + 1
+ buffer[buflen] = "}"
+ end
+ tables[value] = nil
+ else
+ return exception ('unsupported type', value, state, buffer, buflen,
+ "type '" .. valtype .. "' is not supported by JSON.")
+ end
+ return buflen
+end
+
+function json.encode (value, state)
+ state = state or {}
+ local oldbuffer = state.buffer
+ local buffer = oldbuffer or {}
+ state.buffer = buffer
+ updatedecpoint()
+ local ret, msg = encode2 (value, state.indent, state.level or 0,
+ buffer, state.bufferlen or 0, state.tables or {}, state.keyorder, state)
+ if not ret then
+ error (msg, 2)
+ elseif oldbuffer == buffer then
+ state.bufferlen = ret
+ return true
+ else
+ state.bufferlen = nil
+ state.buffer = nil
+ return concat (buffer)
+ end
+end
+
+local function loc (str, where)
+ local line, pos, linepos = 1, 1, 0
+ while true do
+ pos = strfind (str, "\n", pos, true)
+ if pos and pos < where then
+ line = line + 1
+ linepos = pos
+ pos = pos + 1
+ else
+ break
+ end
+ end
+ return "line " .. line .. ", column " .. (where - linepos)
+end
+
+local function unterminated (str, what, where)
+ return nil, strlen (str) + 1, "unterminated " .. what .. " at " .. loc (str, where)
+end
+
+local function scanwhite (str, pos)
+ while true do
+ pos = strfind (str, "%S", pos)
+ if not pos then return nil end
+ local sub2 = strsub (str, pos, pos + 1)
+ if sub2 == "\239\187" and strsub (str, pos + 2, pos + 2) == "\191" then
+ -- UTF-8 Byte Order Mark
+ pos = pos + 3
+ elseif sub2 == "//" then
+ pos = strfind (str, "[\n\r]", pos + 2)
+ if not pos then return nil end
+ elseif sub2 == "/*" then
+ pos = strfind (str, "*/", pos + 2)
+ if not pos then return nil end
+ pos = pos + 2
+ else
+ return pos
+ end
+ end
+end
+
+local escapechars = {
+ ["\""] = "\"", ["\\"] = "\\", ["/"] = "/", ["b"] = "\b", ["f"] = "\f",
+ ["n"] = "\n", ["r"] = "\r", ["t"] = "\t"
+}
+
+local function unichar (value)
+ if value < 0 then
+ return nil
+ elseif value <= 0x007f then
+ return strchar (value)
+ elseif value <= 0x07ff then
+ return strchar (0xc0 + floor(value/0x40),
+ 0x80 + (floor(value) % 0x40))
+ elseif value <= 0xffff then
+ return strchar (0xe0 + floor(value/0x1000),
+ 0x80 + (floor(value/0x40) % 0x40),
+ 0x80 + (floor(value) % 0x40))
+ elseif value <= 0x10ffff then
+ return strchar (0xf0 + floor(value/0x40000),
+ 0x80 + (floor(value/0x1000) % 0x40),
+ 0x80 + (floor(value/0x40) % 0x40),
+ 0x80 + (floor(value) % 0x40))
+ else
+ return nil
+ end
+end
+
+local function scanstring (str, pos)
+ local lastpos = pos + 1
+ local buffer, n = {}, 0
+ while true do
+ local nextpos = strfind (str, "[\"\\]", lastpos)
+ if not nextpos then
+ return unterminated (str, "string", pos)
+ end
+ if nextpos > lastpos then
+ n = n + 1
+ buffer[n] = strsub (str, lastpos, nextpos - 1)
+ end
+ if strsub (str, nextpos, nextpos) == "\"" then
+ lastpos = nextpos + 1
+ break
+ else
+ local escchar = strsub (str, nextpos + 1, nextpos + 1)
+ local value
+ if escchar == "u" then
+ value = tonumber (strsub (str, nextpos + 2, nextpos + 5), 16)
+ if value then
+ local value2
+ if 0xD800 <= value and value <= 0xDBff then
+ -- we have the high surrogate of UTF-16. Check if there is a
+ -- low surrogate escaped nearby to combine them.
+ if strsub (str, nextpos + 6, nextpos + 7) == "\\u" then
+ value2 = tonumber (strsub (str, nextpos + 8, nextpos + 11), 16)
+ if value2 and 0xDC00 <= value2 and value2 <= 0xDFFF then
+ value = (value - 0xD800) * 0x400 + (value2 - 0xDC00) + 0x10000
+ else
+ value2 = nil -- in case it was out of range for a low surrogate
+ end
+ end
+ end
+ value = value and unichar (value)
+ if value then
+ if value2 then
+ lastpos = nextpos + 12
+ else
+ lastpos = nextpos + 6
+ end
+ end
+ end
+ end
+ if not value then
+ value = escapechars[escchar] or escchar
+ lastpos = nextpos + 2
+ end
+ n = n + 1
+ buffer[n] = value
+ end
+ end
+ if n == 1 then
+ return buffer[1], lastpos
+ elseif n > 1 then
+ return concat (buffer), lastpos
+ else
+ return "", lastpos
+ end
+end
+
+local scanvalue -- forward declaration
+
+local function scantable (what, closechar, str, startpos, nullval, objectmeta, arraymeta)
+ local tbl, n = {}, 0
+ local pos = startpos + 1
+ if what == 'object' then
+ setmetatable (tbl, objectmeta)
+ else
+ setmetatable (tbl, arraymeta)
+ end
+ while true do
+ pos = scanwhite (str, pos)
+ if not pos then return unterminated (str, what, startpos) end
+ local char = strsub (str, pos, pos)
+ if char == closechar then
+ return tbl, pos + 1
+ end
+ local val1, err
+ val1, pos, err = scanvalue (str, pos, nullval, objectmeta, arraymeta)
+ if err then return nil, pos, err end
+ pos = scanwhite (str, pos)
+ if not pos then return unterminated (str, what, startpos) end
+ char = strsub (str, pos, pos)
+ if char == ":" then
+ if val1 == nil then
+ return nil, pos, "cannot use nil as table index (at " .. loc (str, pos) .. ")"
+ end
+ pos = scanwhite (str, pos + 1)
+ if not pos then return unterminated (str, what, startpos) end
+ local val2
+ val2, pos, err = scanvalue (str, pos, nullval, objectmeta, arraymeta)
+ if err then return nil, pos, err end
+ tbl[val1] = val2
+ pos = scanwhite (str, pos)
+ if not pos then return unterminated (str, what, startpos) end
+ char = strsub (str, pos, pos)
+ else
+ n = n + 1
+ tbl[n] = val1
+ end
+ if char == "," then
+ pos = pos + 1
+ end
+ end
+end
+
+scanvalue = function (str, pos, nullval, objectmeta, arraymeta)
+ pos = pos or 1
+ pos = scanwhite (str, pos)
+ if not pos then
+ return nil, strlen (str) + 1, "no valid JSON value (reached the end)"
+ end
+ local char = strsub (str, pos, pos)
+ if char == "{" then
+ return scantable ('object', "}", str, pos, nullval, objectmeta, arraymeta)
+ elseif char == "[" then
+ return scantable ('array', "]", str, pos, nullval, objectmeta, arraymeta)
+ elseif char == "\"" then
+ return scanstring (str, pos)
+ else
+ local pstart, pend = strfind (str, "^%-?[%d%.]+[eE]?[%+%-]?%d*", pos)
+ if pstart then
+ local number = str2num (strsub (str, pstart, pend))
+ if number then
+ return number, pend + 1
+ end
+ end
+ pstart, pend = strfind (str, "^%a%w*", pos)
+ if pstart then
+ local name = strsub (str, pstart, pend)
+ if name == "true" then
+ return true, pend + 1
+ elseif name == "false" then
+ return false, pend + 1
+ elseif name == "null" then
+ return nullval, pend + 1
+ end
+ end
+ return nil, pos, "no valid JSON value at " .. loc (str, pos)
+ end
+end
+
+local function optionalmetatables(...)
+ if select("#", ...) > 0 then
+ return ...
+ else
+ return {__jsontype = 'object'}, {__jsontype = 'array'}
+ end
+end
+
+function json.decode (str, pos, nullval, ...)
+ local objectmeta, arraymeta = optionalmetatables(...)
+ return scanvalue (str, pos, nullval, objectmeta, arraymeta)
+end
+
+function json.use_lpeg ()
+ local g = require ("lpeg")
+
+ if g.version() == "0.11" then
+ error "due to a bug in LPeg 0.11, it cannot be used for JSON matching"
+ end
+
+ local pegmatch = g.match
+ local P, S, R = g.P, g.S, g.R
+
+ local function ErrorCall (str, pos, msg, state)
+ if not state.msg then
+ state.msg = msg .. " at " .. loc (str, pos)
+ state.pos = pos
+ end
+ return false
+ end
+
+ local function Err (msg)
+ return g.Cmt (g.Cc (msg) * g.Carg (2), ErrorCall)
+ end
+
+ local function ErrorUnterminatedCall (str, pos, what, state)
+ return ErrorCall (str, pos - 1, "unterminated " .. what, state)
+ end
+
+ local SingleLineComment = P"//" * (1 - S"\n\r")^0
+ local MultiLineComment = P"/*" * (1 - P"*/")^0 * P"*/"
+ local Space = (S" \n\r\t" + P"\239\187\191" + SingleLineComment + MultiLineComment)^0
+
+ local function ErrUnterminated (what)
+ return g.Cmt (g.Cc (what) * g.Carg (2), ErrorUnterminatedCall)
+ end
+
+ local PlainChar = 1 - S"\"\\\n\r"
+ local EscapeSequence = (P"\\" * g.C (S"\"\\/bfnrt" + Err "unsupported escape sequence")) / escapechars
+ local HexDigit = R("09", "af", "AF")
+ local function UTF16Surrogate (_match, _pos, high, low)
+ high, low = tonumber (high, 16), tonumber (low, 16)
+ if 0xD800 <= high and high <= 0xDBff and 0xDC00 <= low and low <= 0xDFFF then
+ return true, unichar ((high - 0xD800) * 0x400 + (low - 0xDC00) + 0x10000)
+ else
+ return false
+ end
+ end
+ local function UTF16BMP (hex)
+ return unichar (tonumber (hex, 16))
+ end
+ local U16Sequence = (P"\\u" * g.C (HexDigit * HexDigit * HexDigit * HexDigit))
+ local UnicodeEscape = g.Cmt (U16Sequence * U16Sequence, UTF16Surrogate) + U16Sequence/UTF16BMP
+ local Char = UnicodeEscape + EscapeSequence + PlainChar
+ local String = P"\"" * (g.Cs (Char ^ 0) * P"\"" + ErrUnterminated "string")
+ local Integer = P"-"^(-1) * (P"0" + (R"19" * R"09"^0))
+ local Fractal = P"." * R"09"^0
+ local Exponent = (S"eE") * (S"+-")^(-1) * R"09"^1
+ local Number = (Integer * Fractal^(-1) * Exponent^(-1))/str2num
+ local Constant = P"true" * g.Cc (true) + P"false" * g.Cc (false) + P"null" * g.Carg (1)
+ local SimpleValue = Number + String + Constant
+ local ArrayContent, ObjectContent
+
+ -- The functions parsearray and parseobject parse only a single value/pair
+ -- at a time and store them directly to avoid hitting the LPeg limits.
+ local function parsearray (str, pos, nullval, state)
+ local obj, cont
+ local start = pos
+ local npos
+ local t, nt = {}, 0
+ repeat
+ obj, cont, npos = pegmatch (ArrayContent, str, pos, nullval, state)
+ if cont == 'end' then
+ return ErrorUnterminatedCall (str, start, "array", state)
+ end
+ pos = npos
+ if cont == 'cont' or cont == 'last' then
+ nt = nt + 1
+ t[nt] = obj
+ end
+ until cont ~= 'cont'
+ return pos, setmetatable (t, state.arraymeta)
+ end
+
+ local function parseobject (str, pos, nullval, state)
+ local obj, key, cont
+ local start = pos
+ local npos
+ local t = {}
+ repeat
+ key, obj, cont, npos = pegmatch (ObjectContent, str, pos, nullval, state)
+ if cont == 'end' then
+ return ErrorUnterminatedCall (str, start, "object", state)
+ end
+ pos = npos
+ if cont == 'cont' or cont == 'last' then
+ t[key] = obj
+ end
+ until cont ~= 'cont'
+ return pos, setmetatable (t, state.objectmeta)
+ end
+
+ local Array = P"[" * g.Cmt (g.Carg(1) * g.Carg(2), parsearray)
+ local Object = P"{" * g.Cmt (g.Carg(1) * g.Carg(2), parseobject)
+ local Value = Space * (Array + Object + SimpleValue)
+ local ExpectedValue = Value + Space * Err "value expected"
+ local ExpectedKey = String + Err "key expected"
+ local End = P(-1) * g.Cc'end'
+ local ErrInvalid = Err "invalid JSON"
+ ArrayContent = (Value * Space * (P"," * g.Cc'cont' + P"]" * g.Cc'last'+ End + ErrInvalid) + g.Cc(nil) * (P"]" * g.Cc'empty' + End + ErrInvalid)) * g.Cp()
+ local Pair = g.Cg (Space * ExpectedKey * Space * (P":" + Err "colon expected") * ExpectedValue)
+ ObjectContent = (g.Cc(nil) * g.Cc(nil) * P"}" * g.Cc'empty' + End + (Pair * Space * (P"," * g.Cc'cont' + P"}" * g.Cc'last' + End + ErrInvalid) + ErrInvalid)) * g.Cp()
+ local DecodeValue = ExpectedValue * g.Cp ()
+
+ jsonlpeg.version = json.version
+ jsonlpeg.encode = json.encode
+ jsonlpeg.null = json.null
+ jsonlpeg.quotestring = json.quotestring
+ jsonlpeg.addnewline = json.addnewline
+ jsonlpeg.encodeexception = json.encodeexception
+ jsonlpeg.using_lpeg = true
+
+ function jsonlpeg.decode (str, pos, nullval, ...)
+ local state = {}
+ state.objectmeta, state.arraymeta = optionalmetatables(...)
+ local obj, retpos = pegmatch (DecodeValue, str, pos, nullval, state)
+ if state.msg then
+ return nil, state.pos, state.msg
+ else
+ return obj, retpos
+ end
+ end
+
+ -- cache result of this function:
+ json.use_lpeg = function () return jsonlpeg end
+ jsonlpeg.use_lpeg = json.use_lpeg
+
+ return jsonlpeg
+end
+
+if always_use_lpeg then
+ return json.use_lpeg()
+end
+
+return json
+
diff --git a/rsc/config/awesome/lain/util/init.lua b/rsc/config/awesome/lain/util/init.lua
new file mode 100644
index 0000000..5ae0523
--- /dev/null
+++ b/rsc/config/awesome/lain/util/init.lua
@@ -0,0 +1,190 @@
+--[[
+
+ Lain
+ Layouts, widgets and utilities for Awesome WM
+
+ Utilities section
+
+ Licensed under GNU General Public License v2
+ * (c) 2013, Luca CPZ
+ * (c) 2010-2012, Peter Hofmann
+
+--]]
+
+local awful = require("awful")
+local sqrt = math.sqrt
+local pairs = pairs
+local client = client
+local tonumber = tonumber
+local wrequire = require("lain.helpers").wrequire
+local setmetatable = setmetatable
+
+-- Lain utilities submodule
+-- lain.util
+local util = { _NAME = "lain.util" }
+
+-- Like awful.menu.clients, but only show clients of currently selected tags
+function util.menu_clients_current_tags(menu, args)
+ -- List of currently selected tags.
+ local cls_tags = awful.screen.focused().selected_tags
+
+ if cls_tags == nil then return nil end
+
+ -- Final list of menu items.
+ local cls_t = {}
+
+ -- For each selected tag get all clients of that tag and add them to
+ -- the menu. A click on a menu item will raise that client.
+ for i = 1,#cls_tags do
+ local t = cls_tags[i]
+ local cls = t:clients()
+
+ for _, c in pairs(cls) do
+ cls_t[#cls_t + 1] = { awful.util.escape(c.name) or "",
+ function ()
+ c.minimized = false
+ client.focus = c
+ c:raise()
+ end,
+ c.icon }
+ end
+ end
+
+ -- No clients? Then quit.
+ if #cls_t <= 0 then return nil end
+
+ -- menu may contain some predefined values, otherwise start with a
+ -- fresh menu.
+ if not menu then menu = {} end
+
+ -- Set the list of items and show the menu.
+ menu.items = cls_t
+ local m = awful.menu(menu)
+ m:show(args)
+
+ return m
+end
+
+-- Magnify a client: set it to "float" and resize it.
+function util.magnify_client(c, width_f, height_f)
+ if c and not c.floating then
+ util.magnified_client = c
+ util.mc(c, width_f, height_f)
+ else
+ util.magnified_client = nil
+ c.floating = false
+ end
+end
+
+-- https://github.com/lcpz/lain/issues/195
+function util.mc(c, width_f, height_f)
+ c = c or util.magnified_client
+ if not c then return end
+
+ c.floating = true
+ local s = awful.screen.focused()
+ local mg = s.workarea
+ local g = {}
+ local mwfact = width_f or s.selected_tag.master_width_factor or 0.5
+ g.width = sqrt(mwfact) * mg.width
+ g.height = sqrt(height_f or mwfact) * mg.height
+ g.x = mg.x + (mg.width - g.width) / 2
+ g.y = mg.y + (mg.height - g.height) / 2
+
+ if c then c:geometry(g) end -- if c is still a valid object
+end
+
+-- Non-empty tag browsing
+-- direction in {-1, 1} <-> {previous, next} non-empty tag
+function util.tag_view_nonempty(direction,sc)
+ direction = direction or 1
+ local s = sc or awful.screen.focused()
+ local tags = s.tags
+ local sel = s.selected_tag
+
+ local i = sel.index
+ repeat
+ i = i + direction
+
+ -- Wrap around when we reach one of the bounds
+ if i > #tags then
+ i = i - #tags
+ end
+ if i < 1 then
+ i = i + #tags
+ end
+
+ local t = tags[i]
+
+ -- Stop when we get back to where we started
+ if t == sel then
+ break
+ end
+
+ -- If it's The One, view it.
+ if #t:clients() > 0 then
+ t:view_only()
+ return
+ end
+ until false
+end
+
+-- {{{ Dynamic tagging
+
+-- Add a new tag
+function util.add_tag(layout)
+ awful.prompt.run {
+ prompt = "New tag name: ",
+ textbox = awful.screen.focused().mypromptbox.widget,
+ exe_callback = function(name)
+ if not name or #name == 0 then return end
+ awful.tag.add(name, { screen = awful.screen.focused(), layout = layout or awful.layout.suit.tile }):view_only()
+ end
+ }
+end
+
+-- Rename current tag
+function util.rename_tag()
+ awful.prompt.run {
+ prompt = "Rename tag: ",
+ textbox = awful.screen.focused().mypromptbox.widget,
+ exe_callback = function(new_name)
+ if not new_name or #new_name == 0 then return end
+ local t = awful.screen.focused().selected_tag
+ if t then
+ t.name = new_name
+ end
+ end
+ }
+end
+
+-- Move current tag
+-- pos in {-1, 1} <-> {previous, next} tag position
+function util.move_tag(pos)
+ local tag = awful.screen.focused().selected_tag
+ if tonumber(pos) <= -1 then
+ awful.tag.move(tag.index - 1, tag)
+ else
+ awful.tag.move(tag.index + 1, tag)
+ end
+end
+
+-- Delete current tag
+-- Any rule set on the tag shall be broken
+function util.delete_tag()
+ local t = awful.screen.focused().selected_tag
+ if not t then return end
+ t:delete()
+end
+
+-- }}}
+
+-- On the fly useless gaps change
+function util.useless_gaps_resize(thatmuch, s, t)
+ local scr = s or awful.screen.focused()
+ local tag = t or scr.selected_tag
+ tag.gap = tag.gap + tonumber(thatmuch)
+ awful.layout.arrange(scr)
+end
+
+return setmetatable(util, { __index = wrequire })
diff --git a/rsc/config/awesome/lain/util/markup.lua b/rsc/config/awesome/lain/util/markup.lua
new file mode 100644
index 0000000..63f9486
--- /dev/null
+++ b/rsc/config/awesome/lain/util/markup.lua
@@ -0,0 +1,66 @@
+--[[
+
+ Licensed under MIT License
+ * (c) 2013, Luca CPZ
+ * (c) 2009, Uli Schlachter
+ * (c) 2009, Majic
+
+--]]
+
+local format = string.format
+local setmetatable = setmetatable
+
+-- Lain markup util submodule
+-- lain.util.markup
+local markup = { fg = {}, bg = {} }
+
+-- Convenience tags
+function markup.bold(text) return format("%s", text) end
+function markup.italic(text) return format("%s", text) end
+function markup.strike(text) return format("%s", text) end
+function markup.underline(text) return format("%s", text) end
+function markup.monospace(text) return format("%s", text) end
+function markup.big(text) return format("%s", text) end
+function markup.small(text) return format("%s", text) end
+
+-- Set the font
+function markup.font(font, text)
+ return format("%s", font, text)
+end
+
+-- Set the foreground
+function markup.fg.color(color, text)
+ return format("%s", color, text)
+end
+
+-- Set the background
+function markup.bg.color(color, text)
+ return format("%s", color, text)
+end
+
+-- Set foreground and background
+function markup.color(fg, bg, text)
+ return format("%s", fg, bg, text)
+end
+
+-- Set font and foreground
+function markup.fontfg(font, fg, text)
+ return format("%s", font, fg, text)
+end
+
+-- Set font and background
+function markup.fontbg(font, bg, text)
+ return format("%s", font, bg, text)
+end
+
+-- Set font, foreground and background
+function markup.fontcolor(font, fg, bg, text)
+ return format("%s", font, fg, bg, text)
+end
+
+-- link markup.{fg,bg}(...) calls to markup.{fg,bg}.color(...)
+setmetatable(markup.fg, { __call = function(_, ...) return markup.fg.color(...) end })
+setmetatable(markup.bg, { __call = function(_, ...) return markup.bg.color(...) end })
+
+-- link markup(...) calls to markup.fg.color(...)
+return setmetatable(markup, { __call = function(_, ...) return markup.fg.color(...) end })
diff --git a/rsc/config/awesome/lain/util/menu_iterator.lua b/rsc/config/awesome/lain/util/menu_iterator.lua
new file mode 100644
index 0000000..d457473
--- /dev/null
+++ b/rsc/config/awesome/lain/util/menu_iterator.lua
@@ -0,0 +1,144 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2017, Simon Désaulniers
+ * (c) 2017, Uli Schlachter
+ * (c) 2017, Jeferson Siqueira
+
+--]]
+
+-- Menu iterator with Naughty notifications
+-- lain.util.menu_iterator
+
+local naughty = require("naughty")
+local helpers = require("lain.helpers")
+local atable = require("awful.util").table
+local assert = assert
+local pairs = pairs
+local tconcat = table.concat
+local unpack = unpack or table.unpack -- lua 5.1 retro-compatibility
+
+local state = { cid = nil }
+
+local function naughty_destroy_callback(reason)
+ local closed = naughty.notificationClosedReason
+ if reason == closed.expired or reason == closed.dismissedByUser then
+ local actions = state.index and state.menu[state.index - 1][2]
+ if actions then
+ for _,action in pairs(actions) do
+ -- don't try to call nil callbacks
+ if action then action() end
+ end
+ state.index = nil
+ end
+ end
+end
+
+-- Iterates over a menu.
+-- After the timeout, callbacks associated to the last visited choice are
+-- executed. Inputs:
+-- * menu: a list of {label, {callbacks}} pairs
+-- * timeout: time to wait before confirming the menu selection
+-- * icon: icon to display in the notification of the chosen label
+local function iterate(menu, timeout, icon)
+ timeout = timeout or 4 -- default timeout for each menu entry
+ icon = icon or nil -- icon to display on the menu
+
+ -- 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
+ local next = state.menu[state.index]
+ state.index = state.index + 1
+
+ if not next then
+ label = "Cancel"
+ state.index = nil
+ else
+ label, _ = unpack(next)
+ end
+
+ state.cid = naughty.notify({
+ text = label,
+ icon = icon,
+ timeout = timeout,
+ screen = mouse.screen,
+ replaces_id = state.cid,
+ destroy = naughty_destroy_callback
+ }).id
+end
+
+-- Generates a menu compatible with the first argument of `iterate` function and
+-- suitable for the following cases:
+-- * all possible choices individually (partition of singletons);
+-- * all possible subsets of the set of choices (powerset).
+--
+-- Inputs:
+-- * args: an array containing the following members:
+-- * choices: Array of choices (string) on which the menu will be
+-- generated.
+-- * name: Displayed name of the menu (in the form "name: choices").
+-- * selected_cb: Callback to execute for each selected choice. Takes
+-- the choice as a string argument. Can be `nil` (no action
+-- to execute).
+-- * rejected_cb: Callback to execute for each rejected choice (possible
+-- choices which are not selected). Takes the choice as a
+-- string argument. Can be `nil` (no action to execute).
+-- * extra_choices: An array of extra { choice_str, callback_fun } pairs to be
+-- added to the menu. Each callback_fun can be `nil`.
+-- * combination: The combination of choices to generate. Possible values:
+-- "powerset" and "single" (default).
+-- Output:
+-- * m: menu to be iterated over.
+local function menu(args)
+ local choices = assert(args.choices or args[1])
+ local name = assert(args.name or args[2])
+ local selected_cb = args.selected_cb
+ local rejected_cb = args.rejected_cb
+ local extra_choices = args.extra_choices or {}
+
+ local ch_combinations = args.combination == "powerset" and helpers.powerset(choices) or helpers.trivial_partition_set(choices)
+
+ for _, c in pairs(extra_choices) do
+ ch_combinations = atable.join(ch_combinations, {{c[1]}})
+ end
+
+ local m = {} -- the menu
+
+ for _,c in pairs(ch_combinations) do
+ if #c > 0 then
+ local cbs = {}
+
+ -- selected choices
+ for _,ch in pairs(c) do
+ if atable.hasitem(choices, ch) then
+ cbs[#cbs + 1] = selected_cb and function() selected_cb(ch) end or nil
+ end
+ end
+
+ -- rejected choices
+ for _,ch in pairs(choices) do
+ if not atable.hasitem(c, ch) and atable.hasitem(choices, ch) then
+ cbs[#cbs + 1] = rejected_cb and function() rejected_cb(ch) end or nil
+ end
+ end
+
+ -- add user extra choices (like the choice "None" for example)
+ for _,x in pairs(extra_choices) do
+ if x[1] == c[1] then
+ cbs[#cbs + 1] = x[2]
+ end
+ end
+
+ m[#m + 1] = { name .. ": " .. tconcat(c, " + "), cbs }
+ end
+ end
+
+ return m
+end
+
+return { iterate = iterate, menu = menu }
diff --git a/rsc/config/awesome/lain/util/quake.lua b/rsc/config/awesome/lain/util/quake.lua
new file mode 100644
index 0000000..8bc68a7
--- /dev/null
+++ b/rsc/config/awesome/lain/util/quake.lua
@@ -0,0 +1,179 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2016, Luca CPZ
+ * (c) 2015, unknown
+
+--]]
+
+local awful = require("awful")
+local capi = { client = client }
+local math = math
+local string = string
+local pairs = pairs
+local screen = screen
+local setmetatable = setmetatable
+
+-- Quake-like Dropdown application spawn
+local quake = {}
+
+-- If you have a rule like "awful.client.setslave" for your terminals,
+-- ensure you use an exception for QuakeDD. Otherwise, you may
+-- run into problems with focus.
+
+function quake:display()
+ if self.followtag then self.screen = awful.screen.focused() end
+
+ -- First, we locate the client
+ local client = nil
+ local i = 0
+ for c in awful.client.iterate(function (c)
+ -- c.name may be changed!
+ return c.instance == self.name
+ end)
+ do
+ i = i + 1
+ if i == 1 then
+ client = c
+ else
+ -- Additional matching clients, let's remove the sticky bit
+ -- which may persist between awesome restarts. We don't close
+ -- them as they may be valuable. They will just turn into
+ -- normal clients.
+ c.sticky = false
+ c.ontop = false
+ c.above = false
+ end
+ end
+
+ if not client and not self.visible then return end
+
+ if not client then
+ -- The client does not exist, we spawn it
+ local cmd = string.format("%s %s %s", self.app,
+ string.format(self.argname, self.name), self.extra)
+ awful.spawn(cmd, { tag = self.screen.selected_tag })
+ return
+ end
+
+ -- Set geometry
+ client.floating = true
+ client.border_width = self.border
+ client.size_hints_honor = false
+ local maximized = client.maximized
+ local fullscreen = client.fullscreen
+ client:geometry(self.geometry[self.screen.index] or self:compute_size())
+
+ -- Set not sticky and on top
+ client.sticky = false
+ client.ontop = true
+ client.above = true
+ client.skip_taskbar = true
+
+ -- Additional user settings
+ if self.settings then self.settings(client) end
+
+ -- Toggle display
+ if self.visible then
+ client.hidden = false
+ client.maximized = self.maximized
+ client.fullscreen = self.fullscreen
+ client:raise()
+ self.last_tag = self.screen.selected_tag
+ client:tags({self.screen.selected_tag})
+ capi.client.focus = client
+ else
+ self.maximized = maximized
+ self.fullscreen = fullscreen
+ client.maximized = false
+ client.fullscreen = false
+ client.hidden = true
+ local ctags = client:tags()
+ for j, _ in pairs(ctags) do
+ ctags[j] = nil
+ end
+ client:tags(ctags)
+ end
+
+ return client
+end
+
+function quake:compute_size()
+ -- skip if we already have a geometry for this screen
+ if not self.geometry[self.screen.index] then
+ local geom
+ if not self.overlap then
+ geom = screen[self.screen.index].workarea
+ else
+ geom = screen[self.screen.index].geometry
+ end
+ local width, height = self.width, self.height
+ if width <= 1 then width = math.floor(geom.width * width) - 2 * self.border end
+ if height <= 1 then height = math.floor(geom.height * height) end
+ local x, y
+ if self.horiz == "left" then x = geom.x
+ elseif self.horiz == "right" then x = geom.width + geom.x - width
+ else x = geom.x + (geom.width - width)/2 end
+ if self.vert == "top" then y = geom.y
+ elseif self.vert == "bottom" then y = geom.height + geom.y - height
+ else y = geom.y + (geom.height - height)/2 end
+ self.geometry[self.screen.index] = { x = x, y = y, width = width, height = height }
+ end
+ return self.geometry[self.screen.index]
+end
+
+function quake:toggle()
+ if self.followtag then self.screen = awful.screen.focused() end
+ local current_tag = self.screen.selected_tag
+ if current_tag and self.last_tag ~= current_tag and self.visible then
+ local c=self:display()
+ if c then
+ c:move_to_tag(current_tag)
+ end
+ else
+ self.visible = not self.visible
+ self:display()
+ end
+end
+
+function quake.new(conf)
+ conf = conf or {}
+
+ conf.app = conf.app or "xterm" -- application to spawn
+ conf.name = conf.name or "QuakeDD" -- window name
+ conf.argname = conf.argname or "-name %s" -- how to specify window name
+ conf.extra = conf.extra or "" -- extra arguments
+ conf.border = conf.border or 1 -- client border width
+ conf.visible = conf.visible or false -- initially not visible
+ conf.followtag = conf.followtag or false -- spawn on currently focused screen
+ conf.overlap = conf.overlap or false -- overlap wibox
+ conf.screen = conf.screen or awful.screen.focused()
+ conf.settings = conf.settings
+
+ -- If width or height <= 1 this is a proportion of the workspace
+ conf.height = conf.height or 0.25 -- height
+ conf.width = conf.width or 1 -- width
+ conf.vert = conf.vert or "top" -- top, bottom or center
+ conf.horiz = conf.horiz or "left" -- left, right or center
+ conf.geometry = {} -- internal use
+
+ conf.maximized = false
+ conf.fullscreen = false
+
+ local dropdown = setmetatable(conf, { __index = quake })
+
+ capi.client.connect_signal("manage", function(c)
+ if c.instance == dropdown.name and c.screen == dropdown.screen then
+ dropdown:display()
+ end
+ end)
+ capi.client.connect_signal("unmanage", function(c)
+ if c.instance == dropdown.name and c.screen == dropdown.screen then
+ dropdown.visible = false
+ end
+ end)
+
+ return dropdown
+end
+
+return setmetatable(quake, { __call = function(_, ...) return quake.new(...) end })
diff --git a/rsc/config/awesome/lain/util/separators.lua b/rsc/config/awesome/lain/util/separators.lua
new file mode 100644
index 0000000..04402bb
--- /dev/null
+++ b/rsc/config/awesome/lain/util/separators.lua
@@ -0,0 +1,118 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2015, Luca CPZ
+ * (c) 2015, plotnikovanton
+
+--]]
+
+local wibox = require("wibox")
+local gears = require("gears")
+local beautiful = require("beautiful")
+
+-- Lain Cairo separators util submodule
+-- lain.util.separators
+local separators = { height = beautiful.separators_height or 0, width = beautiful.separators_width or 9 }
+
+-- [[ Arrow
+
+-- Right
+function separators.arrow_right(col1, col2)
+ local widget = wibox.widget.base.make_widget()
+ widget.col1 = col1
+ widget.col2 = col2
+
+ widget.fit = function(_, _, _)
+ return separators.width, separators.height
+ end
+
+ widget.update = function(_, _)
+ widget.col1 = col1
+ widget.col2 = col2
+ widget:emit_signal("widget::redraw_needed")
+ end
+
+ widget.draw = function(_, _, cr, width, height)
+ if widget.col2 ~= "alpha" then
+ cr:set_source_rgba(gears.color.parse_color(widget.col2))
+ cr:new_path()
+ cr:move_to(0, 0)
+ cr:line_to(width, height/2)
+ cr:line_to(width, 0)
+ cr:close_path()
+ cr:fill()
+
+ cr:new_path()
+ cr:move_to(0, height)
+ cr:line_to(width, height/2)
+ cr:line_to(width, height)
+ cr:close_path()
+ cr:fill()
+ end
+
+ if widget.col1 ~= "alpha" then
+ cr:set_source_rgba(gears.color.parse_color(widget.col1))
+ cr:new_path()
+ cr:move_to(0, 0)
+ cr:line_to(width, height/2)
+ cr:line_to(0, height)
+ cr:close_path()
+ cr:fill()
+ end
+ end
+
+ return widget
+end
+
+-- Left
+function separators.arrow_left(col1, col2)
+ local widget = wibox.widget.base.make_widget()
+ widget.col1 = col1
+ widget.col2 = col2
+
+ widget.fit = function(_, _, _)
+ return separators.width, separators.height
+ end
+
+ widget.update = function(c1, c2)
+ widget.col1 = c1
+ widget.col2 = c2
+ widget:emit_signal("widget::redraw_needed")
+ end
+
+ widget.draw = function(_, _, cr, width, height)
+ if widget.col1 ~= "alpha" then
+ cr:set_source_rgba(gears.color.parse_color(widget.col1))
+ cr:new_path()
+ cr:move_to(width, 0)
+ cr:line_to(0, height/2)
+ cr:line_to(0, 0)
+ cr:close_path()
+ cr:fill()
+
+ cr:new_path()
+ cr:move_to(width, height)
+ cr:line_to(0, height/2)
+ cr:line_to(0, height)
+ cr:close_path()
+ cr:fill()
+ end
+
+ if widget.col2 ~= "alpha" then
+ cr:new_path()
+ cr:move_to(width, 0)
+ cr:line_to(0, height/2)
+ cr:line_to(width, height)
+ cr:close_path()
+
+ cr:set_source_rgba(gears.color.parse_color(widget.col2))
+ cr:fill()
+ end
+ end
+
+ return widget
+end
+
+-- ]]
+
+return separators
diff --git a/rsc/config/awesome/lain/widget/alsa.lua b/rsc/config/awesome/lain/widget/alsa.lua
new file mode 100644
index 0000000..202dc98
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/alsa.lua
@@ -0,0 +1,54 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2013, Luca CPZ
+ * (c) 2010, Adrian C.
+
+--]]
+
+local helpers = require("lain.helpers")
+local shell = require("awful.util").shell
+local wibox = require("wibox")
+local string = string
+
+-- ALSA volume
+-- lain.widget.alsa
+
+local function factory(args)
+ args = args or {}
+ local alsa = { widget = args.widget or wibox.widget.textbox() }
+ local timeout = args.timeout or 5
+ local settings = args.settings or function() end
+
+ alsa.cmd = args.cmd or "amixer"
+ alsa.channel = args.channel or "Master"
+ alsa.togglechannel = args.togglechannel
+
+ local format_cmd = string.format("%s get %s", alsa.cmd, alsa.channel)
+
+ if alsa.togglechannel then
+ format_cmd = { shell, "-c", string.format("%s get %s; %s get %s",
+ alsa.cmd, alsa.channel, alsa.cmd, alsa.togglechannel) }
+ end
+
+ alsa.last = {}
+
+ function alsa.update()
+ helpers.async(format_cmd, function(mixer)
+ local l,s = string.match(mixer, "([%d]+)%%.*%[([%l]*)")
+ l = tonumber(l)
+ if alsa.last.level ~= l or alsa.last.status ~= s then
+ volume_now = { level = l, status = s }
+ widget = alsa.widget
+ settings()
+ alsa.last = volume_now
+ end
+ end)
+ end
+
+ helpers.newtimer(string.format("alsa-%s-%s", alsa.cmd, alsa.channel), timeout, alsa.update)
+
+ return alsa
+end
+
+return factory
diff --git a/rsc/config/awesome/lain/widget/alsabar.lua b/rsc/config/awesome/lain/widget/alsabar.lua
new file mode 100644
index 0000000..8e8cd3a
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/alsabar.lua
@@ -0,0 +1,166 @@
+--[[
+
+ 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
+
+-- ALSA volume bar
+-- lain.widget.alsabar
+
+local function factory(args)
+ local alsabar = {
+ colors = {
+ background = "#000000",
+ mute = "#EB8F8F",
+ unmute = "#A4CE8A"
+ },
+
+ _current_level = 0,
+ _playback = "off"
+ }
+
+ 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 ticks = args.ticks or false
+ local ticks_size = args.ticks_size or 7
+ local tick = args.tick or "|"
+ local tick_pre = args.tick_pre or "["
+ local tick_post = args.tick_post or "]"
+ local tick_none = args.tick_none or " "
+
+ alsabar.cmd = args.cmd or "amixer"
+ alsabar.channel = args.channel or "Master"
+ alsabar.togglechannel = args.togglechannel
+ alsabar.colors = args.colors or alsabar.colors
+ alsabar.followtag = args.followtag or false
+ alsabar.notification_preset = args.notification_preset
+
+ if not alsabar.notification_preset then
+ alsabar.notification_preset = { font = "Monospace 10" }
+ end
+
+ local format_cmd = string.format("%s get %s", alsabar.cmd, alsabar.channel)
+
+ if alsabar.togglechannel then
+ format_cmd = { awful.util.shell, "-c", string.format("%s get %s; %s get %s",
+ alsabar.cmd, alsabar.channel, alsabar.cmd, alsabar.togglechannel) }
+ end
+
+ alsabar.bar = wibox.widget {
+ color = alsabar.colors.unmute,
+ background_color = alsabar.colors.background,
+ forced_height = height,
+ forced_width = width,
+ margins = margins,
+ paddings = margins,
+ ticks = ticks,
+ ticks_size = ticks_size,
+ widget = wibox.widget.progressbar
+ }
+
+ alsabar.tooltip = awful.tooltip({ objects = { alsabar.bar } })
+
+ function alsabar.update(callback)
+ helpers.async(format_cmd, function(mixer)
+ local vol, playback = string.match(mixer, "([%d]+)%%.*%[([%l]*)")
+
+ if not vol or not playback then return end
+
+ if vol ~= alsabar._current_level or playback ~= alsabar._playback then
+ alsabar._current_level = tonumber(vol)
+ alsabar.bar:set_value(alsabar._current_level / 100)
+ if alsabar._current_level == 0 or playback == "off" then
+ alsabar._playback = playback
+ alsabar.tooltip:set_text("[Muted]")
+ alsabar.bar.color = alsabar.colors.mute
+ else
+ alsabar._playback = "on"
+ alsabar.tooltip:set_text(string.format("%s: %s", alsabar.channel, vol))
+ alsabar.bar.color = alsabar.colors.unmute
+ end
+
+ volume_now = {
+ level = alsabar._current_level,
+ status = alsabar._playback
+ }
+
+ settings()
+
+ if type(callback) == "function" then callback() end
+ end
+ end)
+ end
+
+ function alsabar.notify()
+ alsabar.update(function()
+ local preset = alsabar.notification_preset
+
+ preset.title = string.format("%s - %s%%", alsabar.channel, alsabar._current_level)
+
+ if alsabar._playback == "off" then
+ preset.title = preset.title .. " Muted"
+ end
+
+ -- tot is the maximum number of ticks to display in the notification
+ local tot = alsabar.notification_preset.max_ticks
+
+ if not tot then
+ local wib = awful.screen.focused().mywibox
+ -- 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
+ -- fallback: default horizontal wibox height
+ else
+ tot = 20
+ end
+ end
+
+ local int = math.modf((alsabar._current_level / 100) * tot)
+ preset.text = string.format(
+ "%s%s%s%s",
+ tick_pre,
+ string.rep(tick, int),
+ string.rep(tick_none, tot - int),
+ tick_post
+ )
+
+ if alsabar.followtag then preset.screen = awful.screen.focused() end
+
+ if not alsabar.notification then
+ alsabar.notification = naughty.notify {
+ preset = preset,
+ destroy = function() alsabar.notification = nil end
+ }
+ else
+ naughty.replace_text(alsabar.notification, preset.title, preset.text)
+ end
+ end)
+ end
+
+ helpers.newtimer(string.format("alsabar-%s-%s", alsabar.cmd, alsabar.channel), timeout, alsabar.update)
+
+ return alsabar
+end
+
+return factory
diff --git a/rsc/config/awesome/lain/widget/bat.lua b/rsc/config/awesome/lain/widget/bat.lua
new file mode 100644
index 0000000..260f4b9
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/bat.lua
@@ -0,0 +1,236 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2013, Luca CPZ
+ * (c) 2010-2012, Peter Hofmann
+
+--]]
+
+local helpers = require("lain.helpers")
+local fs = require("gears.filesystem")
+local naughty = require("naughty")
+local wibox = require("wibox")
+local math = math
+local string = string
+local ipairs = ipairs
+local tonumber = tonumber
+
+-- Battery infos
+-- lain.widget.bat
+
+local function factory(args)
+ local pspath = args.pspath or "/sys/class/power_supply/"
+
+ if not fs.is_dir(pspath) then
+ naughty.notify { text = "lain.widget.bat: invalid power supply path", timeout = 0 }
+ return
+ end
+
+ args = args or {}
+
+ local bat = { widget = args.widget or wibox.widget.textbox() }
+ local timeout = args.timeout or 30
+ local notify = args.notify or "on"
+ local full_notify = args.full_notify or notify
+ local n_perc = args.n_perc or { 5, 15 }
+ local batteries = args.batteries or (args.battery and {args.battery}) or {}
+ local ac = args.ac or "AC0"
+ local settings = args.settings or function() end
+
+ function bat.get_batteries()
+ helpers.line_callback("ls -1 " .. pspath, function(line)
+ local bstr = string.match(line, "BAT%w+")
+ if bstr then
+ batteries[#batteries + 1] = bstr
+ else
+ ac = string.match(line, "A%w+") or ac
+ end
+ end)
+ end
+
+ if #batteries == 0 then bat.get_batteries() end
+
+ bat_notification_critical_preset = {
+ title = "Battery exhausted",
+ text = "Shutdown imminent",
+ timeout = 15,
+ fg = "#000000",
+ bg = "#FFFFFF"
+ }
+
+ bat_notification_low_preset = {
+ title = "Battery low",
+ text = "Plug the cable!",
+ timeout = 15,
+ fg = "#202020",
+ bg = "#CDCDCD"
+ }
+
+ bat_notification_charged_preset = {
+ title = "Battery full",
+ text = "You can unplug the cable",
+ timeout = 15,
+ fg = "#202020",
+ bg = "#CDCDCD"
+ }
+
+ bat_now = {
+ status = "N/A",
+ ac_status = "N/A",
+ perc = "N/A",
+ time = "N/A",
+ watt = "N/A",
+ capacity = "N/A"
+ }
+
+ bat_now.n_status = {}
+ bat_now.n_perc = {}
+ bat_now.n_capacity = {}
+ for i = 1, #batteries do
+ bat_now.n_status[i] = "N/A"
+ bat_now.n_perc[i] = 0
+ bat_now.n_capacity[i] = 0
+ end
+
+ -- used to notify full charge only once before discharging
+ local fullnotification = false
+
+ function bat.update()
+ -- luacheck: globals bat_now
+ local sum_rate_current = 0
+ local sum_rate_voltage = 0
+ local sum_rate_power = 0
+ local sum_rate_energy = 0
+ local sum_energy_now = 0
+ local sum_energy_full = 0
+ local sum_charge_full = 0
+ local sum_charge_design = 0
+
+ for i, battery in ipairs(batteries) do
+ local bstr = pspath .. battery
+ local present = helpers.first_line(bstr .. "/present")
+
+ if tonumber(present) == 1 then
+ -- current_now(I)[uA], voltage_now(U)[uV], power_now(P)[uW]
+ local rate_current = tonumber(helpers.first_line(bstr .. "/current_now"))
+ local rate_voltage = tonumber(helpers.first_line(bstr .. "/voltage_now"))
+ local rate_power = tonumber(helpers.first_line(bstr .. "/power_now"))
+ local charge_full = tonumber(helpers.first_line(bstr .. "/charge_full"))
+ local charge_design = tonumber(helpers.first_line(bstr .. "/charge_full_design"))
+
+ -- energy_now(P)[uWh], charge_now(I)[uAh]
+ local energy_now = tonumber(helpers.first_line(bstr .. "/energy_now") or
+ helpers.first_line(bstr .. "/charge_now"))
+
+ -- energy_full(P)[uWh], charge_full(I)[uAh]
+ local energy_full = tonumber(helpers.first_line(bstr .. "/energy_full") or
+ charge_full)
+
+ local energy_percentage = tonumber(helpers.first_line(bstr .. "/capacity")) or
+ math.floor((energy_now / energy_full) * 100)
+
+ bat_now.n_status[i] = helpers.first_line(bstr .. "/status") or "N/A"
+ bat_now.n_perc[i] = energy_percentage or bat_now.n_perc[i]
+
+ if not charge_design or charge_design == 0 then
+ bat_now.n_capacity[i] = 0
+ else
+ bat_now.n_capacity[i] = math.floor((charge_full / charge_design) * 100)
+ end
+
+ sum_rate_current = sum_rate_current + (rate_current or 0)
+ sum_rate_voltage = sum_rate_voltage + (rate_voltage or 0)
+ sum_rate_power = sum_rate_power + (rate_power or 0)
+ sum_rate_energy = sum_rate_energy + (rate_power or (((rate_voltage or 0) * (rate_current or 0)) / 1e6))
+ sum_energy_now = sum_energy_now + (energy_now or 0)
+ sum_energy_full = sum_energy_full + (energy_full or 0)
+ sum_charge_full = sum_charge_full + (charge_full or 0)
+ sum_charge_design = sum_charge_design + (charge_design or 0)
+ end
+ end
+
+ bat_now.capacity = math.floor(math.min(100, (sum_charge_full / sum_charge_design) * 100))
+
+ -- When one of the battery is charging, others' status are either
+ -- "Full", "Unknown" or "Charging". When the laptop is not plugged in,
+ -- one or more of the batteries may be full, but only one battery
+ -- discharging suffices to set global status to "Discharging".
+ bat_now.status = bat_now.n_status[1] or "N/A"
+ for _,status in ipairs(bat_now.n_status) do
+ if status == "Discharging" or status == "Charging" then
+ bat_now.status = status
+ end
+ end
+ bat_now.ac_status = tonumber(helpers.first_line(string.format("%s%s/online", pspath, ac))) or "N/A"
+
+ if bat_now.status ~= "N/A" then
+ if bat_now.status ~= "Full" and sum_rate_power == 0 and bat_now.ac_status == 1 then
+ bat_now.perc = math.floor(math.min(100, (sum_energy_now / sum_energy_full) * 100))
+ bat_now.time = "00:00"
+ bat_now.watt = 0
+
+ -- update {perc,time,watt} iff battery not full and rate > 0
+ elseif bat_now.status ~= "Full" then
+ local rate_time = 0
+ -- Calculate time and watt if rates are greater then 0
+ if (sum_rate_power > 0 or sum_rate_current > 0) then
+ local div = (sum_rate_power > 0 and sum_rate_power) or sum_rate_current
+
+ if bat_now.status == "Charging" then
+ rate_time = (sum_energy_full - sum_energy_now) / div
+ else -- Discharging
+ rate_time = sum_energy_now / div
+ end
+
+ if 0 < rate_time and rate_time < 0.01 then -- check for magnitude discrepancies (#199)
+ rate_time_magnitude = math.abs(math.floor(math.log10(rate_time)))
+ rate_time = rate_time * 10^(rate_time_magnitude - 2)
+ end
+ end
+
+ local hours = math.floor(rate_time)
+ local minutes = math.floor((rate_time - hours) * 60)
+ bat_now.perc = math.floor(math.min(100, (sum_energy_now / sum_energy_full) * 100))
+ bat_now.time = string.format("%02d:%02d", hours, minutes)
+ bat_now.watt = tonumber(string.format("%.2f", sum_rate_energy / 1e6))
+ elseif bat_now.status == "Full" then
+ bat_now.perc = 100
+ bat_now.time = "00:00"
+ bat_now.watt = 0
+ end
+ end
+
+ widget = bat.widget
+ settings()
+
+ -- notifications for critical, low, and full levels
+ if notify == "on" then
+ if bat_now.status == "Discharging" then
+ if tonumber(bat_now.perc) <= n_perc[1] then
+ bat.id = naughty.notify({
+ preset = bat_notification_critical_preset,
+ replaces_id = bat.id
+ }).id
+ elseif tonumber(bat_now.perc) <= n_perc[2] then
+ bat.id = naughty.notify({
+ preset = bat_notification_low_preset,
+ replaces_id = bat.id
+ }).id
+ end
+ fullnotification = false
+ elseif bat_now.status == "Full" and full_notify == "on" and not fullnotification then
+ bat.id = naughty.notify({
+ preset = bat_notification_charged_preset,
+ replaces_id = bat.id
+ }).id
+ fullnotification = true
+ end
+ end
+ end
+
+ helpers.newtimer("batteries", timeout, bat.update)
+
+ return bat
+end
+
+return factory
diff --git a/rsc/config/awesome/lain/widget/cal.lua b/rsc/config/awesome/lain/widget/cal.lua
new file mode 100644
index 0000000..97dced8
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/cal.lua
@@ -0,0 +1,191 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2018, Luca CPZ
+
+--]]
+
+local helpers = require("lain.helpers")
+local markup = require("lain.util.markup")
+local awful = require("awful")
+local naughty = require("naughty")
+local floor = math.floor
+local os = os
+local pairs = pairs
+local string = string
+local tconcat = table.concat
+local type = type
+local tonumber = tonumber
+local tostring = tostring
+
+-- Calendar notification
+-- lain.widget.cal
+
+local function factory(args)
+ args = args or {}
+ local cal = {
+ attach_to = args.attach_to or {},
+ week_start = args.week_start or 2,
+ three = args.three or false,
+ followtag = args.followtag or false,
+ week_number = args.week_number or "none",
+ week_number_format = args.week_number_format or args.week_number == "left" and "%3d | " or "| %-3d",
+ icons = args.icons or helpers.icons_dir .. "cal/white/",
+ notification_preset = args.notification_preset or {
+ font = "Monospace 10", fg = "#FFFFFF", bg = "#000000"
+ }
+ }
+
+ function cal.get_week_number(m, st_day, x)
+ local date = os.date("*t", m)
+
+ local week_step = (x ~= 0 and floor((x + st_day) / 7) - 1 or 0);
+
+ local display_time = os.time {
+ year = date.year, month = date.month, day = date.day + 7 * week_step
+ }
+
+ return string.format(cal.week_number_format, os.date("%V", display_time))
+ end
+
+ function cal.sum_week_days(x, y)
+ return (x + y) % 7
+ end
+
+ function cal.build(month, year)
+ local current_month, current_year = tonumber(os.date("%m")), tonumber(os.date("%Y"))
+ local is_current_month = (not month or not year) or (month == current_month and year == current_year)
+ local today = is_current_month and tonumber(os.date("%d")) -- otherwise nil and not highlighted
+ local t = os.time { year = year or current_year, month = month and month+1 or current_month+1, day = 0 }
+ local d = os.date("*t", t)
+ local mth_days, st_day, this_month = d.day, (d.wday-d.day-cal.week_start+1)%7, os.date("%B %Y", t)
+ local notifytable = { [1] = string.format("%s%s\n", string.rep(" ", floor((28 - this_month:len())/2)), markup.bold(this_month)) }
+ for day_num = 0, 6 do
+ notifytable[#notifytable+1] = string.format("%3s ", os.date("%a", os.time { year = 2006, month = 1, day = day_num + cal.week_start }))
+ end
+ notifytable[#notifytable] = string.format("%s\n%s", notifytable[#notifytable]:sub(1, -2), string.rep(" ", st_day*4))
+ local strx
+ for x = 1,mth_days do
+ strx = x
+ if x == today then
+ if x < 10 then x = " " .. x end
+ strx = markup.bold(markup.color(cal.notification_preset.bg, cal.notification_preset.fg, x) .. " ")
+ end
+ strx = string.format("%s%s", string.rep(" ", 3 - tostring(x):len()), strx)
+ notifytable[#notifytable+1] = string.format("%-4s%s", strx, (x+st_day)%7==0 and x ~= mth_days and "\n" or "")
+ end
+ if string.len(cal.icons or "") > 0 and today then cal.icon = cal.icons .. today .. ".png" end
+ cal.month, cal.year = d.month, d.year
+
+ if cal.week_number ~= "none" then
+ local m = os.time { year = year or current_year, month = month and month or current_month, day = 1 }
+ local head_prepend = string.rep(" ", tostring(string.format(cal.week_number_format, 0)):len())
+
+ if cal.week_number == "left" then
+ notifytable[1] = head_prepend .. notifytable[1] -- month-year row
+ notifytable[2] = head_prepend .. notifytable[2] -- weekdays row
+ notifytable[8] = notifytable[8]:gsub("\n", "\n" .. cal.get_week_number(m, st_day, 0)) -- first week of the month
+
+ for x = 10,#notifytable do
+ if cal.sum_week_days(st_day, x) == 2 then
+ notifytable[x] = cal.get_week_number(m, st_day, x) .. notifytable[x]
+ end
+ end
+ elseif cal.week_number == "right" then
+ notifytable[8] = notifytable[8]:gsub("\n", head_prepend .. "\n") -- weekdays row
+ for x = 9,#notifytable do
+ if cal.sum_week_days(st_day, x) == 1 then
+ notifytable[x] = notifytable[x]:gsub("\n", cal.get_week_number(m, st_day, x - 7) .. "\n")
+ end
+ end
+ -- last week of the month
+ local end_days = cal.sum_week_days(st_day, mth_days)
+ if end_days ~= 0 then end_days = 7 - end_days end
+ notifytable[#notifytable] = notifytable[#notifytable] .. string.rep(" ", 4 * end_days) .. cal.get_week_number(m, st_day, mth_days + end_days)
+ end
+ end
+
+ return notifytable
+ end
+
+ function cal.getdate(month, year, offset)
+ if not month or not year then
+ month = tonumber(os.date("%m"))
+ year = tonumber(os.date("%Y"))
+ end
+
+ month = month + offset
+
+ while month > 12 do
+ month = month - 12
+ year = year + 1
+ end
+
+ while month < 1 do
+ month = month + 12
+ year = year - 1
+ end
+
+ return month, year
+ end
+
+ function cal.hide()
+ if not cal.notification then return end
+ naughty.destroy(cal.notification)
+ cal.notification = nil
+ end
+
+ function cal.show(seconds, month, year, scr)
+ local text = tconcat(cal.build(month, year))
+
+ if cal.three then
+ local current_month, current_year = cal.month, cal.year
+ local prev_month, prev_year = cal.getdate(cal.month, cal.year, -1)
+ local next_month, next_year = cal.getdate(cal.month, cal.year, 1)
+ text = string.format("%s\n\n%s\n\n%s",
+ tconcat(cal.build(prev_month, prev_year)), text,
+ tconcat(cal.build(next_month, next_year)))
+ cal.month, cal.year = current_month, current_year
+ end
+
+ if cal.notification then
+ local title = cal.notification_preset.title or nil
+ naughty.replace_text(cal.notification, title, text)
+ return
+ end
+
+ cal.notification = naughty.notify {
+ preset = cal.notification_preset,
+ screen = cal.followtag and awful.screen.focused() or scr or 1,
+ icon = cal.icon,
+ timeout = type(seconds) == "number" and seconds or cal.notification_preset.timeout or 5,
+ text = text
+ }
+ end
+
+ function cal.hover_on() cal.show(0) end
+ function cal.move(offset)
+ offset = offset or 0
+ cal.month, cal.year = cal.getdate(cal.month, cal.year, offset)
+ cal.show(0, cal.month, cal.year)
+ end
+ function cal.prev() cal.move(-1) end
+ function cal.next() cal.move( 1) end
+
+ function cal.attach(widget)
+ widget:connect_signal("mouse::enter", cal.hover_on)
+ widget:connect_signal("mouse::leave", cal.hide)
+ widget:buttons(awful.util.table.join(
+ awful.button({}, 1, cal.prev),
+ awful.button({}, 3, cal.next),
+ awful.button({}, 2, cal.hover_on),
+ awful.button({}, 5, cal.prev),
+ awful.button({}, 4, cal.next)))
+ end
+
+ for _, widget in pairs(cal.attach_to) do cal.attach(widget) end
+
+ return cal
+end
+
+return factory
diff --git a/rsc/config/awesome/lain/widget/contrib/init.lua b/rsc/config/awesome/lain/widget/contrib/init.lua
new file mode 100644
index 0000000..9e863a5
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/contrib/init.lua
@@ -0,0 +1,18 @@
+--[[
+
+ Lain
+ Layouts, widgets and utilities for Awesome WM
+
+ Users contributed widgets section
+
+ Licensed under GNU General Public License v2
+ * (c) 2013, Luca CPZ
+
+--]]
+
+local wrequire = require("lain.helpers").wrequire
+local setmetatable = setmetatable
+
+local widget = { _NAME = "lain.widget.contrib" }
+
+return setmetatable(widget, { __index = wrequire })
diff --git a/rsc/config/awesome/lain/widget/contrib/moc.lua b/rsc/config/awesome/lain/widget/contrib/moc.lua
new file mode 100644
index 0000000..ad6452e
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/contrib/moc.lua
@@ -0,0 +1,97 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2014, anticlockwise
+
+--]]
+
+local helpers = require("lain.helpers")
+local shell = require("awful.util").shell
+local focused = require("awful.screen").focused
+local escape_f = require("awful.util").escape
+local naughty = require("naughty")
+local wibox = require("wibox")
+local os = os
+local string = string
+
+-- MOC audio player
+-- lain.widget.contrib.moc
+
+local function factory(args)
+ args = args or {}
+
+ local moc = { widget = args.widget or wibox.widget.textbox() }
+ local timeout = args.timeout or 2
+ 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 or ""
+ local followtag = args.followtag or false
+ local settings = args.settings or function() end
+
+ moc_notification_preset = { title = "Now playing", timeout = 6 }
+
+ helpers.set_map("current moc track", nil)
+
+ function moc.update()
+ helpers.async("mocp -i", function(f)
+ moc_now = {
+ state = "N/A",
+ file = "N/A",
+ artist = "N/A",
+ title = "N/A",
+ album = "N/A",
+ elapsed = "N/A",
+ total = "N/A"
+ }
+
+ for line in string.gmatch(f, "[^\n]+") do
+ for k, v in string.gmatch(line, "([%w]+):[%s](.*)$") do
+ if k == "State" then moc_now.state = v
+ elseif k == "File" then moc_now.file = v
+ elseif k == "Artist" then moc_now.artist = escape_f(v)
+ elseif k == "SongTitle" then moc_now.title = escape_f(v)
+ elseif k == "Album" then moc_now.album = escape_f(v)
+ elseif k == "CurrentTime" then moc_now.elapsed = escape_f(v)
+ elseif k == "TotalTime" then moc_now.total = escape_f(v)
+ end
+ end
+ end
+
+ moc_notification_preset.text = string.format("%s (%s) - %s\n%s", moc_now.artist,
+ moc_now.album, moc_now.total, moc_now.title)
+ widget = moc.widget
+ settings()
+
+ if moc_now.state == "PLAY" then
+ if moc_now.title ~= helpers.get_map("current moc track") then
+ helpers.set_map("current moc track", moc_now.title)
+
+ if followtag then moc_notification_preset.screen = focused() end
+
+ local common = {
+ preset = moc_notification_preset,
+ icon = default_art,
+ icon_size = cover_size,
+ replaces_id = moc.id,
+ }
+
+ local path = string.format("%s/%s", music_dir, string.match(moc_now.file, ".*/"))
+ local cover = string.format("find '%s' -maxdepth 1 -type f | egrep -i -m1 '%s'", path, cover_pattern)
+ helpers.async({ shell, "-c", cover }, function(current_icon)
+ common.icon = current_icon:gsub("\n", "")
+ moc.id = naughty.notify(common).id
+ end)
+ end
+ elseif moc_now.state ~= "PAUSE" then
+ helpers.set_map("current moc track", nil)
+ end
+ end)
+ end
+
+ moc.timer = helpers.newtimer("moc", timeout, moc.update, true, true)
+
+ return moc
+end
+
+return factory
diff --git a/rsc/config/awesome/lain/widget/contrib/redshift.lua b/rsc/config/awesome/lain/widget/contrib/redshift.lua
new file mode 100644
index 0000000..d91d941
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/contrib/redshift.lua
@@ -0,0 +1,54 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2017, Luca CPZ
+ * (c) 2014, blueluke
+
+--]]
+
+local async = require("lain.helpers").async
+local awful = require("awful")
+local execute = os.execute
+local type = type
+
+-- Redshift
+-- lain.widget.contrib.redshift
+local redshift = { active = false, pid = nil }
+
+function redshift.start()
+ execute("pkill redshift")
+ awful.spawn.with_shell("redshift -x") -- clear adjustments
+ redshift.pid = awful.spawn.with_shell("redshift")
+ redshift.active = true
+ if type(redshift.update_fun) == "function" then
+ redshift.update_fun(redshift.active)
+ end
+end
+
+function redshift.toggle()
+ async({ awful.util.shell, "-c", string.format("ps -p %d -o pid=", redshift.pid) }, function(f)
+ if f and #f > 0 then -- redshift is running
+ -- Sending -USR1 toggles redshift (See project website)
+ execute("pkill -USR1 redshift")
+ redshift.active = not redshift.active
+ else -- not started or killed, (re)start it
+ redshift.start()
+ end
+ redshift.update_fun(redshift.active)
+ end)
+end
+
+-- Attach to a widget
+-- Provides a button which toggles redshift on/off on click
+-- @param widget: Widget to attach to.
+-- @param fun: Function to be run each time redshift is toggled (optional).
+-- Use it to update widget text or icons on status change.
+function redshift.attach(widget, fun)
+ redshift.update_fun = fun or function() end
+ if not redshift.pid then redshift.start() end
+ if widget then
+ widget:buttons(awful.util.table.join(awful.button({}, 1, function () redshift.toggle() end)))
+ end
+end
+
+return redshift
diff --git a/rsc/config/awesome/lain/widget/contrib/task.lua b/rsc/config/awesome/lain/widget/contrib/task.lua
new file mode 100644
index 0000000..2311996
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/contrib/task.lua
@@ -0,0 +1,92 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2013, Jan Xie
+
+--]]
+
+local helpers = require("lain.helpers")
+local markup = require("lain.util").markup
+local awful = require("awful")
+local naughty = require("naughty")
+local mouse = mouse
+
+-- Taskwarrior notification
+-- lain.widget.contrib.task
+local task = {}
+
+function task.hide()
+ if not task.notification then return end
+ naughty.destroy(task.notification)
+ task.notification = nil
+end
+
+function task.show(scr)
+ task.notification_preset.screen = task.followtag and awful.screen.focused() or scr or 1
+
+ helpers.async({ awful.util.shell, "-c", task.show_cmd }, function(f)
+ local widget_focused = true
+
+ if mouse.current_widgets then
+ widget_focused = false
+ for _,v in ipairs(mouse.current_widgets) do
+ if task.widget == v then
+ widget_focused = true
+ break
+ end
+ end
+ end
+
+ if widget_focused then
+ task.hide()
+ task.notification = naughty.notify {
+ preset = task.notification_preset,
+ title = "task next",
+ text = markup.font(task.notification_preset.font,
+ awful.util.escape(f:gsub("\n*$", "")))
+ }
+ end
+ end)
+end
+
+function task.prompt()
+ awful.prompt.run {
+ prompt = task.prompt_text,
+ textbox = awful.screen.focused().mypromptbox.widget,
+ exe_callback = function(t)
+ helpers.async(t, function(f)
+ naughty.notify {
+ preset = task.notification_preset,
+ title = t,
+ text = markup.font(task.notification_preset.font,
+ awful.util.escape(f:gsub("\n*$", "")))
+ }
+ end)
+ end,
+ history_path = awful.util.getdir("cache") .. "/history_task"
+ }
+end
+
+function task.attach(widget, args)
+ args = args or {}
+
+ task.show_cmd = args.show_cmd or "task next"
+ task.prompt_text = args.prompt_text or "Enter task command: "
+ task.followtag = args.followtag or false
+ task.notification_preset = args.notification_preset
+ task.widget = widget
+
+ if not task.notification_preset then
+ task.notification_preset = {
+ font = "Monospace 10",
+ icon = helpers.icons_dir .. "/taskwarrior.png"
+ }
+ end
+
+ if widget then
+ widget:connect_signal("mouse::enter", function () task.show() end)
+ widget:connect_signal("mouse::leave", function () task.hide() end)
+ end
+end
+
+return task
diff --git a/rsc/config/awesome/lain/widget/contrib/tp_smapi.lua b/rsc/config/awesome/lain/widget/contrib/tp_smapi.lua
new file mode 100644
index 0000000..87c5e51
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/contrib/tp_smapi.lua
@@ -0,0 +1,147 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2018, Luca CPZ
+ * (c) 2013, Conor Heine
+
+--]]
+
+local helpers = require("lain.helpers")
+local focused = require("awful.screen").focused
+local naughty = require("naughty")
+local wibox = require("wibox")
+local string = string
+local type = type
+
+-- ThinkPad battery infos and widget creator
+-- http://www.thinkwiki.org/wiki/Tp_smapi
+-- lain.widget.contrib.tp_smapi
+
+local function factory(apipath)
+ local tp_smapi = {
+ path = apipath or "/sys/devices/platform/smapi"
+ }
+
+ function tp_smapi.get(batid, feature)
+ return helpers.first_line(string.format("%s/%s/%s", tp_smapi.path, batid or "BAT0", feature or ""))
+ end
+
+ function tp_smapi.installed(batid)
+ return tp_smapi.get(batid, "installed") == "1"
+ end
+
+ function tp_smapi.status(batid)
+ return tp_smapi.get(batid, "state")
+ end
+
+ function tp_smapi.percentage(batid)
+ return tp_smapi.get(batid, "remaining_percent")
+ end
+
+ -- either running or charging time
+ function tp_smapi.time(batid)
+ local status = tp_smapi.status(batid)
+ local mins_left = tp_smapi.get(batid, string.match(string.lower(status), "discharging") and "remaining_running_time" or "remaining_charging_time")
+ if not string.find(mins_left, "^%d+") then return "N/A" end
+ return string.format("%02d:%02d", math.floor(mins_left / 60), mins_left % 60) -- HH:mm
+ end
+
+ function tp_smapi.hide()
+ if not tp_smapi.notification then return end
+ naughty.destroy(tp_smapi.notification)
+ tp_smapi.notification = nil
+ end
+
+ function tp_smapi.show(batid, seconds, scr)
+ if not tp_smapi.installed(batid) then return end
+
+ local mfgr = tp_smapi.get(batid, "manufacturer") or "no_mfgr"
+ local model = tp_smapi.get(batid, "model") or "no_model"
+ local chem = tp_smapi.get(batid, "chemistry") or "no_chem"
+ local status = tp_smapi.get(batid, "state")
+ local time = tp_smapi.time(batid)
+ local msg
+
+ if status and status ~= "idle" then
+ msg = string.format("[%s] %s %s", status, time ~= "N/A" and time or "unknown remaining time",
+ string.lower(status):gsub(" ", ""):gsub("\n", "") == "charging" and " until charged" or " remaining")
+ else
+ msg = "On AC power"
+ end
+
+ tp_smapi.hide()
+ tp_smapi.notification = naughty.notify {
+ title = string.format("%s: %s %s (%s)", batid, mfgr, model, chem),
+ text = msg,
+ timeout = type(seconds) == "number" and seconds or 0,
+ screen = scr or focused()
+ }
+ end
+
+ function tp_smapi.create_widget(args)
+ args = args or {}
+
+ local pspath = args.pspath or "/sys/class/power_supply/"
+ local batteries = args.batteries or (args.battery and {args.battery}) or {}
+ local timeout = args.timeout or 30
+ local settings = args.settings or function() end
+
+ if #batteries == 0 then
+ helpers.line_callback("ls -1 " .. pspath, function(line)
+ local bstr = string.match(line, "BAT%w+")
+ if bstr then batteries[#batteries + 1] = bstr end
+ end)
+ end
+
+ local all_batteries_installed = true
+
+ for _, battery in ipairs(batteries) do
+ if not tp_smapi.installed(battery) then
+ naughty.notify {
+ preset = naughty.config.critical,
+ title = "tp_smapi: error while creating widget",
+ text = string.format("battery %s is not installed", battery)
+ }
+ all_batteries_installed = false
+ break
+ end
+ end
+
+ if not all_batteries_installed then return end
+
+ tpbat = {
+ batteries = batteries,
+ widget = args.widget or wibox.widget.textbox()
+ }
+
+ function tpbat.update()
+ tpbat_now = {
+ n_status = {},
+ n_perc = {},
+ n_time = {},
+ status = "N/A"
+ }
+
+ for i = 1, #batteries do
+ tpbat_now.n_status[i] = tp_smapi.status(batteries[i]) or "N/A"
+ tpbat_now.n_perc[i] = tp_smapi.percentage(batteries[i])
+ tpbat_now.n_time[i] = tp_smapi.time(batteries[i]) or "N/A"
+
+ if not tpbat_now.n_status[i]:lower():match("full") then
+ tpbat_now.status = tpbat_now.n_status[i]
+ end
+ end
+
+ widget = tpbat.widget -- backwards compatibility
+ settings()
+ end
+
+ helpers.newtimer("thinkpad-batteries", timeout, tpbat.update)
+
+ return tpbat
+ end
+
+ return tp_smapi
+end
+
+return factory
diff --git a/rsc/config/awesome/lain/widget/cpu.lua b/rsc/config/awesome/lain/widget/cpu.lua
new file mode 100644
index 0000000..6c51115
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/cpu.lua
@@ -0,0 +1,75 @@
+--[[
+
+ 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 math = math
+local string = string
+
+-- CPU usage
+-- lain.widget.cpu
+
+local function factory(args)
+ args = args or {}
+
+ local cpu = { core = {}, widget = args.widget or wibox.widget.textbox() }
+ local timeout = args.timeout or 2
+ local settings = args.settings or function() end
+
+ function cpu.update()
+ -- Read the amount of time the CPUs have spent performing
+ -- different kinds of work. Read the first line of /proc/stat
+ -- which is the sum of all CPUs.
+ for index,time in pairs(helpers.lines_match("cpu","/proc/stat")) do
+ local coreid = index - 1
+ local core = cpu.core[coreid] or
+ { last_active = 0 , last_total = 0, usage = 0 }
+ local at = 1
+ local idle = 0
+ local total = 0
+
+ for field in string.gmatch(time, "[%s]+([^%s]+)") do
+ -- 4 = idle, 5 = ioWait. Essentially, the CPUs have done
+ -- nothing during these times.
+ if at == 4 or at == 5 then
+ idle = idle + field
+ end
+ total = total + field
+ at = at + 1
+ end
+
+ local active = total - idle
+
+ if core.last_active ~= active or core.last_total ~= total then
+ -- Read current data and calculate relative values.
+ local dactive = active - core.last_active
+ local dtotal = total - core.last_total
+ local usage = math.ceil(math.abs((dactive / dtotal) * 100))
+
+ core.last_active = active
+ core.last_total = total
+ core.usage = usage
+
+ -- Save current data for the next run.
+ cpu.core[coreid] = core
+ end
+ end
+
+ cpu_now = cpu.core
+ cpu_now.usage = cpu_now[0].usage
+ widget = cpu.widget
+
+ settings()
+ end
+
+ helpers.newtimer("cpu", timeout, cpu.update)
+
+ return cpu
+end
+
+return factory
diff --git a/rsc/config/awesome/lain/widget/fs.lua b/rsc/config/awesome/lain/widget/fs.lua
new file mode 100644
index 0000000..b3a2dad
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/fs.lua
@@ -0,0 +1,156 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2018, Uli Schlacter
+ * (c) 2018, Otto Modinos
+ * (c) 2013, Luca CPZ
+
+--]]
+
+local helpers = require("lain.helpers")
+local Gio = require("lgi").Gio
+local focused = require("awful.screen").focused
+local wibox = require("wibox")
+local naughty = require("naughty")
+local gears = require("gears")
+local math = math
+local string = string
+local tconcat = table.concat
+local type = type
+local query_size = Gio.FILE_ATTRIBUTE_FILESYSTEM_SIZE
+local query_free = Gio.FILE_ATTRIBUTE_FILESYSTEM_FREE
+local query_used = Gio.FILE_ATTRIBUTE_FILESYSTEM_USED
+local query = query_size .. "," .. query_free .. "," .. query_used
+
+-- File systems info
+-- lain.widget.fs
+
+local function factory(args)
+ args = args or {}
+
+ local fs = {
+ widget = args.widget or wibox.widget.textbox(),
+ units = {
+ [1] = "Kb", [2] = "Mb", [3] = "Gb",
+ [4] = "Tb", [5] = "Pb", [6] = "Eb",
+ [7] = "Zb", [8] = "Yb"
+ }
+ }
+
+ function fs.hide()
+ if not fs.notification then return end
+ naughty.destroy(fs.notification)
+ fs.notification = nil
+ end
+
+ function fs.show(seconds, scr)
+ fs.hide()
+ fs.update(function()
+ fs.notification_preset.screen = fs.followtag and focused() or scr or 1
+ fs.notification = naughty.notify {
+ preset = fs.notification_preset,
+ timeout = type(seconds) == "number" and seconds or 5
+ }
+ end)
+ end
+
+ local timeout = args.timeout or 600
+ local partition = args.partition
+ local threshold = args.threshold or 99
+ local showpopup = args.showpopup or "on"
+ local settings = args.settings or function() end
+
+ fs.followtag = args.followtag or false
+ fs.notification_preset = args.notification_preset
+
+ if not fs.notification_preset then
+ fs.notification_preset = {
+ font = "Monospace 10",
+ fg = "#FFFFFF",
+ bg = "#000000"
+ }
+ end
+
+ local function update_synced()
+ local pathlen = 10
+ fs_now = {}
+
+ local notifypaths = {}
+ for _, mount in ipairs(Gio.unix_mounts_get()) do
+ local path = Gio.unix_mount_get_mount_path(mount)
+ local root = Gio.File.new_for_path(path)
+ local info = root:query_filesystem_info(query)
+
+ if info then
+ local size = info:get_attribute_uint64(query_size)
+ local used = info:get_attribute_uint64(query_used)
+ local free = info:get_attribute_uint64(query_free)
+
+ if size > 0 then
+ local units = math.floor(math.log(size)/math.log(1024))
+
+ fs_now[path] = {
+ units = fs.units[units],
+ percentage = math.floor(100 * used / size), -- used percentage
+ size = size / math.pow(1024, units),
+ used = used / math.pow(1024, units),
+ free = free / math.pow(1024, units)
+ }
+
+ if fs_now[path].percentage > 0 then -- don't notify unused file systems
+ notifypaths[#notifypaths+1] = path
+
+ if #path > pathlen then
+ pathlen = #path
+ end
+ end
+ end
+ end
+ end
+
+ widget = fs.widget
+ settings()
+
+ if partition and fs_now[partition] and fs_now[partition].percentage >= threshold then
+ if not helpers.get_map(partition) then
+ naughty.notify {
+ preset = naughty.config.presets.critical,
+ title = "Warning",
+ text = string.format("%s is above %d%% (%d%%)", partition, threshold, fs_now[partition].percentage)
+ }
+ helpers.set_map(partition, true)
+ else
+ helpers.set_map(partition, false)
+ end
+ end
+
+ local fmt = "%-" .. tostring(pathlen) .. "s %4s\t%6s\t%6s\n"
+ local notifytable = { [1] = string.format(fmt, "path", "used", "free", "size") }
+ fmt = "\n%-" .. tostring(pathlen) .. "s %3s%%\t%6.2f\t%6.2f %s"
+ for _, path in ipairs(notifypaths) do
+ notifytable[#notifytable+1] = string.format(fmt, path, fs_now[path].percentage, fs_now[path].free, fs_now[path].size, fs_now[path].units)
+ end
+
+ fs.notification_preset.text = tconcat(notifytable)
+ end
+
+ function fs.update(callback)
+ Gio.Async.start(gears.protected_call.call)(function()
+ update_synced()
+ if type(callback) == "function" and callback then
+ callback()
+ end
+ end)
+ end
+
+ if showpopup == "on" then
+ fs.widget:connect_signal('mouse::enter', function () fs.show(0) end)
+ fs.widget:connect_signal('mouse::leave', function () fs.hide() end)
+ end
+
+ helpers.newtimer(partition or "fs", timeout, fs.update)
+
+ return fs
+end
+
+return factory
diff --git a/rsc/config/awesome/lain/widget/imap.lua b/rsc/config/awesome/lain/widget/imap.lua
new file mode 100644
index 0000000..e3f7baa
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/imap.lua
@@ -0,0 +1,94 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2013, Luca CPZ
+
+--]]
+
+local helpers = require("lain.helpers")
+local naughty = require("naughty")
+local wibox = require("wibox")
+local awful = require("awful")
+local string = string
+local type = type
+local tonumber = tonumber
+
+-- Mail IMAP check
+-- lain.widget.imap
+
+local function factory(args)
+ args = args or {}
+
+ local imap = { widget = args.widget or wibox.widget.textbox() }
+ local server = args.server
+ local mail = args.mail
+ local password = args.password
+ local port = args.port or 993
+ local timeout = args.timeout or 60
+ local pwdtimeout = args.pwdtimeout or 10
+ local is_plain = args.is_plain or false
+ local followtag = args.followtag or false
+ local notify = args.notify or "on"
+ local settings = args.settings or function() end
+
+ local head_command = "curl --connect-timeout 3 -fsm 3"
+ local request = "-X 'STATUS INBOX (MESSAGES RECENT UNSEEN)'"
+
+ if not server or not mail or not password then return end
+
+ mail_notification_preset = {
+ icon = helpers.icons_dir .. "mail.png",
+ position = "top_left"
+ }
+
+ helpers.set_map(mail, 0)
+
+ if not is_plain then
+ if type(password) == "string" or type(password) == "table" then
+ helpers.async(password, function(f) password = f:gsub("\n", "") end)
+ elseif type(password) == "function" then
+ imap.pwdtimer = helpers.newtimer(mail .. "-password", pwdtimeout, function()
+ local retrieved_password, try_again = password()
+ if not try_again then
+ imap.pwdtimer:stop() -- stop trying to retrieve
+ password = retrieved_password or "" -- failsafe
+ end
+ end, true, true)
+ end
+ end
+
+ function imap.update()
+ -- do not update if the password has not been retrieved yet
+ if type(password) ~= "string" then return end
+
+ local curl = string.format("%s --url imaps://%s:%s/INBOX -u %s:'%s' %s -k",
+ head_command, server, port, mail, password, request)
+
+ helpers.async(curl, function(f)
+ imap_now = { ["MESSAGES"] = 0, ["RECENT"] = 0, ["UNSEEN"] = 0 }
+
+ for s,d in f:gmatch("(%w+)%s+(%d+)") do imap_now[s] = tonumber(d) end
+ mailcount = imap_now["UNSEEN"] -- backwards compatibility
+ widget = imap.widget
+
+ settings()
+
+ if notify == "on" and mailcount and mailcount >= 1 and mailcount > helpers.get_map(mail) then
+ if followtag then mail_notification_preset.screen = awful.screen.focused() end
+ naughty.notify {
+ preset = mail_notification_preset,
+ text = string.format("%s has %d new message%s", mail, mailcount, mailcount == 1 and "" or "s")
+ }
+ end
+
+ helpers.set_map(mail, imap_now["UNSEEN"])
+ end)
+
+ end
+
+ imap.timer = helpers.newtimer(mail, timeout, imap.update, true, true)
+
+ return imap
+end
+
+return factory
diff --git a/rsc/config/awesome/lain/widget/init.lua b/rsc/config/awesome/lain/widget/init.lua
new file mode 100644
index 0000000..57b86bb
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/init.lua
@@ -0,0 +1,19 @@
+--[[
+
+ Lain
+ Layouts, widgets and utilities for Awesome WM
+
+ Widgets section
+
+ Licensed under GNU General Public License v2
+ * (c) 2013, Luca CPZ
+ * (c) 2010-2012, Peter Hofmann
+
+--]]
+
+local wrequire = require("lain.helpers").wrequire
+local setmetatable = setmetatable
+
+local widget = { _NAME = "lain.widget" }
+
+return setmetatable(widget, { __index = wrequire })
diff --git a/rsc/config/awesome/lain/widget/mem.lua b/rsc/config/awesome/lain/widget/mem.lua
new file mode 100644
index 0000000..0318494
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/mem.lua
@@ -0,0 +1,51 @@
+--[[
+
+ 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 gmatch, lines, floor = string.gmatch, io.lines, math.floor
+
+-- Memory usage (ignoring caches)
+-- lain.widget.mem
+
+local function factory(args)
+ args = args or {}
+
+ local mem = { widget = args.widget or wibox.widget.textbox() }
+ local timeout = args.timeout or 2
+ local settings = args.settings or function() end
+
+ function mem.update()
+ mem_now = {}
+ for line in lines("/proc/meminfo") do
+ for k, v in gmatch(line, "([%a]+):[%s]+([%d]+).+") do
+ if k == "MemTotal" then mem_now.total = floor(v / 1024 + 0.5)
+ elseif k == "MemFree" then mem_now.free = floor(v / 1024 + 0.5)
+ elseif k == "Buffers" then mem_now.buf = floor(v / 1024 + 0.5)
+ elseif k == "Cached" then mem_now.cache = floor(v / 1024 + 0.5)
+ elseif k == "SwapTotal" then mem_now.swap = floor(v / 1024 + 0.5)
+ elseif k == "SwapFree" then mem_now.swapf = floor(v / 1024 + 0.5)
+ elseif k == "SReclaimable" then mem_now.srec = floor(v / 1024 + 0.5)
+ end
+ end
+ end
+
+ mem_now.used = mem_now.total - mem_now.free - mem_now.buf - mem_now.cache - mem_now.srec
+ mem_now.swapused = mem_now.swap - mem_now.swapf
+ mem_now.perc = math.floor(mem_now.used / mem_now.total * 100)
+
+ widget = mem.widget
+ settings()
+ end
+
+ helpers.newtimer("mem", timeout, mem.update)
+
+ return mem
+end
+
+return factory
diff --git a/rsc/config/awesome/lain/widget/mpd.lua b/rsc/config/awesome/lain/widget/mpd.lua
new file mode 100644
index 0000000..55d3649
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/mpd.lua
@@ -0,0 +1,135 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2013, Luca CPZ
+ * (c) 2010, Adrian C.
+
+--]]
+
+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)
+ args = args or {}
+
+ local mpd = { widget = args.widget or wibox.widget.textbox() }
+ 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",
+ volume = "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"
+ elseif k == "volume" then mpd_now.volume = v
+ 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/rsc/config/awesome/lain/widget/net.lua b/rsc/config/awesome/lain/widget/net.lua
new file mode 100644
index 0000000..9b7b165
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/net.lua
@@ -0,0 +1,122 @@
+--[[
+
+ 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)
+ args = args or {}
+
+ local net = { widget = args.widget or wibox.widget.textbox(), devices = {} }
+ 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 format = args.format or "%.1f"
+ 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_devices()
+ net.iface = {} -- reset at every call
+ 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_devices() 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(format, dev_now.sent)
+ dev_now.received = string.format(format, 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" then
+ dev_now.wifi = true
+ if string.match(dev_now.carrier, "1") then
+ dev_now.signal = tonumber(string.match(helpers.lines_from("/proc/net/wireless")[3], "(%-%d+%.)")) or nil
+ end
+ else
+ dev_now.wifi = false
+ end
+
+ if eth_state == "on" and helpers.first_line(string.format("/sys/class/net/%s/uevent", dev)) ~= "DEVTYPE=wlan" then
+ dev_now.ethernet = true
+ else
+ dev_now.ethernet = false
+ 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(format, net_now.sent)
+ net_now.received = string.format(format, net_now.received)
+
+ widget = net.widget
+ settings()
+ end
+
+ helpers.newtimer("network", timeout, net.update)
+
+ return net
+end
+
+return factory
diff --git a/rsc/config/awesome/lain/widget/pulse.lua b/rsc/config/awesome/lain/widget/pulse.lua
new file mode 100644
index 0000000..69f4d70
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/pulse.lua
@@ -0,0 +1,58 @@
+--[[
+
+ 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)
+ args = args or {}
+
+ local pulse = { widget = args.widget or wibox.widget.textbox(), device = "N/A" }
+ 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/rsc/config/awesome/lain/widget/pulsebar.lua b/rsc/config/awesome/lain/widget/pulsebar.lua
new file mode 100644
index 0000000..19e73b9
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/pulsebar.lua
@@ -0,0 +1,175 @@
+--[[
+
+ 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_background = "#000000",
+ mute = "#EB8F8F",
+ unmute = "#A4CE8A"
+ },
+
+ _current_level = 0,
+ _mute = "no",
+ device = "N/A"
+ }
+
+ 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
+ local tick = args.tick or "|"
+ local tick_pre = args.tick_pre or "["
+ local tick_post = args.tick_post or "]"
+ local tick_none = args.tick_none or " "
+
+ 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
+ pulsebar.bar.background_color = pulsebar.colors.mute_background
+ else
+ pulsebar._mute = "no"
+ pulsebar.tooltip:set_text(string.format("%s %s: %s", pulsebar.devicetype, pulsebar.device, volu))
+ pulsebar.bar.color = pulsebar.colors.unmute
+ pulsebar.bar.background_color = pulsebar.colors.background
+ 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
+
+ local int = math.modf((pulsebar._current_level / 100) * tot)
+ preset.text = string.format(
+ "%s%s%s%s",
+ tick_pre,
+ string.rep(tick, int),
+ string.rep(tick_none, tot - int),
+ tick_post
+ )
+
+ 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-%s", pulsebar.devicetype, pulsebar.device), timeout, pulsebar.update)
+
+ return pulsebar
+end
+
+return factory
diff --git a/rsc/config/awesome/lain/widget/sysload.lua b/rsc/config/awesome/lain/widget/sysload.lua
new file mode 100644
index 0000000..7260524
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/sysload.lua
@@ -0,0 +1,39 @@
+--[[
+
+ 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)
+ args = args or {}
+
+ local sysload = { widget = args.widget or wibox.widget.textbox() }
+ 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/rsc/config/awesome/lain/widget/temp.lua b/rsc/config/awesome/lain/widget/temp.lua
new file mode 100644
index 0000000..99f8700
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/temp.lua
@@ -0,0 +1,50 @@
+--[[
+
+ Licensed under GNU General Public License v2
+ * (c) 2013, Luca CPZ
+
+--]]
+
+local helpers = require("lain.helpers")
+local wibox = require("wibox")
+local tonumber = tonumber
+
+-- {thermal,core} temperature info
+-- lain.widget.temp
+
+local function factory(args)
+ args = args or {}
+
+ local temp = { widget = args.widget or wibox.widget.textbox() }
+ local timeout = args.timeout or 30
+ local tempfile = args.tempfile or "/sys/devices/virtual/thermal/thermal_zone0/temp"
+ local format = args.format or "%.1f"
+ local settings = args.settings or function() end
+
+ function temp.update()
+ helpers.async({"find", "/sys/devices", "-type", "f", "-name", "*temp*"}, function(f)
+ temp_now = {}
+ local temp_fl, temp_value
+ for t in f:gmatch("[^\n]+") do
+ temp_fl = helpers.first_line(t)
+ if temp_fl then
+ temp_value = tonumber(temp_fl)
+ temp_now[t] = temp_value and temp_value/1e3 or temp_fl
+ end
+ end
+ if temp_now[tempfile] then
+ coretemp_now = string.format(format, temp_now[tempfile])
+ else
+ coretemp_now = "N/A"
+ end
+ widget = temp.widget
+ settings()
+ end)
+ end
+
+ helpers.newtimer("thermal", timeout, temp.update)
+
+ return temp
+end
+
+return factory
diff --git a/rsc/config/awesome/lain/widget/weather.lua b/rsc/config/awesome/lain/widget/weather.lua
new file mode 100644
index 0000000..93028b5
--- /dev/null
+++ b/rsc/config/awesome/lain/widget/weather.lua
@@ -0,0 +1,149 @@
+--[[
+
+ 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 type = type
+local tonumber = tonumber
+
+-- OpenWeatherMap
+-- current weather and X-days forecast
+-- lain.widget.weather
+
+local function factory(args)
+ args = args or {}
+
+ local weather = { widget = args.widget or wibox.widget.textbox() }
+ local APPID = args.APPID -- mandatory api key
+ local timeout = args.timeout or 900 -- 15 min
+ local current_call = args.current_call or "curl -s 'https://api.openweathermap.org/data/2.5/weather?lat=%s&lon=%s&APPID=%s&units=%s&lang=%s'"
+ local forecast_call = args.forecast_call or "curl -s 'https://api.openweathermap.org/data/2.5/forecast?lat=%s&lon=%s&APPID=%s&cnt=%s&units=%s&lang=%s'"
+ local lat = args.lat or 0 -- placeholder
+ local lon = args.lon or 0 -- placeholder
+ local units = args.units or "metric"
+ local lang = args.lang or "en"
+ local cnt = args.cnt or 5
+ 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 temp = math.floor(wn["main"]["temp"])
+ local desc = wn["weather"][1]["description"]
+ return string.format("%s: %s, %d ", day, desc, temp)
+ 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(seconds)
+ 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 {
+ preset = notification_preset,
+ text = weather.notification_text,
+ icon = weather.icon_path,
+ timeout = type(seconds) == "number" and seconds or notification_preset.timeout
+ }
+ 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, lat, lon, APPID, cnt, units, lang)
+
+ helpers.async(cmd, function(f)
+ local err
+ weather_now, _, 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"], math.floor(weather_now["cnt"] / 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, lat, lon, APPID, units, lang)
+
+ helpers.async(cmd, function(f)
+ local err
+ weather_now, _, 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()
+
+ 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-" .. lat .. ":" .. lon, timeout, weather.update, false, true)
+ weather.timer_forecast = helpers.newtimer("weather_forecast-" .. lat .. ":" .. lon, timeout, weather.forecast_update, false, true)
+
+ return weather
+end
+
+return factory
diff --git a/rsc/config/awesome/rc.default.lua b/rsc/config/awesome/rc.default.lua
new file mode 100644
index 0000000..18815c8
--- /dev/null
+++ b/rsc/config/awesome/rc.default.lua
@@ -0,0 +1,564 @@
+-- If LuaRocks is installed, make sure that packages installed through it are
+-- found (e.g. lgi). If LuaRocks is not installed, do nothing.
+pcall(require, "luarocks.loader")
+
+-- Standard awesome library
+local gears = require("gears")
+local awful = require("awful")
+require("awful.autofocus")
+-- Widget and layout library
+local wibox = require("wibox")
+-- Theme handling library
+local beautiful = require("beautiful")
+-- Notification library
+local naughty = require("naughty")
+local menubar = require("menubar")
+local hotkeys_popup = require("awful.hotkeys_popup")
+-- Enable hotkeys help widget for VIM and other apps
+-- when client with a matching name is opened:
+require("awful.hotkeys_popup.keys")
+
+-- {{{ Error handling
+-- Check if awesome encountered an error during startup and fell back to
+-- another config (This code will only ever execute for the fallback config)
+if awesome.startup_errors then
+ naughty.notify({ preset = naughty.config.presets.critical,
+ title = "Oops, there were errors during startup!",
+ text = awesome.startup_errors })
+end
+
+-- Handle runtime errors after startup
+do
+ local in_error = false
+ awesome.connect_signal("debug::error", function (err)
+ -- Make sure we don't go into an endless error loop
+ if in_error then return end
+ in_error = true
+
+ naughty.notify({ preset = naughty.config.presets.critical,
+ title = "Oops, an error happened!",
+ text = tostring(err) })
+ in_error = false
+ end)
+end
+-- }}}
+
+-- {{{ Variable definitions
+-- Themes define colours, icons, font and wallpapers.
+beautiful.init(gears.filesystem.get_themes_dir() .. "default/theme.lua")
+
+-- This is used later as the default terminal and editor to run.
+terminal = "xterm"
+editor = os.getenv("EDITOR") or "nano"
+editor_cmd = terminal .. " -e " .. editor
+
+-- Default modkey.
+-- Usually, Mod4 is the key with a logo between Control and Alt.
+-- If you do not like this or do not have such a key,
+-- I suggest you to remap Mod4 to another key using xmodmap or other tools.
+-- However, you can use another modifier like Mod1, but it may interact with others.
+modkey = "Mod4"
+
+-- Table of layouts to cover with awful.layout.inc, order matters.
+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,
+}
+-- }}}
+
+-- {{{ Menu
+-- Create a launcher widget and a main menu
+myawesomemenu = {
+ { "hotkeys", function() hotkeys_popup.show_help(nil, awful.screen.focused()) end },
+ { "manual", terminal .. " -e man awesome" },
+ { "edit config", editor_cmd .. " " .. awesome.conffile },
+ { "restart", awesome.restart },
+ { "quit", function() awesome.quit() end },
+}
+
+mymainmenu = awful.menu({ items = { { "awesome", myawesomemenu, beautiful.awesome_icon },
+ { "open terminal", terminal }
+ }
+ })
+
+mylauncher = awful.widget.launcher({ image = beautiful.awesome_icon,
+ menu = mymainmenu })
+
+-- Menubar configuration
+menubar.utils.terminal = terminal -- Set the terminal for applications that require it
+-- }}}
+
+-- Keyboard map indicator and switcher
+mykeyboardlayout = awful.widget.keyboardlayout()
+
+-- {{{ Wibar
+-- Create a textclock widget
+mytextclock = wibox.widget.textclock()
+
+-- Create a wibox for each screen and add it
+local taglist_buttons = gears.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)
+ )
+
+local tasklist_buttons = gears.table.join(
+ awful.button({ }, 1, function (c)
+ if c == client.focus then
+ c.minimized = true
+ else
+ c:emit_signal(
+ "request::activate",
+ "tasklist",
+ {raise = true}
+ )
+ end
+ end),
+ awful.button({ }, 3, function()
+ awful.menu.client_list({ theme = { width = 250 } })
+ end),
+ awful.button({ }, 4, function ()
+ awful.client.focus.byidx(1)
+ end),
+ awful.button({ }, 5, function ()
+ awful.client.focus.byidx(-1)
+ end))
+
+local function set_wallpaper(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
+
+-- Re-set wallpaper when a screen's geometry changes (e.g. different resolution)
+screen.connect_signal("property::geometry", set_wallpaper)
+
+awful.screen.connect_for_each_screen(function(s)
+ -- Wallpaper
+ set_wallpaper(s)
+
+ -- Each screen has its own tag table.
+ awful.tag({ "1", "2", "3", "4", "5", "6", "7", "8", "9" }, s, awful.layout.layouts[1])
+
+ -- Create a promptbox for each screen
+ s.mypromptbox = awful.widget.prompt()
+ -- Create an imagebox widget which will contain an icon indicating which layout we're using.
+ -- We need one layoutbox per screen.
+ s.mylayoutbox = awful.widget.layoutbox(s)
+ s.mylayoutbox:buttons(gears.table.join(
+ awful.button({ }, 1, function () awful.layout.inc( 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 {
+ screen = s,
+ filter = awful.widget.taglist.filter.all,
+ buttons = taglist_buttons
+ }
+
+ -- Create a tasklist widget
+ s.mytasklist = awful.widget.tasklist {
+ screen = s,
+ filter = awful.widget.tasklist.filter.currenttags,
+ buttons = tasklist_buttons
+ }
+
+ -- Create the wibox
+ s.mywibox = awful.wibar({ position = "top", screen = s })
+
+ -- Add widgets to the wibox
+ s.mywibox:setup {
+ layout = wibox.layout.align.horizontal,
+ { -- Left widgets
+ layout = wibox.layout.fixed.horizontal,
+ mylauncher,
+ s.mytaglist,
+ s.mypromptbox,
+ },
+ s.mytasklist, -- Middle widget
+ { -- Right widgets
+ layout = wibox.layout.fixed.horizontal,
+ mykeyboardlayout,
+ wibox.widget.systray(),
+ mytextclock,
+ s.mylayoutbox,
+ },
+ }
+end)
+-- }}}
+
+-- {{{ Mouse bindings
+root.buttons(gears.table.join(
+ awful.button({ }, 3, function () mymainmenu:toggle() end),
+ awful.button({ }, 4, awful.tag.viewnext),
+ awful.button({ }, 5, awful.tag.viewprev)
+))
+-- }}}
+
+-- {{{ Key bindings
+globalkeys = gears.table.join(
+ awful.key({ modkey, }, "s", hotkeys_popup.show_help,
+ {description="show help", group="awesome"}),
+ 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"}),
+
+ awful.key({ modkey, }, "j",
+ function ()
+ awful.client.focus.byidx( 1)
+ end,
+ {description = "focus next by index", group = "client"}
+ ),
+ awful.key({ modkey, }, "k",
+ function ()
+ awful.client.focus.byidx(-1)
+ end,
+ {description = "focus previous by index", group = "client"}
+ ),
+ awful.key({ modkey, }, "w", function () 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"}),
+
+ -- 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({ modkey, }, "l", function () awful.tag.incmwfact( 0.05) end,
+ {description = "increase master width factor", group = "layout"}),
+ awful.key({ modkey, }, "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
+ c:emit_signal(
+ "request::activate", "key.unminimize", {raise = true}
+ )
+ end
+ end,
+ {description = "restore minimized", group = "client"}),
+
+ -- 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"}),
+ -- Menubar
+ awful.key({ modkey }, "p", function() menubar.show() end,
+ {description = "show the menubar", group = "launcher"})
+)
+
+clientkeys = gears.table.join(
+ 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 = "(un)maximize", group = "client"}),
+ awful.key({ modkey, "Control" }, "m",
+ function (c)
+ c.maximized_vertical = not c.maximized_vertical
+ c:raise()
+ end ,
+ {description = "(un)maximize vertically", group = "client"}),
+ awful.key({ modkey, "Shift" }, "m",
+ function (c)
+ c.maximized_horizontal = not c.maximized_horizontal
+ c:raise()
+ end ,
+ {description = "(un)maximize horizontally", group = "client"})
+)
+
+-- Bind all key numbers to tags.
+-- Be careful: we use keycodes to make it work on any keyboard layout.
+-- This should map on the top row of your keyboard, usually 1 to 9.
+for i = 1, 9 do
+ globalkeys = gears.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,
+ {description = "view tag #"..i, group = "tag"}),
+ -- 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,
+ {description = "toggle tag #" .. i, group = "tag"}),
+ -- 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,
+ {description = "move focused client to tag #"..i, group = "tag"}),
+ -- 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,
+ {description = "toggle focused client on tag #" .. i, group = "tag"})
+ )
+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
+ }
+ },
+
+ -- Floating clients.
+ { rule_any = {
+ instance = {
+ "DTA", -- Firefox addon DownThemAll.
+ "copyq", -- Includes session name in class.
+ "pinentry",
+ },
+ class = {
+ "Arandr",
+ "Blueman-manager",
+ "Gpick",
+ "Kruler",
+ "MessageWin", -- kalarm.
+ "Sxiv",
+ "Tor Browser", -- Needs a fixed window size to avoid fingerprinting by screen size.
+ "Wpa_gui",
+ "veromix",
+ "xtightvncviewer"},
+
+ -- Note that the name property shown in xprop might be set slightly after creation of the client
+ -- and the name shown there might not match defined rules here.
+ name = {
+ "Event Tester", -- xev.
+ },
+ role = {
+ "AlarmWindow", -- Thunderbird's calendar.
+ "ConfigManager", -- Thunderbird's about:config.
+ "pop-up", -- e.g. Google Chrome's (detached) Developer Tools.
+ }
+ }, properties = { floating = true }},
+
+ -- Add titlebars to normal clients and dialogs
+ { rule_any = {type = { "normal", "dialog" }
+ }, properties = { titlebars_enabled = true }
+ },
+
+ -- Set Firefox to always map on the tag named "2" on screen 1.
+ -- { rule = { class = "Firefox" },
+ -- properties = { screen = 1, tag = "2" } },
+}
+-- }}}
+
+-- {{{ 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)
+ -- buttons for the titlebar
+ local buttons = gears.table.join(
+ awful.button({ }, 1, function()
+ c:emit_signal("request::activate", "titlebar", {raise = true})
+ awful.mouse.client.move(c)
+ end),
+ awful.button({ }, 3, function()
+ c:emit_signal("request::activate", "titlebar", {raise = true})
+ awful.mouse.client.resize(c)
+ end)
+ )
+
+ awful.titlebar(c) : 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 = false})
+end)
+
+client.connect_signal("focus", function(c) c.border_color = beautiful.border_focus end)
+client.connect_signal("unfocus", function(c) c.border_color = beautiful.border_normal end)
+-- }}}
\ No newline at end of file
diff --git a/rsc/config/awesome/rc.lua b/rsc/config/awesome/rc.lua
new file mode 100644
index 0000000..1da777b
--- /dev/null
+++ b/rsc/config/awesome/rc.lua
@@ -0,0 +1,678 @@
+-- If LuaRocks is installed, make sure that packages installed through it are
+-- found (e.g. lgi). If LuaRocks is not installed, do nothing.
+pcall(require, "luarocks.loader")
+
+-- Standard awesome library
+local gears = require("gears")
+local awful = require("awful")
+require("awful.autofocus")
+-- Widget and layout library
+local wibox = require("wibox")
+-- Theme handling library
+local beautiful = require("beautiful")
+-- Notification library
+local naughty = require("naughty")
+local menubar = require("menubar")
+local hotkeys_popup = require("awful.hotkeys_popup")
+-- Enable hotkeys help widget for VIM and other apps
+-- when client with a matching name is opened:
+require("awful.hotkeys_popup.keys")
+local lain = require("lain")
+local theme = require("theme")
+local autostart = require("autostart")
+
+-- {{{ Error handling
+-- Check if awesome encountered an error during startup and fell back to
+-- another config (This code will only ever execute for the fallback config)
+if awesome.startup_errors then
+ naughty.notify({
+ preset = naughty.config.presets.critical,
+ title = "Oops, there were errors during startup!",
+ text = awesome.startup_errors
+ })
+end
+
+-- Handle runtime errors after startup
+do
+ local in_error = false
+ awesome.connect_signal("debug::error", function(err)
+ -- Make sure we don't go into an endless error loop
+ if in_error then
+ return
+ end
+ in_error = true
+
+ naughty.notify({
+ preset = naughty.config.presets.critical,
+ title = "Oops, an error happened!",
+ text = tostring(err)
+ })
+ in_error = false
+ end)
+end
+-- }}}
+
+-- {{{ Variable definitions
+-- Themes define colours, icons, font and wallpapers.
+beautiful.init(gears.filesystem.get_configuration_dir() .. "theme.lua")
+
+-- This is used later as the default terminal and editor to run.
+terminal = "alacritty"
+editor = os.getenv("EDITOR") or "nvim"
+editor_cmd = terminal .. " -e " .. editor
+
+-- Default modkey.
+-- Usually, Mod4 is the key with a logo between Control and Alt.
+-- If you do not like this or do not have such a key,
+-- I suggest you to remap Mod4 to another key using xmodmap or other tools.
+-- However, you can use another modifier like Mod1, but it may interact with others.
+modkey = "Mod4"
+
+-- Table of layouts to cover with awful.layout.inc, order matters.
+awful.layout.layouts = {awful.layout.suit.tile -- awful.layout.suit.spiral
+-- 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.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,
+}
+-- }}}
+
+-- {{{ Wibar
+-- Create a textclock widget
+local clockicon = wibox.widget.imagebox(theme.widget_clock)
+local clock = awful.widget.textclock(" %H:%M")
+
+local calendaricon = wibox.widget.imagebox(theme.widget_calendar)
+local calendar = awful.widget.textclock(" %a, %d. %b")
+
+local memicon = wibox.widget.imagebox(theme.widget_mem)
+local mem = lain.widget.mem {
+ settings = function()
+ widget:set_markup(" " .. string.format("%.2f", mem_now.used / 1024) .. "GiB")
+ end
+}
+
+local cpuicon = wibox.widget.imagebox(theme.widget_cpu)
+local cpu = lain.widget.cpu {
+ settings = function()
+ widget:set_markup(" " .. cpu_now.usage .. "%")
+ end
+}
+
+local baticon = wibox.widget.imagebox(theme.widget_bat)
+local bat = lain.widget.bat {
+ settings = function()
+ widget:set_markup(" " .. bat_now.perc .."%")
+ end
+}
+
+-- Create a wibox for each screen and add it
+local taglist_buttons = gears.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))
+
+local tasklist_buttons = gears.table.join(awful.button({}, 1, function(c)
+ if c == client.focus then
+ c.minimized = true
+ else
+ c:emit_signal("request::activate", "tasklist", {
+ raise = true
+ })
+ end
+end), awful.button({}, 3, function()
+ awful.menu.client_list({
+ theme = {
+ width = 250
+ }
+ })
+end), awful.button({}, 4, function()
+ awful.client.focus.byidx(1)
+end), awful.button({}, 5, function()
+ awful.client.focus.byidx(-1)
+end))
+
+local function set_wallpaper(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
+
+-- Re-set wallpaper when a screen's geometry changes (e.g. different resolution)
+screen.connect_signal("property::geometry", set_wallpaper)
+
+awful.screen.connect_for_each_screen(function(s)
+ -- Wallpaper
+ set_wallpaper(s)
+
+ -- Each screen has its own tag table.
+ awful.tag({"1", "2", "3", "4", "5", "6", "7", "8", "9"}, s, awful.layout.layouts[1])
+
+ -- Create a promptbox for each screen
+ s.mypromptbox = awful.widget.prompt()
+ -- Create an imagebox widget which will contain an icon indicating which layout we're using.
+ -- We need one layoutbox per screen.
+ s.mylayoutbox = awful.widget.layoutbox(s)
+ s.mylayoutbox:buttons(gears.table.join(awful.button({}, 1, function()
+ awful.layout.inc(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 {
+ screen = s,
+ filter = awful.widget.taglist.filter.all,
+ buttons = taglist_buttons
+ }
+
+ -- Create a tasklist widget
+ s.mytasklist = awful.widget.tasklist {
+ screen = s,
+ filter = awful.widget.tasklist.filter.currenttags,
+ buttons = tasklist_buttons
+ }
+
+ -- Create the wibox
+ s.mywibox = awful.wibar({
+ position = "top",
+ screen = s,
+ bg = theme.bg_normal,
+ fg = theme.fg_normal
+ })
+
+ local systray = wibox.widget.systray()
+ local tray = wibox.widget {
+ systray,
+ valign = 'center',
+ halign = 'center',
+ widget = wibox.container.place
+ }
+
+ local cpuiconContainer = wibox.widget {
+ cpuicon,
+ valign = 'center',
+ halign = 'center',
+ widget = wibox.container.place
+ }
+
+ local baticonContainer = wibox.widget {
+ baticon,
+ valign = 'center',
+ halign = 'center',
+ widget = wibox.container.place
+ }
+
+ local memiconContainer = wibox.widget {
+ memicon,
+ valign = 'center',
+ halign = 'center',
+ widget = wibox.container.place
+ }
+
+ local clockiconContainer = wibox.widget {
+ clockicon,
+ valign = 'center',
+ halign = 'center',
+ widget = wibox.container.place
+ }
+
+ local calendariconContainer = wibox.widget {
+ calendaricon,
+ valign = 'center',
+ halign = 'center',
+ widget = wibox.container.place
+ }
+
+ local separator = wibox.widget {
+ markup = ' | ',
+ align = 'center',
+ valign = 'center',
+ widget = wibox.widget.textbox
+ }
+
+ local l_separator = wibox.widget {
+ markup = ' [ ',
+ align = 'center',
+ valign = 'center',
+ widget = wibox.widget.textbox
+ }
+
+ local m_separator = wibox.widget {
+ markup = ' ][ ',
+ align = 'center',
+ valign = 'center',
+ widget = wibox.widget.textbox
+ }
+
+ local r_separator = wibox.widget {
+ markup = ' ] ',
+ align = 'center',
+ valign = 'center',
+ widget = wibox.widget.textbox
+ }
+
+ systray.base_size = s.mywibox.height * 0.6
+
+ clockicon.forced_height = s.mywibox.height * 0.7
+ clockicon.forced_width = s.mywibox.height * 0.7
+
+ calendaricon.forced_height = s.mywibox.height * 0.7
+ calendaricon.forced_width = s.mywibox.height * 0.7
+
+ memicon.forced_height = s.mywibox.height * 0.7
+ memicon.forced_width = s.mywibox.height * 0.7
+
+ cpuicon.forced_height = s.mywibox.height * 0.7
+ cpuicon.forced_width = s.mywibox.height * 0.7
+
+ baticon.forced_height = s.mywibox.height * 0.7
+ baticon.forced_width = s.mywibox.height * 0.7
+
+ -- Add widgets to the wibox
+ s.mywibox:setup{
+ layout = wibox.layout.align.horizontal,
+ { -- Left widgets
+ layout = wibox.layout.fixed.horizontal,
+ s.mytaglist,
+ s.mypromptbox
+ },
+ s.mytasklist, -- Middle widget
+ { -- Right widgets
+ layout = wibox.layout.fixed.horizontal,
+ tray,
+ separator,
+ memiconContainer,
+ mem,
+ separator,
+ cpuiconContainer,
+ cpu,
+ separator,
+ -- baticonContainer,
+ -- bat,
+ -- separator,
+ calendariconContainer,
+ calendar,
+ separator,
+ clockiconContainer,
+ clock,
+ separator
+ }
+ }
+end)
+
+-- }}}
+
+client.connect_signal("property::maximized", function(c)
+ if c.maximized then
+ c.maximized = false
+ end
+end)
+
+-- {{{ Key bindings
+globalkeys = gears.table.join(awful.key({modkey}, "x", function()
+ awful.util.spawn("/home/ghoscht/.config/rofi/powermenu/type-2/powermenu.sh")
+end, {
+ description = "Rofi powermenu",
+ group = "launcher"
+}), awful.key({modkey}, "space", function()
+ awful.util.spawn("/home/ghoscht/.config/rofi/launchers/type-2/launcher.sh")
+end, {
+ description = "Rofi application menu",
+ group = "launcher"
+}), awful.key({modkey}, "s", hotkeys_popup.show_help, {
+ description = "show help",
+ group = "awesome"
+}), 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"
+}), awful.key({modkey}, "j", function()
+ awful.client.focus.byidx(1)
+end, {
+ description = "focus next by index",
+ group = "client"
+}), awful.key({modkey}, "k", function()
+ awful.client.focus.byidx(-1)
+end, {
+ description = "focus previous by index",
+ group = "client"
+}), awful.key({modkey}, "w", function()
+ 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"
+}), -- Standard program
+awful.key({modkey}, "Return", function()
+ awful.spawn(terminal)
+end, {
+ description = "open a terminal",
+ group = "launcher"
+}),
+awful.key({modkey}, "l", function()
+ awful.spawn("librewolf")
+end, {
+ description = "open the browser",
+ group = "launcher"
+}),
+awful.key({modkey, "Shift"}, "l", function()
+ awful.spawn("librewolf --private-window")
+end, {
+ description = "open the browser",
+ 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({modkey}, "l", function()
+ awful.tag.incmwfact(0.05)
+end, {
+ description = "increase master width factor",
+ group = "layout"
+}), awful.key({modkey}, "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, "Control"}, "n", function()
+ local c = awful.client.restore()
+ -- Focus restored client
+ if c then
+ c:emit_signal("request::activate", "key.unminimize", {
+ raise = true
+ })
+ end
+end, {
+ description = "restore minimized",
+ group = "client"
+}))
+
+clientkeys = gears.table.join(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 = "(un)maximize",
+ group = "client"
+}), awful.key({modkey, "Control"}, "m", function(c)
+ c.maximized_vertical = not c.maximized_vertical
+ c:raise()
+end, {
+ description = "(un)maximize vertically",
+ group = "client"
+}), awful.key({modkey, "Shift"}, "m", function(c)
+ c.maximized_horizontal = not c.maximized_horizontal
+ c:raise()
+end, {
+ description = "(un)maximize horizontally",
+ group = "client"
+}))
+
+-- Bind all key numbers to tags.
+-- Be careful: we use keycodes to make it work on any keyboard layout.
+-- This should map on the top row of your keyboard, usually 1 to 9.
+for i = 1, 9 do
+ globalkeys = gears.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, {
+ description = "view tag #" .. i,
+ group = "tag"
+ }), -- 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, {
+ description = "toggle tag #" .. i,
+ group = "tag"
+ }), -- 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, {
+ description = "move focused client to tag #" .. i,
+ group = "tag"
+ }), -- 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, {
+ description = "toggle focused client on tag #" .. i,
+ group = "tag"
+ }))
+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
+ }
+}, -- Floating clients.
+{
+ rule_any = {
+ instance = {"DTA", -- Firefox addon DownThemAll.
+ "copyq", -- Includes session name in class.
+ "pinentry"},
+ class = {"Arandr", "Blueman-manager", "Gpick", "Kruler", "MessageWin", -- kalarm.
+ "Sxiv", "Tor Browser", -- Needs a fixed window size to avoid fingerprinting by screen size.
+ "Wpa_gui", "veromix", "xtightvncviewer"},
+
+ -- Note that the name property shown in xprop might be set slightly after creation of the client
+ -- and the name shown there might not match defined rules here.
+ name = {"Event Tester" -- xev.
+ },
+ role = {"AlarmWindow", -- Thunderbird's calendar.
+ "ConfigManager", -- Thunderbird's about:config.
+ "pop-up" -- e.g. Google Chrome's (detached) Developer Tools.
+ }
+ },
+ properties = {
+ floating = true
+ }
+} -- Set Firefox to always map on the tag named "2" on screen 1.
+-- { rule = { class = "Firefox" },
+-- properties = { screen = 1, tag = "2" } },
+}
+-- }}}
+
+-- {{{ 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)
+
+-- Enable sloppy focus, so that focus follows mouse.
+client.connect_signal("mouse::enter", function(c)
+ c:emit_signal("request::activate", "mouse_enter", {
+ raise = false
+ })
+end)
+
+client.connect_signal("focus", function(c)
+ c.border_color = beautiful.border_focus
+end)
+client.connect_signal("unfocus", function(c)
+ c.border_color = beautiful.border_normal
+end)
+-- }}}
+
+-- Autostart Applications
+autostart.exec(awful)
diff --git a/rsc/config/awesome/theme.lua b/rsc/config/awesome/theme.lua
new file mode 100644
index 0000000..6193620
--- /dev/null
+++ b/rsc/config/awesome/theme.lua
@@ -0,0 +1,96 @@
+--[[
+
+ Powerarrow Dark Awesome WM theme
+ github.com/lcpz
+
+--]]
+
+local gears = require("gears")
+local lain = require("lain")
+local awful = require("awful")
+local wibox = require("wibox")
+local dpi = require("beautiful.xresources").apply_dpi
+local colors = require("colors")
+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/"
+theme.wallpaper_dir = os.getenv("HOME") .. "/.wallpapers/"
+theme.wallpaper = theme.wallpaper_dir .. "/city.jpg"
+theme.font = "JetBrainsMono Nerd Font 9"
+theme.fg_normal = colors.text
+theme.fg_focus = colors.teal
+theme.fg_urgent = "#CC9393"
+theme.bg_normal = colors.base
+theme.bg_focus = colors.crust
+theme.bg_urgent = "#1e1e2e"
+theme.border_width = dpi(1)
+theme.border_normal = colors.surface0
+theme.border_focus = colors.teal
+theme.border_marked = "#CC9393"
+theme.tasklist_bg_focus = colors.base
+theme.titlebar_bg_focus = theme.bg_focus
+theme.titlebar_bg_normal = theme.bg_normal
+theme.titlebar_fg_focus = theme.fg_focus
+theme.menu_height = dpi(16)
+theme.menu_width = dpi(140)
+theme.systray_icon_spacing = 2
+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.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/custom/mem.png"
+theme.widget_clock = theme.dir .. "/icons/custom/clock.png"
+theme.widget_calendar = theme.dir .. "/icons/custom/calendar.png"
+theme.widget_cpu = theme.dir .. "/icons/custom/chip.png"
+theme.widget_bat = theme.dir .. "/icons/custom/bat.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_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.tasklist_plain_task_name = true
+theme.tasklist_disable_icon = true
+theme.useless_gap = dpi(7)
+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"
+
+return theme
diff --git a/rsc/config/flameshot/flameshot.ini b/rsc/config/flameshot/flameshot.ini
new file mode 100644
index 0000000..3535c6d
--- /dev/null
+++ b/rsc/config/flameshot/flameshot.ini
@@ -0,0 +1,15 @@
+[General]
+autoCloseIdleDaemon=true
+checkForUpdates=false
+contrastOpacity=188
+contrastUiColor=#ff0400
+copyAndCloseAfterUpload=false
+disabledTrayIcon=true
+savePath=/home/ghoscht/Pictures/screenshots
+savePathFixed=false
+showDesktopNotification=false
+showHelp=false
+showMagnifier=true
+showStartupLaunchMessage=false
+uiColor=#94e2d5
+uploadWithoutConfirmation=false
diff --git a/rsc/config/picom/picom.conf b/rsc/config/picom/picom.conf
new file mode 100644
index 0000000..bdf9af2
--- /dev/null
+++ b/rsc/config/picom/picom.conf
@@ -0,0 +1,524 @@
+#################################
+# Animations #
+#################################
+# requires https://github.com/jonaburg/picom
+# (These are also the default values)
+transition-length = 300
+transition-pow-x = 0.1
+transition-pow-y = 0.1
+transition-pow-w = 0.1
+transition-pow-h = 0.1
+size-transition = true
+
+
+#################################
+# Corners #
+#################################
+# requires: https://github.com/sdhand/compton or https://github.com/jonaburg/picom
+corner-radius = 10.0;
+rounded-corners-exclude = [
+ #"window_type = 'normal'",
+ "class_g = 'awesome'",
+ "class_g = 'URxvt'",
+ "class_g = 'XTerm'",
+ "class_g = 'kitty'",
+ #"class_g = 'Alacritty'",
+ "class_g = 'Polybar'",
+ "class_g = 'code-oss'",
+ #"class_g = 'TelegramDesktop'",
+ #"class_g = 'LibreWolf'",
+ "class_g = 'Thunderbird'"
+];
+round-borders = 1;
+round-borders-exclude = [
+ #"class_g = 'TelegramDesktop'",
+];
+
+#################################
+# Shadows #
+#################################
+
+
+# Enabled client-side shadows on windows. Note desktop windows
+# (windows with '_NET_WM_WINDOW_TYPE_DESKTOP') never get shadow,
+# unless explicitly requested using the wintypes option.
+#
+# shadow = false
+shadow = false;
+
+# The blur radius for shadows, in pixels. (defaults to 12)
+# shadow-radius = 12
+shadow-radius = 7;
+
+# The opacity of shadows. (0.0 - 1.0, defaults to 0.75)
+# shadow-opacity = .75
+
+# The left offset for shadows, in pixels. (defaults to -15)
+# shadow-offset-x = -15
+shadow-offset-x = -7;
+
+# The top offset for shadows, in pixels. (defaults to -15)
+# shadow-offset-y = -15
+shadow-offset-y = -7;
+
+# Avoid drawing shadows on dock/panel windows. This option is deprecated,
+# you should use the *wintypes* option in your config file instead.
+#
+# no-dock-shadow = false
+
+# Don't draw shadows on drag-and-drop windows. This option is deprecated,
+# you should use the *wintypes* option in your config file instead.
+#
+# no-dnd-shadow = false
+
+# Red color value of shadow (0.0 - 1.0, defaults to 0).
+# shadow-red = 0
+
+# Green color value of shadow (0.0 - 1.0, defaults to 0).
+# shadow-green = 0
+
+# Blue color value of shadow (0.0 - 1.0, defaults to 0).
+# shadow-blue = 0
+
+# Do not paint shadows on shaped windows. Note shaped windows
+# here means windows setting its shape through X Shape extension.
+# Those using ARGB background is beyond our control.
+# Deprecated, use
+# shadow-exclude = 'bounding_shaped'
+# or
+# shadow-exclude = 'bounding_shaped && !rounded_corners'
+# instead.
+#
+# shadow-ignore-shaped = ''
+
+# Specify a list of conditions of windows that should have no shadow.
+#
+# examples:
+# shadow-exclude = "n:e:Notification";
+#
+# shadow-exclude = []
+shadow-exclude = [
+ "name = 'Notification'",
+ "class_g = 'Conky'",
+ "class_g ?= 'Notify-osd'",
+ "class_g = 'Cairo-clock'",
+ "class_g = 'slop'",
+ "class_g = 'Polybar'",
+ "_GTK_FRAME_EXTENTS@:c"
+];
+
+# Specify a X geometry that describes the region in which shadow should not
+# be painted in, such as a dock window region. Use
+# shadow-exclude-reg = "x10+0+0"
+# for example, if the 10 pixels on the bottom of the screen should not have shadows painted on.
+#
+# shadow-exclude-reg = ""
+
+# Crop shadow of a window fully on a particular Xinerama screen to the screen.
+# xinerama-shadow-crop = false
+
+
+#################################
+# Fading #
+#################################
+
+
+# Fade windows in/out when opening/closing and when opacity changes,
+# unless no-fading-openclose is used.
+# fading = false
+fading = true;
+
+# Opacity change between steps while fading in. (0.01 - 1.0, defaults to 0.028)
+# fade-in-step = 0.028
+fade-in-step = 0.03;
+
+# Opacity change between steps while fading out. (0.01 - 1.0, defaults to 0.03)
+# fade-out-step = 0.03
+fade-out-step = 0.03;
+
+# The time between steps in fade step, in milliseconds. (> 0, defaults to 10)
+# fade-delta = 10
+
+# Specify a list of conditions of windows that should not be faded.
+# don't need this, we disable fading for all normal windows with wintypes: {}
+fade-exclude = [
+ "class_g = 'slop'" # maim
+]
+
+# Do not fade on window open/close.
+# no-fading-openclose = false
+
+# Do not fade destroyed ARGB windows with WM frame. Workaround of bugs in Openbox, Fluxbox, etc.
+# no-fading-destroyed-argb = false
+
+
+#################################
+# Transparency / Opacity #
+#################################
+
+
+# Opacity of inactive windows. (0.1 - 1.0, defaults to 1.0)
+# inactive-opacity = 1
+inactive-opacity = 1;
+
+# Opacity of window titlebars and borders. (0.1 - 1.0, disabled by default)
+# frame-opacity = 1.0
+frame-opacity = 1.0;
+
+# Default opacity for dropdown menus and popup menus. (0.0 - 1.0, defaults to 1.0)
+# menu-opacity = 1.0
+# menu-opacity is depreciated use dropdown-menu and popup-menu instead.
+
+#If using these 2 below change their values in line 510 & 511 aswell
+popup_menu = { opacity = 0.8; }
+dropdown_menu = { opacity = 0.8; }
+
+
+# Let inactive opacity set by -i override the '_NET_WM_OPACITY' values of windows.
+# inactive-opacity-override = true
+inactive-opacity-override = false;
+
+# Default opacity for active windows. (0.0 - 1.0, defaults to 1.0)
+active-opacity = 1.0;
+
+# Dim inactive windows. (0.0 - 1.0, defaults to 0.0)
+# inactive-dim = 0.0
+
+# Specify a list of conditions of windows that should always be considered focused.
+# focus-exclude = []
+focus-exclude = [
+ "class_g = 'Cairo-clock'",
+ "class_g = 'Bar'", # lemonbar
+ "class_g = 'slop'", # maim
+ "class_g = 'LibreWolf'",
+ "class_g = 'Steam'"
+];
+
+# Use fixed inactive dim value, instead of adjusting according to window opacity.
+inactive-dim-fixed = 1.0;
+
+# Specify a list of opacity rules, in the format `PERCENT:PATTERN`,
+# like `50:name *= "Firefox"`. picom-trans is recommended over this.
+# Note we don't make any guarantee about possible conflicts with other
+# programs that set '_NET_WM_WINDOW_OPACITY' on frame or client windows.
+# example:
+# opacity-rule = [ "80:class_g = 'URxvt'" ];
+#
+# opacity-rule = []
+opacity-rule = [
+ "100:fullscreen",
+ "80:class_g = 'Bar'", # lemonbar
+ "100:class_g = 'slop'", # maim
+ "100:class_g = 'XTerm'",
+ "100:class_g = 'URxvt'",
+ "100:class_g = 'kitty'",
+ "100:class_g = 'Alacritty'",
+ "80:class_g = 'Polybar'",
+ "100:class_g = 'code-oss'",
+ "100:class_g = 'Meld'",
+ "70:class_g = 'TelegramDesktop'",
+ "90:class_g = 'Joplin'",
+ "100:class_g = 'firefox'",
+ "100:class_g = 'Thunderbird'",
+ "100:class_g = 'LibreWolf'",
+ "100:class_g = 'Steam'"
+];
+
+
+#################################
+# Background-Blurring #
+#################################
+
+
+# Parameters for background blurring, see the *BLUR* section for more information.
+blur-method = "dual_kawase";
+blur-strength = 15;
+
+# blur-size = 12
+#
+# blur-deviation = false
+
+# Blur background of semi-transparent / ARGB windows.
+# Bad in performance, with driver-dependent behavior.
+# The name of the switch may change without prior notifications.
+#
+# blur-background = true;
+
+# Blur background of windows when the window frame is not opaque.
+# Implies:
+# blur-background
+# Bad in performance, with driver-dependent behavior. The name may change.
+#
+# blur-background-frame = false;
+
+
+# Use fixed blur strength rather than adjusting according to window opacity.
+# blur-background-fixed = false;
+
+
+# Specify the blur convolution kernel, with the following format:
+# example:
+# blur-kern = "5,5,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1";
+#
+# blur-kern = ''
+# blur-kern = "3x3box";
+
+blur: {
+ # requires: https://github.com/ibhagwan/picom
+ method = "kawase";
+ #method = "kernel";
+ strength = 7;
+ # deviation = 1.0;
+ # kernel = "11x11gaussian";
+ background = false;
+ background-frame = false;
+ background-fixed = false;
+ kern = "3x3box";
+}
+
+# Exclude conditions for background blur.
+blur-background-exclude = [
+ "class_g = 'LibreWolf'",
+ #"window_type = 'dock'",
+ "window_type = 'desktop'",
+ # "class_g = 'Rofi'",
+ #"class_g = 'URxvt'",
+ #
+ # prevents picom from blurring the background
+ # when taking selection screenshot with `main`
+ # https://github.com/naelstrof/maim/issues/130
+ "class_g = 'slop'",
+ "_GTK_FRAME_EXTENTS@:c"
+];
+
+
+#################################
+# General Settings #
+#################################
+
+# Daemonize process. Fork to background after initialization. Causes issues with certain (badly-written) drivers.
+# daemon = false
+
+# Specify the backend to use: `xrender`, `glx`, or `xr_glx_hybrid`.
+# `xrender` is the default one.
+#
+experimental-backends = true;
+backend = "glx";
+#backend = "xrender";
+
+
+# Enable/disable VSync.
+# vsync = false
+vsync = true
+
+# Enable remote control via D-Bus. See the *D-BUS API* section below for more details.
+# dbus = false
+
+# Try to detect WM windows (a non-override-redirect window with no
+# child that has 'WM_STATE') and mark them as active.
+#
+# mark-wmwin-focused = false
+mark-wmwin-focused = true;
+
+# Mark override-redirect windows that doesn't have a child window with 'WM_STATE' focused.
+# mark-ovredir-focused = false
+mark-ovredir-focused = true;
+
+# Try to detect windows with rounded corners and don't consider them
+# shaped windows. The accuracy is not very high, unfortunately.
+#
+# detect-rounded-corners = false
+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.
+#
+# detect-client-opacity = false
+detect-client-opacity = true;
+
+# Specify refresh rate of the screen. If not specified or 0, picom will
+# try detecting this with X RandR extension.
+#
+# refresh-rate = 60
+refresh-rate = 0
+
+# Limit picom 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 =
+
+# Use EWMH '_NET_ACTIVE_WINDOW' to determine currently focused window,
+# rather than listening to 'FocusIn'/'FocusOut' event. Might have more accuracy,
+# provided that the WM supports it.
+#
+# use-ewmh-active-win = false
+
+# Unredirect all windows if a full-screen opaque window is detected,
+# to maximize performance for full-screen windows. Known to cause flickering
+# when redirecting/unredirecting windows. paint-on-overlay may make the flickering less obvious.
+#
+unredir-if-possible = true;
+
+# Delay before unredirecting the window, in milliseconds. Defaults to 0.
+# unredir-if-possible-delay = 0
+
+# Conditions of windows that shouldn't be considered full-screen for unredirecting screen.
+unredir-if-possible-exclude = [
+ "class_g = 'looking-glass-client' && !focused",
+ "class_g = 'LibreWolf'"
+];
+
+# Use 'WM_TRANSIENT_FOR' to group windows, and consider windows
+# in the same group focused at the same time.
+#
+# detect-transient = false
+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 = false
+detect-client-leader = true
+
+# Resize damaged region by a specific number of pixels.
+# A positive value enlarges it while a negative one shrinks it.
+# If the value is positive, those additional pixels will not be actually painted
+# to screen, only used in blur calculation, and such. (Due to technical limitations,
+# with use-damage, those pixels will still be incorrectly painted to screen.)
+# Primarily used to fix the line corruption issues of blur,
+# in which case you should use the blur radius value here
+# (e.g. with a 3x3 kernel, you should use `--resize-damage 1`,
+# with a 5x5 one you use `--resize-damage 2`, and so on).
+# May or may not work with *--glx-no-stencil*. Shrinking doesn't function correctly.
+#
+# resize-damage = 1
+
+# Specify a list of conditions of windows that should be painted with inverted color.
+# Resource-hogging, and is not well tested.
+#
+# invert-color-include = []
+
+# GLX backend: Avoid using stencil buffer, useful if you don't have a stencil buffer.
+# Might cause incorrect opacity when rendering transparent content (but never
+# practically happened) and may not work with blur-background.
+# My tests show a 15% performance boost. Recommended.
+#
+# glx-no-stencil = false
+
+# 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, xf86-video-intel, etc.).
+# Recommended if it works.
+#
+# glx-no-rebind-pixmap = false
+
+# Disable the use of damage information.
+# This cause the whole screen to be redrawn everytime, instead of the part of the screen
+# has actually changed. Potentially degrades the performance, but might fix some artifacts.
+# The opposing option is use-damage
+#
+# no-use-damage = false
+#use-damage = true (Causing Weird Black semi opaque rectangles when terminal is opened)
+#Changing use-damage to false fixes the problem
+use-damage = false
+
+# Use X Sync fence to sync clients' draw calls, to make sure all draw
+# calls are finished before picom starts drawing. Needed on nvidia-drivers
+# with GLX backend for some users.
+#
+# xrender-sync-fence = false
+
+# GLX backend: Use specified GLSL fragment shader for rendering window contents.
+# See `compton-default-fshader-win.glsl` and `compton-fake-transparency-fshader-win.glsl`
+# in the source tree for examples.
+#
+# glx-fshader-win = ''
+
+# Force all windows to be painted with blending. Useful if you
+# have a glx-fshader-win that could turn opaque pixels transparent.
+#
+# force-win-blend = false
+
+# Do not use EWMH to detect fullscreen windows.
+# Reverts to checking if a window is fullscreen based only on its size and coordinates.
+#
+# no-ewmh-fullscreen = false
+
+# Dimming bright windows so their brightness doesn't exceed this set value.
+# Brightness of a window is estimated by averaging all pixels in the window,
+# so this could comes with a performance hit.
+# Setting this to 1.0 disables this behaviour. Requires --use-damage to be disabled. (default: 1.0)
+#
+# max-brightness = 1.0
+
+# Make transparent windows clip other windows like non-transparent windows do,
+# instead of blending on top of them.
+#
+# transparent-clipping = false
+
+# Set the log level. Possible values are:
+# "trace", "debug", "info", "warn", "error"
+# in increasing level of importance. Case doesn't matter.
+# If using the "TRACE" log level, it's better to log into a file
+# using *--log-file*, since it can generate a huge stream of logs.
+#
+# log-level = "debug"
+log-level = "info";
+
+# Set the log file.
+# If *--log-file* is never specified, logs will be written to stderr.
+# Otherwise, logs will to written to the given file, though some of the early
+# logs might still be written to the stderr.
+# When setting this option from the config file, it is recommended to use an absolute path.
+#
+# log-file = '/path/to/your/log/file'
+
+# Show all X errors (for debugging)
+# show-all-xerrors = false
+
+# Write process ID to a file.
+# write-pid-path = '/path/to/your/log/file'
+
+# Window type settings
+#
+# 'WINDOW_TYPE' is one of the 15 window types defined in EWMH standard:
+# "unknown", "desktop", "dock", "toolbar", "menu", "utility",
+# "splash", "dialog", "normal", "dropdown_menu", "popup_menu",
+# "tooltip", "notification", "combo", and "dnd".
+#
+# Following per window-type options are available: ::
+#
+# fade, shadow:::
+# Controls window-type-specific shadow and fade settings.
+#
+# opacity:::
+# Controls default opacity of the window type.
+#
+# focus:::
+# Controls whether the window of this type is to be always considered focused.
+# (By default, all window types except "normal" and "dialog" has this on.)
+#
+# full-shadow:::
+# Controls whether shadow is drawn under the parts of the window that you
+# normally won't be able to see. Useful when the window has parts of it
+# transparent, and you want shadows in those areas.
+#
+# redir-ignore:::
+# Controls whether this type of windows should cause screen to become
+# redirected again after been unredirected. If you have unredir-if-possible
+# set, and doesn't want certain window to cause unnecessary screen redirection,
+# you can set this to `true`.
+#
+wintypes:
+{
+ normal = { fade = false; shadow = false; }
+ tooltip = { fade = true; shadow = true; opacity = 0.75; focus = true; full-shadow = false; };
+ dock = { shadow = false; }
+ dnd = { shadow = false; }
+ popup_menu = { opacity = 0.8; }
+ dropdown_menu = { opacity = 0.8; }
+};