Compare commits

..

3 Commits

Author SHA1 Message Date
e22fbbd504 Focus new windows 2025-10-29 19:34:49 +00:00
2c253da55d Formatting fixes 2025-10-29 19:32:44 +00:00
de595a0802 Final touches 2025-10-29 18:17:59 +00:00
9 changed files with 165 additions and 83 deletions

View File

@@ -4,14 +4,14 @@ require "socket"
SOCK_PATH = "/tmp/kutu.sock" SOCK_PATH = "/tmp/kutu.sock"
client = Socket.new(:UNIX, :DGRAM)
cli_addr_path = "/tmp/kutu_client#{Process.pid}.sock" cli_addr_path = "/tmp/kutu_client#{Process.pid}.sock"
File.delete(cli_addr_path) if File.exist?(cli_addr_path) File.delete cli_addr_path if File.exist? cli_addr_path
client.bind(Socket.pack_sockaddr_un(cli_addr_path))
client.send(ARGV.join(" "), 0, Socket.pack_sockaddr_un(SOCK_PATH)) client = Socket.new :UNIX, :DGRAM
client.bind Socket.pack_sockaddr_un(cli_addr_path)
data, = client.recvfrom(1024) client.send ARGV.join(" "), 0, Socket.pack_sockaddr_un(SOCK_PATH)
data, = client.recvfrom 1024
puts data puts data

25
kutu.rb
View File

@@ -16,7 +16,7 @@ require 'socket'
if X.deploy >= 0 if X.deploy >= 0
puts "Started kutu WM for X11 successfully! :)" puts "Started kutu WM for X11 successfully! :)"
else else
raise "Failed to deploy X" raise "Failed to deploy X, try running using xinit or startx."
end end
@@ -50,8 +50,6 @@ $mousebind_actions = {}
$mouse_data = {} $mouse_data = {}
$root = X.get_root $root = X.get_root
$rect = {} $rect = {}
@@ -74,28 +72,25 @@ load File.join(__dir__, "./src/ruby/bindings.rb")
# Setup unix socket # Setup unix socket
SOCK_PATH = "/tmp/kutu.sock" SOCK_PATH = "/tmp/kutu.sock"
File.delete(SOCK_PATH) if File.exist?(SOCK_PATH) File.delete SOCK_PATH if File.exist? SOCK_PATH
$socket = Socket.new(:UNIX, :DGRAM) $socket = Socket.new :UNIX, :DGRAM
$socket.bind(Socket.pack_sockaddr_un(SOCK_PATH)) $socket.bind Socket.pack_sockaddr_un(SOCK_PATH)
# Main loop # Main loop
loop do loop do
sleep 0.001
if IO.select([$socket], nil, nil, 0) if IO.select([$socket], nil, nil, 0)
command, sender = $socket.recvfrom(1024) command, sender = $socket.recvfrom 1024
reply = handle_command(command) reply = handle_command command
$socket.send(JSON.generate(reply), 0, sender) $socket.send JSON.generate(reply), 0, sender
end end
event_pointer = X.next_event event_pointer = X.next_event
if !event_pointer.null? handle_event X.translate_event(event_pointer) if !event_pointer.null?
event = X.translate_event(event_pointer)
handle_event(event)
end
X.flush X.flush
sleep 0.001
end end

14
compile.sh → setup.sh Executable file → Normal file
View File

@@ -24,3 +24,17 @@ if [ ! -f "$DIR/build/X-kutu.so" ]; then
else else
echo "Success: $DIR/build/X-kutu.so is compiled." echo "Success: $DIR/build/X-kutu.so is compiled."
fi fi
for f in "$DIR/kutu-run.rb" "$DIR/kutu.rb" "$DIR/src/shell/"*; do
chmod +x "$f" || {
echo "Error: Failed to chmod $f" >&2
exit 1
}
done
if ! command -v kutu.rb >/dev/null 2>&1 || ! command -v kutu-run.rb >/dev/null 2>&1; then
echo "Tip: Add $DIR to your PATH to run 'kutu.rb' and 'kutu-run.rb' from anywhere:"
echo "export PATH=\"\$PATH:$DIR\""
fi
echo -e "\e[32mAll done! Build successful.\e[0m"

View File

