1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
|
#+title: My Qtile Config (Mocha)
#+author: Preston Pan
#+date: <2023-06-09 Fri>
#+description: a catppuccin qtile configuration written in emacs
#+html_head: <link rel="stylesheet" type="text/css" href="../style.css" />
* Goals
** We want the configuration to have efficient and ergonomic keybindings.
Commonly used keybindings should be two keys, less common ones should use more.
** We want it to work with other programs that it references.
For example, the qutebrowser and qtile configurations should not cause conflict
and they should have mutual consistency.
*** This will therefore make the configuration self sufficient.
** General enough that it will work for most purposes.
Specialized programs or tools should be in a separate section which will make
it easy to parse and remove.
* Configuration
** Imports and Definitions
All of this is also in the default configuration.
#+begin_src python :tangle config.py
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 import hook
from libqtile.backend.wayland import InputConfig
import os
import subprocess
mod = "mod4"
terminal = "kitty"
wl_import_rules = None
auto_minimize = True
wmname = "LG3D"
#+end_src
** Hex Colors
We then load the catppuccin colors for the bar and window borders.
#+begin_src python :tangle config.py
def get_colors(theme):
if theme == "city-lights":
return [
# Normal colors
'#45475a',
'#f38ba8',
'#a6e3a1',
'#f9e2af',
'#89b4fa',
'#f5c2e7',
'#94e2d5',
'#bac2de',
# Bright colors
'#585b70',
'#f38ba8',
'#a6e3a1',
'#f9e2af',
'#89b4fa',
'#f5c2e7',
'#94e2d5',
'#a6adc8',
# background
'#1e1e2e',
# foreground
'#cdd6f4',
]
elif theme == "gruvbox":
return [
# normal colors
'#282828',
'#cc241d',
'#98971a',
'#d79921',
'#458588',
'#b16286',
'#689d6a',
'#a89984',
#bright colors
'#928374',
'#fb4934',
'#b8bb26',
'#fabd2f',
'#83a598',
'#d3869b',
'#8ec07c',
'#ebdbb2',
# background
'#282828',
# foreground
'#ebdbb2',
]
colors = get_colors("gruvbox")
#+end_src
** Keybindings
The _keys_ variable is going to be our final list of keybindings. We start by initializing it
wth our window manipulation bindings with vim keys:
*** Focus controls
Vi inspired keybindings to manipulate focus:
#+begin_src python :tangle config.py
keys = [
Key([mod], "h", lazy.layout.left(), desc="Move focus to left"),
Key([mod], "l", lazy.layout.right(), desc="Move focus to right"),
Key([mod], "j", lazy.layout.down(), desc="Move focus down"),
Key([mod], "k", lazy.layout.up(), desc="Move focus up"),
Key([mod], "space", lazy.layout.next(), desc="Move window focus to other window"),
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"),
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], "n", lazy.layout.normalize(), desc="Reset all window sizes"),
Key(
[mod, "shift"],
"Return",
lazy.layout.toggle_split(),
desc="Toggle between split and unsplit sides of stack",
),
Key([mod], "Tab", lazy.next_layout(), desc="Toggle between layouts"),
]
#+end_src
*** Quit/Restart
#+begin_src python :tangle config.py
keys.extend([
Key([mod], "q", lazy.window.kill(), desc="Kill focused window"),
Key([mod, "control"], "r", lazy.reload_config(), desc="Reload the config"),
Key([mod, "control"], "q", lazy.shutdown(), desc="Shutdown Qtile"),
])
#+end_src
*** Programs
These are our keybindings for user programs.
#+begin_src python :tangle config.py
keys.extend([
Key([mod], "r", lazy.spawncmd(), desc="Spawn a command using a prompt widget"),
Key([mod], "Return", lazy.spawn(terminal), desc="Launch terminal"),
Key([mod], "e", lazy.spawn("emacs"), desc="Run emacs"),
Key([mod], "w", lazy.spawn("qutebrowser"), desc="Run Qutebrowser"),
Key([mod], "f", lazy.spawn("firefox"), desc="Run Firefox"),
Key([mod], "b", lazy.spawn("blender"), desc="Run Blender"),
Key([mod], "p", lazy.spawn("krita"), desc="Run Krita"),
Key([mod], "v", lazy.spawn("inkscape"), desc="Run Inkscape"),
Key([mod], "g", lazy.spawn("gimp"), desc="Run GIMP"),
Key([mod], "t", lazy.spawn("torbrowser-launcher"), desc="Run Tor Browser"),
Key([mod], "i", lazy.spawn("emacsclient --eval \"(emacs-everywhere)\""), desc="Emacs Everywhere!"),
Key([mod], "d", lazy.spawn("rofi -show run"), desc="rofi command launcher"),
])
#+end_src
*** XF86
Now we need keybindings for the function keys:
#+begin_src python :tangle config.py
keys.extend([
Key([], 'XF86AudioLowerVolume', lazy.spawn("pactl set-sink-volume @DEFAULT_SINK@ -5%")),
Key([], 'XF86AudioRaiseVolume', lazy.spawn("pactl set-sink-volume @DEFAULT_SINK@ +5%")),
Key([], 'XF86AudioMute', lazy.spawn("pactl set-sink-mute @DEFAULT_SINK@ toggle")),
Key([], 'XF86MonBrightnessUp', lazy.spawn("light -A 10")),
Key([], 'XF86MonBrightnessDown', lazy.spawn("light -U 10")),
Key([], 'XF86AudioNext', lazy.spawn("mpc next")),
Key([], 'XF86AudioPrev', lazy.spawn("mpc prev")),
Key([], "XF86AudioPlay", lazy.spawn("mpc toggle"), desc="Play/Pause player"),
Key([], "Print", lazy.spawn("scrot '%Y-%m-%d-%s_screenshot_$wx$h.jpg' -e 'mv $f ~/img/scrot")),
])
#+end_src
** Groups
Now we name our groups:
#+begin_src python :tangle config.py
groups = [Group(i) for i in "123456789"]
for i in groups:
keys.extend(
[
Key(
[mod],
i.name,
lazy.group[i.name].toscreen(),
desc="Switch to group {}".format(i.name),
),
Key(
[mod, "shift"],
i.name,
lazy.window.togroup(i.name, switch_group=True),
desc="Switch to & move focused window to group {}".format(i.name),
),
]
)
#+end_src
** Layouts
This is our list of enabled layouts. You can enable more of them if you want.
#+begin_src python :tangle config.py
layouts = [
layout.Columns(border_focus=colors[2], border_normal=colors[0], border_width=4, margin=7),
layout.Max(),
# Try more layouts by unleashing below layouts.
# layout.Stack(num_stacks=2),
# layout.Bsp(),
# layout.Matrix(),
layout.MonadTall(border_focus=colors[2], border_normal=colors[0], border_width=4, margin=7),
# layout.MonadWide(),
# layout.RatioTile(),
# layout.Tile(),
# layout.TreeTab(),
# layout.VerticalTile(),
# layout.Zoomy(),
]
#+end_src
** Bar
Now we define our bar. I only have the need to see the time, current workspace, battery percentage,
and MPD. Also, you may need to manually change your font size depending on your screen.
#+begin_src python :tangle config.py
widget_defaults = dict(
font="FiraCode Nerd Font",
fontsize=16,
padding=4,
foreground=colors[17],
background=colors[16],
)
extension_defaults = widget_defaults.copy()
# screens = [
# Screen(
# top=bar.Bar(
# [
# # widget.CurrentLayout(),
# widget.GroupBox(active=colors[6], inactive=colors[15], this_current_screen_border=colors[4], highlight_colorsr=colors[3]),
# widget.Prompt(),
# widget.WindowName(),
# widget.Chord(
# chords_colors={
# "launch": ("#ff0000", "#ffffff"),
# },
# name_transform=lambda name: name.upper(),
# ),
# # widget.StatusNotifier(),
# widget.Systray(),
# widget.Battery(charge_char="🔋", discharge_char="🔋", full_char="🔋", format="{char} {percent:2.0%}"),
# # widget.TextBox("|", foreground=colors[1]),
# widget.Sep(padding=16, size_percent=80, foreground=colors[1]),
# widget.Clock(format="🕒 %a %I:%M %p"),
# widget.Sep(padding=16, size_percent=80, foreground=colors[1]),
# widget.Mpd2(),
# widget.TextBox(" "),
# ],
# 24,
# # border_width=[2, 0, 2, 0], # Draw top and bottom borders
# # border_colorsr=["ff00ff", "000000", "ff00ff", "000000"] # Borders are magenta
# ),
# bottom=bar.Gap(4),
# left=bar.Gap(3),
# right=bar.Gap(3),
# ),
# ]
def pline(rl, fg, bg):
if rl == 0:
uc = ""
else:
uc = ""
return widget.TextBox(text = uc,
padding = 0,
fontsize = 22,
foreground=fg,
background=bg)
screens = [
Screen(
wallpaper="~/.config/qtile/wallpaper",
wallpaper_mode="fill",
top=bar.Bar(
[
widget.CurrentLayoutIcon(
scale=0.75,
background=colors[3]
),
pline(0, colors[3], colors[6]),
widget.GroupBox(
highlight_method="block",
background=colors[6],
this_current_screen_border="#7daea3"
),
pline(0, colors[6], colors[7]),
widget.TaskList(
highlight_method="block",
max_title_width=300,
border="#d3869b",
padding=2,
background=colors[7]
),
pline(0, colors[7], colors[0]),
widget.Spacer(),
pline(1, colors[2], colors[0]),
widget.Net( # requires python-psutil
interface="wlp0s20f3",
format="📡 {total}",
update_interval=30,
background=colors[2]
),
pline(1, colors[5], colors[2]),
widget.Backlight(
format="💡 {percent:2.0%}",
backlight_name="intel_backlight",
background=colors[5]
),
pline(1, colors[3], colors[5]),
widget.Volume(
emoji=True,
background=colors[3]
),
widget.Volume(
background=colors[3]
),
pline(1, colors[4], colors[3]),
widget.BatteryIcon(
background=colors[4]
),
widget.Battery(
charge_char="now ",
discharge_char="left",
format="{percent:2.0%} {char}",
background=colors[4]
),
pline(1, colors[1], colors[4]),
widget.Clock(
format="%Y-%m-%d %a %I:%M %p",
background=colors[1]
),
],
26,
),
),
]
#+end_src
** Mouse
We configure the mouse to interact with floating windows.
#+begin_src python :tangle config.py
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()),
]
#+end_src
Also, we need to toggle some options:
#+begin_src python :tangle config.py
dgroups_app_rules = [] # type: list
follow_mouse_focus = True
bring_front_click = False
cursor_warp = False
#+end_src
And then we add the applications that need to start in floating:
#+begin_src python :tangle config.py
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="confirmreset"), # gitk
Match(wm_class="makebranch"), # gitk
Match(wm_class="maketag"), # gitk
Match(wm_class="ssh-askpass"), # ssh-askpass
|