From 2dc212d04f251b6cb9ef7b32e557ecd671add79e Mon Sep 17 00:00:00 2001 From: Joseph Ferano Date: Wed, 8 Nov 2023 13:00:27 +0700 Subject: [PATCH] Switching to qtile X11, initial config setup to my liking --- .config/qtile/autostart.sh | 3 + .config/qtile/config.py | 321 +++++++++++++++++++++++++++++++++++++ 2 files changed, 324 insertions(+) create mode 100755 .config/qtile/autostart.sh create mode 100644 .config/qtile/config.py diff --git a/.config/qtile/autostart.sh b/.config/qtile/autostart.sh new file mode 100755 index 0000000..86431e5 --- /dev/null +++ b/.config/qtile/autostart.sh @@ -0,0 +1,3 @@ +#! /usr/bin/bash +xset r rate 270 60 +picom -b diff --git a/.config/qtile/config.py b/.config/qtile/config.py new file mode 100644 index 0000000..0b754f4 --- /dev/null +++ b/.config/qtile/config.py @@ -0,0 +1,321 @@ +# Copyright (c) 2010 Aldo Cortesi +# Copyright (c) 2010, 2014 dequis +# Copyright (c) 2012 Randall Ma +# Copyright (c) 2012-2014 Tycho Andersen +# Copyright (c) 2012 Craig Barnes +# Copyright (c) 2013 horsik +# Copyright (c) 2013 Tao Sauvage +# +# 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. +import os +import subprocess +import re +import random +from datetime import datetime +from Xlib import display as xdisplay + +from typing import Callable + +from libqtile import bar, layout, widget +from libqtile.config import Click, Drag, Group, Key, Match, Screen +from libqtile.lazy import lazy +from libqtile.utils import guess_terminal +from libqtile.log_utils import logger +from libqtile import hook + +@hook.subscribe.startup_once +def autostart(): + home = os.path.expanduser('~/.config/qtile/autostart.sh') + subprocess.Popen([home]) + +mod = "mod4" +alt = "mod1" +terminal = "kitty" +kbd_lights = ["off", "low", "med", "high"] +current_kbd_light = [0] +def kbd_brightness_up(qtile): + idx = current_kbd_light[0] + current_kbd_light[0] = min(len(kbd_lights) - 1, idx+1) + value = kbd_lights[current_kbd_light[0]] + qtile.spawn(f"asusctl -k {value}") + +def kbd_brightness_down(qtile): + idx = current_kbd_light[0] + current_kbd_light[0] = max(0, idx-1) + value = kbd_lights[current_kbd_light[0]] + qtile.spawn(f"asusctl -k {value}") + +keys = [ + # A list of available commands that can be bound to keys can be found + # at https://docs.qtile.org/en/latest/manual/config/lazy.html + # Switch between windows + # .when(wm_class != 'kitty') + # Key([mod], "h", lazy.function(move_focus, direction='left'), desc="Move focus to left"), + Key([mod], "h", lazy.layout.left().when(focused=Match(wm_class=re.compile('(?![Ee]macs)'))), desc="Move focus to left"), + Key([mod], "j", lazy.layout.down().when(focused=Match(wm_class=re.compile('(?![Ee]macs)'))), desc="Move focus to left"), + Key([mod], "k", lazy.layout.up().when(focused=Match(wm_class=re.compile('(?![Ee]macs)'))), desc="Move focus to left"), + Key([mod], "l", lazy.layout.right().when(focused=Match(wm_class=re.compile('(?![Ee]macs)'))), desc="Move focus to left"), + Key([alt], "tab", lazy.layout.next(), desc="Move window focus to other window"), + Key([mod], "tab", lazy.screen.toggle_group(), desc="Toggle Group"), + Key([mod], "space", lazy.spawn("rofi -show drun"), desc="Open Rofi"), + + # Move windows between left/right columns or move up/down in current stack. + # Moving out of range in Columns layout will create new column. + Key([mod, "shift"], "h", lazy.layout.shuffle_left(), desc="Move window to the left"), + Key([mod, "shift"], "l", lazy.layout.shuffle_right(), desc="Move window to the right"), + Key([mod, "shift"], "j", lazy.layout.shuffle_down(), desc="Move window down"), + Key([mod, "shift"], "k", lazy.layout.shuffle_up(), desc="Move window up"), + # Grow windows. If current window is on the edge of screen and direction + # will be to screen edge - window would shrink. + Key([mod, "control"], "h", lazy.layout.grow_left(), desc="Grow window to the left"), + Key([mod, "control"], "l", lazy.layout.grow_right(), desc="Grow window to the right"), + Key([mod, "control"], "j", lazy.layout.grow_down(), desc="Grow window down"), + Key([mod, "control"], "k", lazy.layout.grow_up(), desc="Grow window up"), + + # Key([mod], "d", lazy.groups.use_previous_layout(), desc="Move focus to left"), + # Key([mod], "a", lazy.layout.use_next_layout(), desc="Move focus to left"), + + # TODO: Figure out another binding for this + # Key([mod], "n", lazy.layout.normalize(), desc="Reset all window sizes"), + + # Toggle between split and unsplit sides of stack. + # Split = all windows displayed + # Unsplit = 1 window displayed, like Max layout, but still with + # multiple stack panes + Key( + [mod, "shift"], + "Return", + lazy.layout.toggle_split(), + desc="Toggle between split and unsplit sides of stack", + ), + Key([mod], "Return", lazy.spawn(terminal), desc="Launch terminal"), + # Toggle between different layouts as defined below + Key([mod], "c", lazy.window.center(), desc="Center focused window"), + # Key([mod], "Tab", lazy.next_layout(), desc="Toggle between layouts"), + Key([mod], "q", lazy.window.kill(), desc="Kill focused window"), + Key( + [mod], + "f", + lazy.window.toggle_fullscreen(), + desc="Toggle fullscreen on the focused window", + ), + Key([mod], "t", lazy.window.toggle_floating(), desc="Toggle floating on the focused window"), + Key([mod, "control"], "r", lazy.reload_config(), desc="Reload the config"), + Key([mod, "control"], "q", lazy.shutdown(), desc="Shutdown Qtile"), + Key([mod], "r", lazy.spawncmd(), desc="Spawn a command using a prompt widget"), + Key([], "XF86AudioLowerVolume", lazy.spawn("amixer sset Master 5%-")), + Key([], "XF86AudioRaiseVolume", lazy.spawn("amixer sset Master 5%+")), + Key([], "XF86Launch1", lazy.spawn("playerctl play-pause")), + Key([], "XF86AudioPlay", lazy.spawn("playerctl play-pause")), + Key([], "XF86MonBrightnessDown", lazy.spawn("brightnessctl set 5%-")), + Key([], "XF86MonBrightnessUp", lazy.spawn("brightnessctl set +5%")), + # This uses discrete values + Key([], "XF86KbdBrightnessDown", lazy.function(kbd_brightness_down)), + Key([], "XF86KbdBrightnessUp", lazy.function(kbd_brightness_up)), +] + +pape_dir = '/home/joe/Pictures/Wallpapers/' +papes = [os.path.join(pape_dir, p) for p in os.listdir(pape_dir)] +pape = random.choice(papes) + +def get_num_monitors(): + number_of_monitors = 0 + try: + x_display = xdisplay.Display() + resources = x_display.screen().root.xrandr_get_screen_resources() + + for output in resources.outputs: + monitor = x_display.xrandr_get_output_info( + output, resources.config_timestamp) + + preferred = False + if hasattr(monitor, "preferred"): + preferred = monitor.preferred + elif hasattr(monitor, "num_preferred"): + preferred = monitor.num_preferred + if preferred: + number_of_monitors += 1 + except Exception as e: + # always setup at least one monitor + logger.warning(e) + return 1 + else: + return number_of_monitors + +group_box = widget.GroupBox() +current_layout = widget.CurrentLayout() +prompt = widget.Prompt() +window_name = widget.WindowName() +window_name2 = widget.WindowName() +chord = widget.Chord(chords_colors={"launch": ("#ff0000", "#ffffff"),}, name_transform=lambda name: name.upper()) +systray = widget.Systray() +battery = widget.Battery() +clock = widget.Clock(format="%Y-%m-%d %a %I:%M %p") +volume = widget.Volume() + +screens = [ + Screen( + wallpaper=pape, + wallpaper_mode='fill', + top=bar.Bar( + [ + group_box, current_layout, prompt, window_name, chord, + systray, volume, battery, clock, + ], + 24, + # border_width=[2, 0, 2, 0], # Draw top and bottom borders + # border_color=["ff00ff", "000000", "ff00ff", "000000"] # Borders are magenta + ), + ), +] + +groups = [Group(i) for i in "123456789"] +num_monitors = get_num_monitors() +logger.warning(num_monitors) +if num_monitors > 1: + side_screen = Screen( + wallpaper=pape, + wallpaper_mode='fill', + # bottom=bar.Bar( + # [ + # # group_box, current_layout, prompt, window_name2, chord, volume, battery, + # ], + # 24, + # # border_width=[2, 0, 2, 0], # Draw top and bottom borders + # # border_color=["ff00ff", "000000", "ff00ff", "000000"] # Borders are magenta + # ), + ) + screens.insert(0, side_screen) + groups[0].screen_affinity = 0 + for g in groups[1:]: + g.screen_affinity = 1 + +groups[0].label = '🎵' + + +# groups.insert(0, Group('1', screen_affinity=0, label='🎵')) + +# This likely won't work when we only have 1 monitor +def go_to_group(name: str) -> Callable: + def _inner(qtile) -> None: + if num_monitors > 1: + if name == '1': + qtile.focus_screen(0) + else: + qtile.focus_screen(1) + qtile.groups_map[name].toscreen() + return _inner + +for i in groups: + keys.extend( + [ + # mod1 + letter of group = switch to group + Key( + [mod], + i.name, + # lazy.group[i.name].toscreen(), + lazy.function(go_to_group(i.name)), + desc="Switch to group {}".format(i.name), + ), + # mod1 + shift + letter of group = switch to & move focused window to group + Key( + [mod, "shift"], + i.name, + lazy.window.togroup(i.name, switch_group=True), + desc="Switch to & move focused window to group {}".format(i.name), + ), + # Or, use below if you prefer not to switch to that group. + # # mod1 + shift + letter of group = move focused window to group + # Key([mod, "shift"], i.name, lazy.window.togroup(i.name), + # desc="move focused window to group {}".format(i.name)), + ] + ) + +# keys.append(Key([mod], 'm', lazy.function(go_to_music), desc="Music")) + +layouts = [ + layout.Max(), + layout.Columns(border_focus_stack=["#d75f5f", "#8f3d3d"], border_width=4), + # Try more layouts by unleashing below layouts. + # layout.Stack(num_stacks=2), + # layout.Bsp(), + # layout.Matrix(), + # layout.MonadTall(), + # layout.MonadWide(), + # layout.RatioTile(), + # layout.Tile(), + # layout.TreeTab(), + # layout.VerticalTile(), + # layout.Zoomy(), +] + +widget_defaults = dict( + font="JetBrains Mono", + fontsize=12, + padding=5, +) + +# Drag floating layouts. +mouse = [ + Drag([mod], "Button1", lazy.window.set_position_floating(), start=lazy.window.get_position()), + Drag([mod], "Button3", lazy.window.set_size_floating(), start=lazy.window.get_size()), + Click([mod], "Button2", lazy.window.bring_to_front()), +] + +dgroups_key_binder = None +dgroups_app_rules = [] # type: list +follow_mouse_focus = True +bring_front_click = False +floats_kept_above = True +cursor_warp = False +floating_layout = layout.Floating( + float_rules=[ + # Run the utility of `xprop` to see the wm class and name of an X client. + *layout.Floating.default_float_rules, + Match(wm_class="pygame"), + Match(wm_class="confirmreset"), # gitk + Match(wm_class="makebranch"), # gitk + Match(wm_class="maketag"), # gitk + Match(wm_class="ssh-askpass"), # ssh-askpass + Match(title="branchdialog"), # gitk + Match(title="pinentry"), # GPG key password entry + ] +) +auto_fullscreen = True +focus_on_window_activation = "smart" +reconfigure_screens = True + +# If things like steam games want to auto-minimize themselves when losing +# focus, should we respect this or not? +auto_minimize = True + +# When using the Wayland backend, this can be used to configure input devices. +wl_input_rules = None + +# XXX: Gasp! We're lying here. In fact, nobody really uses or cares about this +# string besides java UI toolkits; you can see several discussions on the +# mailing lists, GitHub issues, and other WM documentation that suggest setting +# this string if your java app doesn't work correctly. We may as well just lie +# and say that we're a working one by default. +# +# We choose LG3D to maximize irony: it is a 3D non-reparenting WM written in +# java that happens to be on java's whitelist. +wmname = "LG3D" + +logger.warning('Loaded: ' + str(datetime.now()))