import re import os import subprocess import psutil from libqtile import layout, bar, qtile from libqtile.backend.base import Window from libqtile.core.manager import Qtile from libqtile.config import Click, Drag, Group, KeyChord, EzKey, EzKeyChord, Match, Screen, ScratchPad, DropDown, Key from libqtile.lazy import lazy from libqtile.utils import guess_terminal from libqtile import hook from libqtile.log_utils import logger from libqtile.backend.wayland import InputConfig import qtile_extras.widget as widget from qtile_extras.widget.decorations import BorderDecoration, PowerLineDecoration, RectDecoration from tasklist import TaskList from mpris2widget import Mpris2 from bluetooth import Bluetooth from nixenvironment import setupLocation, configLocation colors = { 'primary': '51afef', 'active': '8babf0', 'inactive': '555e60', 'secondary': '55eddc', 'background_primary': '222223', 'background_secondary': '444444', 'urgent': 'c45500', 'white': 'd5d5d5', 'grey': '737373', 'black': '121212', } # ##################################### # Utility functions @lazy.function def focus_window_by_class(qtile: Qtile, wmclass: str): match = Match(wm_class=wmclass) windows = [w for w in qtile.windows_map.values() if isinstance(w, Window) and match.compare(w)] if len(windows) == 0: return window = windows[0] group = window.group group.toscreen() group.focus(window) @lazy.function def go_to_group(qtile: Qtile, group_name: str, switch_monitor: bool = False): found = False current_group = qtile.current_group if group_name == current_group.name: return if switch_monitor: for screen in qtile.screens: if screen.group.name == group_name: qtile.focus_screen(screen.index) found = True break qtile.groups_map[group_name].toscreen() for window in current_group.windows: if window.fullscreen: window.toggle_fullscreen() if not switch_monitor or not found: window: Window for window in qtile.groups_map[group_name].windows: if window.fullscreen: window.toggle_fullscreen() # expands list of keys with the rest of regular keys, # mainly usable for KeyChords, where you want any other key # to exit the key chord instead. def expand_with_rest_keys(keys: list[EzKey], global_prefix: str) -> list[EzKey]: all_keys = ['', '', ''] + [chr(c) for c in range(ord('a'), ord('z') + 1)] prefixes = ['', 'M-', 'M-S-', 'M-C-', 'C-', 'S-'] for prefix in prefixes: for potentially_add_key in all_keys: potentially_add_key = prefix + potentially_add_key if potentially_add_key == global_prefix: continue found = False for existing_key in keys: if existing_key.key.lower() == potentially_add_key: found = True break if not found: keys.append(EzKey(potentially_add_key, lazy.spawn(f'notify-send "Not registered key {global_prefix} {potentially_add_key}"'))) return keys # ##################################### # Environment mod = 'mod4' terminal = 'alacritty' layout_theme = { 'border_focus': colors['active'], 'border_normal': colors['inactive'], 'border_width': 1, 'margin': 3, } layouts = [ layout.MonadTall(**layout_theme), layout.Max(**layout_theme), layout.MonadWide(**layout_theme), ] widget_defaults = dict( font = 'Roboto Bold', fontsize = 13, padding = 3, background = colors['background_primary'], foreground = colors['white'], ) extension_defaults = widget_defaults.copy() powerline = { 'decorations': [ PowerLineDecoration(path = 'forward_slash') ] } def create_top_bar(systray = False): widgets = [ widget.Sep(padding = 5, size_percent = 0, background = colors['background_secondary']), widget.CurrentScreen( active_text = 'I', active_color = colors['active'], padding = 3, fontsize = 16, background = colors['background_secondary'], ), widget.GroupBox( markup = False, highlight_method = 'line', rounded = False, margin_x = 2, disable_drag = True, use_mouse_wheel = True, active = colors['white'], inactive = colors['grey'], urgent_alert_method = 'line', urgent_border = colors['urgent'], this_current_screen_border = colors['active'], this_screen_border = colors['secondary'], other_screen_border = colors['inactive'], other_current_screen_border = '6989c0', background = colors['background_secondary'], ), widget.CurrentScreen( active_text = 'I', active_color = colors['active'], padding = 3, fontsize = 16, background = colors['background_secondary'], decorations = [ PowerLineDecoration(path = 'forward_slash'), ], ), widget.Sep( linewidth=2, size_percent=0, padding=5, ), widget.Prompt(), widget.WindowName( foreground = colors['primary'], width = bar.CALCULATED, padding = 10, empty_group_string = 'Desktop', max_chars = 160, decorations = [ RectDecoration( colour = colors['black'], radius = 0, padding_y = 4, padding_x = 0, filled = True, clip = True, ), ], ), widget.Spacer(), widget.Chord( padding = 15, decorations = [ RectDecoration( colour = colors['black'], radius = 0, padding_y = 4, padding_x = 6, filled = True, clip = True, ), ] ), widget.Net( interface = 'enp24s0', prefix='M', format = '{down:6.2f} {down_suffix:<2}↓↑{up:6.2f} {up_suffix:<2}', background = colors['background_secondary'], **powerline, ), widget.Memory( format = '{MemFree: .0f}{mm}', fmt = '{} free', **powerline, ), widget.CPU( format = '{load_percent} %', fmt = ' {}', background = colors['background_secondary'], **powerline, ), widget.DF( update_interval = 60, partition = '/', #format = '[{p}] {uf}{m} ({r:.0f}%)', format = '{uf}{m} free', fmt = ' {}', visible_on_warn = False, **powerline, ), widget.GenPollText( func = lambda: subprocess.check_output(['xkblayout-state', 'print', '%s']).decode('utf-8').upper(), fmt = '⌨ {}', update_interval = 0.5, **powerline, ), widget.Clock( timezone='Europe/Prague', foreground = colors['primary'], format='%A, %B %d - %H:%M:%S', background = colors['background_secondary'], **powerline ), widget.Volume( fmt = '🕫 {}', ), widget.Sep( foreground = colors['background_secondary'], size_percent = 70, linewidth = 3, ), Bluetooth( hci = '/dev_88_C9_E8_49_93_16', format_connected = ' {battery} %', format_disconnected = ' Disconnected', format_unpowered = '' ), ] if systray: widgets.append(widget.Sep( foreground = colors['background_secondary'], size_percent = 70, linewidth = 2, )) widgets.append(widget.Systray()) widgets.append(widget.Sep(padding = 5, size_percent = 0)) return bar.Bar(widgets, 30) def create_bottom_bar(): return bar.Bar([ TaskList( parse_text = lambda text : re.split(' [–—-] ', text)[-1], highlight_method = 'line', txt_floating = '🗗 ', txt_maximized = '🗖 ', txt_minimized = '🗕 ', borderwidth = 3, ), widget.Spacer(), Mpris2( display_metadata = ['xesam:title'], playerfilter = '.*Firefox.*', paused_text = '', #'  {track}', playing_text = ' {track}', padding = 10, decorations = [ RectDecoration( colour = colors['black'], radius = 0, padding_y = 4, padding_x = 5, filled = True, clip = True, ), ], ), Mpris2( display_metadata = ['xesam:title', 'xesam:artist'], objname = 'org.mpris.MediaPlayer2.spotify', paused_text = '', #'  {track}', playing_text = ' {track}', # '  {track}', padding = 10, decorations = [ RectDecoration( colour = colors['black'], radius = 0, padding_y = 4, padding_x = 5, filled = True, clip = True, ), ], ), widget.Sep( size_percent = 0, padding = 5, **powerline, ), widget.Wttr( location = {'Odolena_Voda': ''}, format = '%t %c', background = colors['background_secondary'], **powerline, ), ], 30) def init_screen(top_bar, wallpaper, index): return Screen(top=top_bar, bottom=create_bottom_bar(), wallpaper=wallpaper, x=1920*index, y=0, width=1920, height=1080) screens = [ init_screen(create_top_bar(), '/usr/share/backgrounds/dark futuristic city 3.png', 0), init_screen(create_top_bar(systray = True), '/usr/share/backgrounds/dark futuristic city č.png', 1), init_screen(create_top_bar(), '/usr/share/backgrounds/synthwave futuristic city.png', 2), ] # Keys keys = [] # up, down, main navigation keys.extend([ EzKey('M-j', lazy.layout.down(), desc = 'Move focus down'), EzKey('M-k', lazy.layout.up(), desc = 'Move focus up'), EzKey('M-S-j', lazy.layout.shuffle_down(), desc = 'Move focus down'), EzKey('M-S-k', lazy.layout.shuffle_up(), desc = 'Move focus up'), EzKey('M-C-h', lazy.layout.shrink_main(), desc = 'Move focus down'), EzKey('M-C-l', lazy.layout.grow_main(), desc = 'Move focus up'), EzKey('M-m', lazy.layout.focus_first(), desc = 'Focus main window'), EzKey('M-u', lazy.next_urgent(), desc = 'Focus urgent window'), ]) keys.extend([ EzKey('M-n', lazy.layout.normalize()), EzKey('M-t', lazy.window.disable_floating()), EzKey('M-f', lazy.window.toggle_fullscreen()), EzKey('M-', lazy.layout.swap_main()), EzKey('M-', lazy.next_layout()), EzKey('M-S-', lazy.to_layout_index(0), desc = 'Default layout'), ]) # Spwning programs keys.extend([ EzKey('M-', lazy.spawn('rofi -show drun')), EzKey('A-', lazy.spawn('rofi -show windowcd -modi window,windowcd')), EzKey('M-S-', lazy.spawn('rofi -show window -modi window,windowcd')), EzKey('M-S-', lazy.spawn(terminal)), ]) keys.extend([ # social navigation EzKeyChord('M-a', expand_with_rest_keys([ EzKey('b', focus_window_by_class('discord')), EzKey('n', focus_window_by_class('Cinny')), EzKey('m', focus_window_by_class('Cinny')), # notifications EzKey('l', lazy.spawn(f'{setupLocation}/scripts/notifications/clear-popups.sh')), EzKey('p', lazy.spawn(f'{setupLocation}/scripts/notifications/pause.sh')), EzKey('u', lazy.spawn(f'{setupLocation}/scripts/notifications/unpause.sh')), EzKey('t', lazy.spawn(f'{setupLocation}/scripts/notifications/show-center.sh')), EzKey('e', lazy.spawn('emacsclient -c')), ], 'M-a'), name = 'a') ]) keys.extend([ EzKeyChord('M-s', expand_with_rest_keys([ EzKey('e', lazy.spawn('emacsclient -c')), EzKey('c', lazy.group['scratchpad'].dropdown_toggle('ipcam')), EzKey('s', lazy.group['scratchpad'].dropdown_toggle('spotify')), EzKey('b', lazy.group['scratchpad'].dropdown_toggle('bluetooth')), EzKey('a', lazy.group['scratchpad'].dropdown_toggle('audio')), EzKey('m', lazy.group['scratchpad'].dropdown_toggle('mail')), EzKey('p', lazy.group['scratchpad'].dropdown_toggle('proton')), ], 'M-s'), name = 's') ]) # bars keys.extend([ EzKey('M-b', lazy.hide_show_bar('all')), EzKey('M-v', lazy.hide_show_bar('bottom')), ]) # media keys keys.extend([ EzKey('', lazy.spawn('playerctl play')), EzKey('', lazy.spawn('playerctl pause')), EzKey('', lazy.spawn('playerctl stop')), EzKey('', lazy.spawn('playerctl next')), EzKey('', lazy.spawn('playerctl previous')), EzKey('', lazy.spawn('amixer -D pulse set Master 1+ toggle')), EzKey('', lazy.spawn('xbacklight -inc 5')), EzKey('', lazy.spawn('xbacklight -dec 5')), ]) # Printscreen keys.extend([ EzKey('', lazy.spawn('flameshot gui')), ]) # Qtile control keys.extend([ EzKey('M-S-c', lazy.window.kill()), EzKey('M-C-r', lazy.reload_config()), EzKey('M-C-q', lazy.shutdown()), ]) # Monitor navigation monitor_navigation_keys = ['w', 'e', 'r'] monitor_index_map = [ 1, 2, 0 ] for i, key in enumerate(monitor_navigation_keys): keys.extend([ EzKey(f'M-{key}', lazy.to_screen(monitor_index_map[i]), desc = f'Move focus to screen {i}'), EzKey(f'M-S-{key}', lazy.window.toscreen(monitor_index_map[i]), desc = f'Move focus to screen {i}'), ]) if qtile.core.name == 'x11': keys.append(EzKey('M-S-z', lazy.spawn('clipmenu'))) elif qtile.core.name == 'wayland': keys.append( EzKey( 'M-S-z', lazy.spawn('sh -c "cliphist list | dmenu -l 10 | cliphist decode | wl-copy"') ) ) group_defaults = { '9': { 'layout': 'max' } } logger.info(group_defaults.get('9')) groups = [Group(i) if not (i in group_defaults.keys()) else Group(i, **group_defaults.get(i)) for i in '123456789'] for i in groups: keys.extend( [ EzKey( f'M-{i.name}', go_to_group(i.name), desc='Switch to group {}'.format(i.name), ), Key( [mod, 'shift'], i.name, lazy.window.togroup(i.name, switch_group=False), desc='Switch to & move focused window to group {}'.format(i.name), ), ] ) scratch_pad_middle = { 'x': 0.2, 'y': 0.2, 'height': 0.6, 'width': 0.6, } groups.append( ScratchPad('scratchpad', [ DropDown( 'spotify', ['spotify'], on_focus_lost_hide = True, **scratch_pad_middle ), DropDown( 'bluetooth', ['blueman-manager'], on_focus_lost_hide = True, **scratch_pad_middle ), DropDown( 'audio', ['pavucontrol'], on_focus_lost_hide = True, **scratch_pad_middle ), DropDown( 'mail', ['thunderbird'], on_focus_lost_hide = True, x = 0.025, y = 0.025, width = 0.95, height = 0.95, opacity = 1, ), ]) ) # 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 = False cursor_warp = True floating_layout = layout.Floating( float_rules=[ *layout.Floating.default_float_rules, ] ) auto_fullscreen = True focus_on_window_activation = 'urgent' reconfigure_screens = False auto_minimize = True bring_front_click = False wl_input_rules = {} wmname = 'LG3D' # Swallow windows, # when a process with window spawns # another process with a window as a child, minimize the first # winddow. Turn off the minimization after the child process # is done. @hook.subscribe.client_new def _swallow(window): pid = window.window.get_net_wm_pid() ppid = psutil.Process(pid).ppid() cpids = {c.window.get_net_wm_pid(): wid for wid, c in window.qtile.windows_map.items()} for i in range(5): if not ppid: return if ppid in cpids: parent = window.qtile.windows_map.get(cpids[ppid]) parent.minimized = True window.parent = parent return ppid = psutil.Process(ppid).ppid() @hook.subscribe.client_killed def _unswallow(window): if hasattr(window, 'parent'): window.parent.minimized = False # Startup setup, # windows to correct workspaces, # start autostart.sh @hook.subscribe.startup_once def autostart(): subprocess.call([f'{configLocation}/autostart.sh']) firefoxInstance = 0 @hook.subscribe.client_new def startup_applications(client: Window): global firefoxInstance if not isinstance(client, Window): return if client.match(Match(wm_class = 'firefox')) and firefoxInstance <= 1: client.togroup(groups[firefoxInstance].name) firefoxInstance += 1 elif client.match(Match(wm_class = 'discord')) or client.match(Match(wm_class = 'cinny')): client.togroup(groups[8].name) # Turn off fullscreen on unfocus @hook.subscribe.client_focus def exit_fullscreen_on_focus_changed(client: Window): windows = client.group.windows window: Window for window in windows: if window != client and window.fullscreen: window.toggle_fullscreen() # Start scratchpads @hook.subscribe.startup_complete def scratchpad_startup(): scratchpad: ScratchPad = qtile.groups_map['scratchpad'] for dropdown_name, dropdown_config in scratchpad._dropdownconfig.items(): scratchpad._spawn(dropdown_config) def wrapper(name): def hide_dropdown(_): dropdown = scratchpad.dropdowns.get(name) if dropdown: dropdown.hide() hook.unsubscribe.client_managed(hide_dropdown) return hide_dropdown hook.subscribe.client_managed(wrapper(dropdown_name))