@@ -11,23 +11,56 @@ keybind 25 do |_event|
end end
keybind 26 do |_event| keybind 26 do |_event|
run "~/dotfiles/scripts/power.sh" run File.join(__dir__, "../shell/power.sh")
end end
keybind 123, 0 do |_event| keybind 123, 0 do |_event|
run "pactl set-sink-volume @DEFAULT_SINK@ +5%;vol=$(pactl get-sink-volume @DEFAULT_SINK@ | grep -o '[0-9]\+%' | head -n1);dunstify 'Volume Adjusted' '' -h int:value:$vol -r 997" run %q(
pactl set-sink-volume @DEFAULT_SINK@ +5%;
vol=$(pactl get-sink-volume @DEFAULT_SINK@ | grep -o '[0-9]\+%' | head -n1);
dunstify "Volume Adjusted" "" -h int:value:$vol -r 997
)
end end
keybind 122, 0 do |_event| keybind 122, 0 do |_event|
run "pactl set-sink-volume @DEFAULT_SINK@ -5%;vol=$(pactl get-sink-volume @DEFAULT_SINK@ | grep -o '[0-9]\+%' | head -n1);dunstify 'Volume Adjusted' '' -h int:value:$vol -r 997" run %q(
pactl set-sink-volume @DEFAULT_SINK@ -5%;
vol=$(pactl get-sink-volume @DEFAULT_SINK@ | grep -o '[0-9]\+%' | head -n1);
dunstify "Volume Adjusted" "" -h int:value:$vol -r 997
)
end end
keybind 121, 0 do |_event| keybind 121, 0 do |_event|
run "pactl set-sink-mute @DEFAULT_SINK@ toggle;vol=$(pactl get-sink-volume @DEFAULT_SINK@ | grep -o '[0-9]\+%' | head -n1);dunstify 'Volume Adjusted' '' -h int:value:$vol -r 997" run %q(
pactl set-sink-mute @DEFAULT_SINK@ toggle;
vol=$(pactl get-sink-volume @DEFAULT_SINK@ | grep -o '[0-9]\\+%' | head -n1);
dunstify "Volume Adjusted" "" -h int:value:$vol -r 997
)
end end
keybind 38 do |_event| keybind 38 do |_event|
run "maim -c 0.3,0.5,1.0,0.8 -s | tee /tmp/screenshot_temp.png | xclip -selection clipboard -t image/png && if [ -s '/tmp/screenshot_temp.png' ]; then mv /tmp/screenshot_temp.png ~/screenshots/$(date +%Y-%m-%d_%H:%M:%S).png; fi" run %Q(
maim -c 0.3,0.5,1.0,0.8 -s | tee /tmp/screenshot_temp.png | xclip -selection clipboard -t image/png;
if [ -s '/tmp/screenshot_temp.png' ]; then
mv /tmp/screenshot_temp.png ~/screenshots/$(date +%Y-%m-%d_%H:%M:%S).png;
fi
)
end
keybind 232, 0 do |_event|
run %q(
brightnessctl set 5%-;
pct=$(brightnessctl -m | cut -d, -f4 | tr -d ' %');
dunstify "Brightness" "" -h int:value:$pct -r 998
)
end
keybind 233, 0 do |_event|
run %q(
brightnessctl set 5%+;
pct=$(brightnessctl -m | cut -d, -f4 | tr -d ' %');
dunstify "Brightness" "" -h int:value:$pct -r 998
)
end end
keybind 54 do |_event| keybind 54 do |_event|
@@ -39,7 +72,7 @@ keybind 53 do |_event|
end end
keybind 40 do |_event| keybind 40 do |_event|
run "~/dotfiles/scripts/run.sh" run File.join(__dir__, "../shell/run.sh")
end end
keybind 33 do |_event| keybind 33 do |_event|
@@ -47,7 +80,7 @@ keybind 33 do |_event|
end end
keybind 45 do |_event| keybind 45 do |_event|
run "~/dotfiles/scripts/caffiene.sh" run File.join(__dir__, "../shell/caffiene.sh")
end end
keybind 56 do |_event| keybind 56 do |_event|
@@ -75,8 +108,11 @@ keybind 57 do |_event|
end end
keybind 110, 0 do |_event| keybind 110, 0 do |_event|
run "dbus-send --print-reply --dest=org.mpris.MediaPlayer2.spotify /org/mpris/MediaPlayer2 org.mpris.MediaPlayer2.Player.Pause;" \ run %q(
"dbus-send --print-reply --dest=$(busctl --user list | grep -oP 'org.mpris.MediaPlayer2.firefox.instance_1_\d+') /org/mpris/MediaPlayer2 org.mpris.MediaPlayer2.Player.PlayPause" dbus-send --print-reply --dest=org.mpris.MediaPlayer2.spotify /org/mpris/MediaPlayer2 org.mpris.MediaPlayer2.Player.Pause;
dbus-send --print-reply --dest=$(busctl --user list | grep -oP 'org.mpris.MediaPlayer2.firefox.instance_1_\\d+') \
/org/mpris/MediaPlayer2 org.mpris.MediaPlayer2.Player.PlayPause
)
end end
keybind 110 do |_event| keybind 110 do |_event|
@@ -95,6 +131,19 @@ keybind 110 do |_event|
end end
end end
keybind 115 do |event|
monitor = current_monitor
persistence_path = File.join(__dir__, ".num.json")
persistence = File.exist?(persistence_path) ?
JSON.parse(File.read(persistence_path), symbolize_names: true) :
{}
ws = monitor[:workspaces][
monitor[:selected_workspace] == 0 ?
monitor[:workspaces][persistence[$monitors.key(monitor)]&.[](:saved) || 1] : 0
]
ws.drop ws.tiled_windows.length, $windows[event[:window]] if $windows[event[:window]]
end
keybind 118, 0 do |_event| keybind 118, 0 do |_event|
run "dbus-send --print-reply --dest=org.mpris.MediaPlayer2.spotify /org/mpris/MediaPlayer2 org.mpris.MediaPlayer2.Player.PlayPause" run "dbus-send --print-reply --dest=org.mpris.MediaPlayer2.spotify /org/mpris/MediaPlayer2 org.mpris.MediaPlayer2.Player.PlayPause"
end end
@@ -106,20 +155,11 @@ keybind 117, 0 do |_event|
select_workspace next_ws, monitor select_workspace next_ws, monitor
end end
keybind 117 do |_event| keybind 117 do |event|
pointer = X.get_pointer monitor = current_monitor
monitor = current_monitor pointer
next if monitor[:selected_workspace].zero? next if monitor[:selected_workspace].zero?
ws = monitor[:workspaces][(monitor[:selected_workspace] % (monitor[:workspaces].length - 1)) + 1]
window = monitor[:workspaces][monitor[:selected_workspace]].windows.find do |w| ws.drop ws.tiled_windows.length, $windows[event[:window]] if $windows[event[:window]]
pointer[:x] >= w.x &&
pointer[:x] < w.x + w.width &&
pointer[:y] >= w.y &&
pointer[:y] < w.y + w.height
end
next_ws = (monitor[:selected_workspace] % (monitor[:workspaces].length - 1)) + 1
monitor[:workspaces][next_ws].drop monitor[:workspaces][next_ws].tiled_windows.length, window if window
end end
keybind 112, 0 do |_event| keybind 112, 0 do |_event|
@@ -129,20 +169,11 @@ keybind 112, 0 do |_event|
select_workspace next_ws, monitor select_workspace next_ws, monitor
end end
keybind 112 do |_event| keybind 112 do |event|
pointer = X.get_pointer monitor = current_monitor
monitor = current_monitor pointer
next if monitor[:selected_workspace].zero? next if monitor[:selected_workspace].zero?
ws = monitor[:workspaces][((monitor[:selected_workspace] - 2) % (monitor[:workspaces].length - 1)) + 1]
window = monitor[:workspaces][monitor[:selected_workspace]].windows.find do |w| ws.drop ws.tiled_windows.length, $windows[event[:window]] if $windows[event[:window]]
pointer[:x] >= w.x &&
pointer[:x] < w.x + w.width &&
pointer[:y] >= w.y &&
pointer[:y] < w.y + w.height
end
next_ws = ((monitor[:selected_workspace] - 2) % (monitor[:workspaces].length - 1)) + 1
monitor[:workspaces][next_ws].drop monitor[:workspaces][next_ws].tiled_windows.length, window if window
end end
keybind 27 do |_event| keybind 27 do |_event|
@@ -161,7 +192,6 @@ keybind 39 do |event|
window.toggle_floating if window window.toggle_floating if window
end end
mousebind 1 mousebind 1
mousebind 3 mousebind 3
@@ -174,19 +204,10 @@ mousebind 9, 0 do |_event|
end end
mousebind 9 do |_event| mousebind 9 do |_event|
pointer = X.get_pointer monitor = current_monitor
monitor = current_monitor pointer
next if monitor[:selected_workspace].zero? next if monitor[:selected_workspace].zero?
ws = monitor[:workspaces][(monitor[:selected_workspace] % (monitor[:workspaces].length - 1)) + 1]
window = monitor[:workspaces][monitor[:selected_workspace]].windows.find do |w| ws.drop ws.tiled_windows.length, $windows[event[:window]] if $windows[event[:window]]
pointer[:x] >= w.x &&
pointer[:x] < w.x + w.width &&
pointer[:y] >= w.y &&
pointer[:y] < w.y + w.height
end
next_ws = (monitor[:selected_workspace] % (monitor[:workspaces].length - 1)) + 1
monitor[:workspaces][next_ws].drop monitor[:workspaces][next_ws].tiled_windows.length, window if window
end end
mousebind 8, 0 do |_event| mousebind 8, 0 do |_event|
@@ -197,17 +218,8 @@ mousebind 8, 0 do |_event|
end end
mousebind 8 do |_event| mousebind 8 do |_event|
pointer = X.get_pointer monitor = current_monitor
monitor = current_monitor pointer
next if monitor[:selected_workspace].zero? next if monitor[:selected_workspace].zero?
ws = monitor[:workspaces][((monitor[:selected_workspace] - 2) % (monitor[:workspaces].length - 1)) + 1]
window = monitor[:workspaces][monitor[:selected_workspace]].windows.find do |w| ws.drop ws.tiled_windows.length, $windows[event[:window]] if $windows[event[:window]]
pointer[:x] >= w.x &&
pointer[:x] < w.x + w.width &&
pointer[:y] >= w.y &&
pointer[:y] < w.y + w.height
end
next_ws = ((monitor[:selected_workspace] - 2) % (monitor[:workspaces].length - 1)) + 1
monitor[:workspaces][next_ws].drop monitor[:workspaces][next_ws].tiled_windows.length, window if window
end end

View File

@@ -45,6 +45,7 @@ def handle_event(event)
X.send_to_top window if above.include?("ABOVE") X.send_to_top window if above.include?("ABOVE")
end end
end end
X.focus event[:window]
when :mouse_press when :mouse_press

View File

@@ -9,8 +9,8 @@ def mousebind(btn, mod = 1, &block)
end end
def run(command) def run(cmd)
pid = spawn command pid = spawn "bash", "-c", cmd
Process.detach pid Process.detach pid
end end

9
src/shell/caffiene.sh Executable file
View File

@@ -0,0 +1,9 @@
#!/usr/bin/env bash
if [[ "$(xset q | awk '/timeout:/ {print $2}')" == "0" ]]; then
xset s on
xset +dpms
else
xset s off
xset -dpms
fi

34
src/shell/power.sh Executable file
View File

@@ -0,0 +1,34 @@
#!/usr/bin/env bash
set -euo pipefail
opts="
󰒲 Suspend
 Stop KutuWM
 Shutdown
󰜉 Reboot"
sel=$(printf "%s\n" "$opts" | dmenu -i -p "Select Power Option:" \
-nf '#e0af68' -nb '#1f2335' -sb '#f7768e' -sf '#1a1b26' -fn 'HurmitNerdFont-16')
# user pressed Esc
[ -z "$sel" ] && exit 0
case "$sel" in
*Shutdown*)
confirm=$(printf "No\nYes" | dmenu -i -p "Are you sure you want to shutdown? :" \
-nf '#e0af68' -nb '#1f2335' -sb '#f7768e' -sf '#1a1b26' -fn 'HurmitNerdFont-16')
[ "$confirm" = "Yes" ] && exec shutdown -h now
;;
*Reboot*)
confirm=$(printf "No\nYes" | dmenu -i -p "Are you sure you want to reboot? :" \
-nf '#e0af68' -nb '#1f2335' -sb '#f7768e' -sf '#1a1b26' -fn 'HurmitNerdFont-16')
[ "$confirm" = "Yes" ] && exec reboot
;;
*Stop\ KutuWM*)
exec kutu-run.rb stop
;;
*Suspend*)
~/dotfiles/scripts/lock.sh &
exec systemctl suspend
;;
esac

17
src/shell/run.sh Executable file
View File

@@ -0,0 +1,17 @@
#!/usr/bin/env bash
choice=$(
(
printf "%s\n" godot mvox mox aseprite terraria
compgen -c
) |
grep -v -E '^(if|fi|case|esac|for|done|while|until|select|function|return|continue|break|time|exec|source|alias|builtin|read|export|unset|local|set|declare|typeset|:|\.|\[|coproc|l|ll|ls|then|else|elif|do|in|\{|\}|!|\[\[|\]\]|_.*|compgen)$' |
sort |
dmenu -i -p "Enter command  " \
-nf '#4abaaf' -nb '#1f2335' -sb '#7aa2f7' -sf '#102030' -fn 'HurmitNerdFont-16'
)
[ -z "$choice" ] && exit 0
fish -c "$choice" >/dev/null 2>&1 &
disown