mirror of
https://github.com/pinnacle-comp/pinnacle.git
synced 2025-01-15 15:42:06 +01:00
400 lines
12 KiB
Rust
400 lines
12 KiB
Rust
use pinnacle_api::input::libinput::LibinputSetting;
|
|
use pinnacle_api::input::KeybindInfo;
|
|
use pinnacle_api::input::Keysym;
|
|
use pinnacle_api::layout::{
|
|
CornerLayout, CornerLocation, CyclingLayoutManager, DwindleLayout, FairLayout, MasterSide,
|
|
MasterStackLayout, SpiralLayout,
|
|
};
|
|
use pinnacle_api::output::OutputSetup;
|
|
use pinnacle_api::pinnacle::Backend;
|
|
use pinnacle_api::signal::WindowSignal;
|
|
use pinnacle_api::util::{Axis, Batch};
|
|
use pinnacle_api::window::rules::{DecorationMode, WindowRule, WindowRuleCondition};
|
|
use pinnacle_api::{
|
|
input::{Mod, MouseButton, MouseEdge},
|
|
ApiModules,
|
|
};
|
|
|
|
// Pinnacle needs to perform some setup before and after your config,
|
|
// which is what this macro does.
|
|
//
|
|
// By default, logging is disabled here because this config is embedded inside Pinnacle
|
|
// and that would cause a panic. Remove `internal_tracing = false` if you want to
|
|
// enable logging for debugging.
|
|
#[pinnacle_api::config(internal_tracing = false)]
|
|
async fn main() {
|
|
// Deconstruct to get all the APIs.
|
|
#[allow(unused_variables)]
|
|
let ApiModules {
|
|
pinnacle,
|
|
process,
|
|
window,
|
|
input,
|
|
output,
|
|
tag,
|
|
layout,
|
|
render,
|
|
#[cfg(feature = "snowcap")]
|
|
snowcap,
|
|
..
|
|
} = ApiModules::new();
|
|
|
|
// Change the mod key to `Alt` when running as a nested window.
|
|
let mod_key = match pinnacle.backend() {
|
|
Backend::Tty => Mod::Super,
|
|
Backend::Window => Mod::Alt,
|
|
};
|
|
|
|
let terminal = "alacritty";
|
|
|
|
//------------------------
|
|
// Mousebinds |
|
|
//------------------------
|
|
|
|
// `mod_key + left click` starts moving a window
|
|
input.mousebind([mod_key], MouseButton::Left, MouseEdge::Press, || {
|
|
window.begin_move(MouseButton::Left);
|
|
});
|
|
|
|
// `mod_key + right click` starts resizing a window
|
|
input.mousebind([mod_key], MouseButton::Right, MouseEdge::Press, || {
|
|
window.begin_resize(MouseButton::Right);
|
|
});
|
|
|
|
//------------------------
|
|
// Keybinds |
|
|
//------------------------
|
|
|
|
// `mod_key + s` shows the keybind overlay
|
|
#[cfg(feature = "snowcap")]
|
|
input.keybind(
|
|
[mod_key],
|
|
's',
|
|
|| {
|
|
snowcap.integration.keybind_overlay().show();
|
|
},
|
|
KeybindInfo {
|
|
group: Some("Compositor".into()),
|
|
description: Some("Show the keybind overlay".into()),
|
|
},
|
|
);
|
|
|
|
// `mod_key + shift + q` quits Pinnacle
|
|
input.keybind(
|
|
[mod_key, Mod::Shift],
|
|
'q',
|
|
|| {
|
|
#[cfg(feature = "snowcap")]
|
|
snowcap.integration.quit_prompt().show();
|
|
#[cfg(not(feature = "snowcap"))]
|
|
pinnacle.quit();
|
|
},
|
|
KeybindInfo {
|
|
group: Some("Compositor".into()),
|
|
description: Some("Quit Pinnacle".into()),
|
|
},
|
|
);
|
|
|
|
// `mod_key + ctrl + r` reloads the config
|
|
input.keybind(
|
|
[mod_key, Mod::Ctrl],
|
|
'r',
|
|
|| {
|
|
pinnacle.reload_config();
|
|
},
|
|
KeybindInfo {
|
|
group: Some("Compositor".into()),
|
|
description: Some("Reload the config".into()),
|
|
},
|
|
);
|
|
|
|
// `mod_key + shift + c` closes the focused window
|
|
input.keybind(
|
|
[mod_key, Mod::Shift],
|
|
'c',
|
|
|| {
|
|
if let Some(window) = window.get_focused() {
|
|
window.close();
|
|
}
|
|
},
|
|
KeybindInfo {
|
|
group: Some("Window".into()),
|
|
description: Some("Close the focused window".into()),
|
|
},
|
|
);
|
|
|
|
// `mod_key + Return` spawns a terminal
|
|
input.keybind(
|
|
[mod_key],
|
|
Keysym::Return,
|
|
move || {
|
|
process.spawn([terminal]);
|
|
},
|
|
KeybindInfo {
|
|
group: Some("Process".into()),
|
|
description: Some(format!("Spawn `{terminal}`")),
|
|
},
|
|
);
|
|
|
|
// `mod_key + ctrl + space` toggles floating
|
|
input.keybind(
|
|
[mod_key, Mod::Ctrl],
|
|
Keysym::space,
|
|
|| {
|
|
if let Some(window) = window.get_focused() {
|
|
window.toggle_floating();
|
|
window.raise();
|
|
}
|
|
},
|
|
KeybindInfo {
|
|
group: Some("Window".into()),
|
|
description: Some("Toggle floating on the focused window".into()),
|
|
},
|
|
);
|
|
|
|
// `mod_key + f` toggles fullscreen
|
|
input.keybind(
|
|
[mod_key],
|
|
'f',
|
|
|| {
|
|
if let Some(window) = window.get_focused() {
|
|
window.toggle_fullscreen();
|
|
window.raise();
|
|
}
|
|
},
|
|
KeybindInfo {
|
|
group: Some("Window".into()),
|
|
description: Some("Toggle fullscreen on the focused window".into()),
|
|
},
|
|
);
|
|
|
|
// `mod_key + m` toggles maximized
|
|
input.keybind(
|
|
[mod_key],
|
|
'm',
|
|
|| {
|
|
if let Some(window) = window.get_focused() {
|
|
window.toggle_maximized();
|
|
window.raise();
|
|
}
|
|
},
|
|
KeybindInfo {
|
|
group: Some("Window".into()),
|
|
description: Some("Toggle maximized on the focused window".into()),
|
|
},
|
|
);
|
|
|
|
//------------------------
|
|
// Window rules |
|
|
//------------------------
|
|
// You can define window rules to get windows to open with desired properties.
|
|
// See `pinnacle_api::window::rules` in the docs for more information.
|
|
|
|
//------------------------
|
|
// Layouts |
|
|
//------------------------
|
|
|
|
// Pinnacle does not manage layouts compositor-side.
|
|
// Instead, it delegates computation of layouts to your config,
|
|
// which provides an interface to calculate the size and location of
|
|
// windows that the compositor will use to position windows.
|
|
//
|
|
// If you're familiar with River's layout generators, you'll understand the system here
|
|
// a bit better.
|
|
//
|
|
// The Rust API provides two layout system abstractions:
|
|
// 1. Layout managers, and
|
|
// 2. Layout generators.
|
|
//
|
|
// ### Layout Managers ###
|
|
// A layout manager is a struct that implements the `LayoutManager` trait.
|
|
// A manager is meant to keep track of and choose various layout generators
|
|
// across your usage of the compositor.
|
|
//
|
|
// ### Layout generators ###
|
|
// A layout generator is a struct that implements the `LayoutGenerator` trait.
|
|
// It takes in layout arguments and computes a vector of geometries that will
|
|
// determine the size and position of windows being laid out.
|
|
//
|
|
// There is one built-in layout manager and five built-in layout generators,
|
|
// as shown below.
|
|
//
|
|
// Additionally, this system is designed to be user-extensible;
|
|
// you are free to create your own layout managers and generators for
|
|
// maximum customizability! Docs for doing so are in the works, so sit tight.
|
|
|
|
// Create a `CyclingLayoutManager` that can cycle between layouts on different tags.
|
|
//
|
|
// It takes in some layout generators that need to be boxed and dyn-coerced.
|
|
let layout_requester = layout.set_manager(CyclingLayoutManager::new([
|
|
Box::<MasterStackLayout>::default() as _,
|
|
Box::new(MasterStackLayout {
|
|
master_side: MasterSide::Right,
|
|
..Default::default()
|
|
}) as _,
|
|
Box::new(MasterStackLayout {
|
|
master_side: MasterSide::Top,
|
|
..Default::default()
|
|
}) as _,
|
|
Box::new(MasterStackLayout {
|
|
master_side: MasterSide::Bottom,
|
|
..Default::default()
|
|
}) as _,
|
|
Box::<DwindleLayout>::default() as _,
|
|
Box::<SpiralLayout>::default() as _,
|
|
Box::<CornerLayout>::default() as _,
|
|
Box::new(CornerLayout {
|
|
corner_loc: CornerLocation::TopRight,
|
|
..Default::default()
|
|
}) as _,
|
|
Box::new(CornerLayout {
|
|
corner_loc: CornerLocation::BottomLeft,
|
|
..Default::default()
|
|
}) as _,
|
|
Box::new(CornerLayout {
|
|
corner_loc: CornerLocation::BottomRight,
|
|
..Default::default()
|
|
}) as _,
|
|
Box::<FairLayout>::default() as _,
|
|
Box::new(FairLayout {
|
|
axis: Axis::Horizontal,
|
|
..Default::default()
|
|
}) as _,
|
|
]));
|
|
|
|
let mut layout_requester_clone = layout_requester.clone();
|
|
|
|
// `mod_key + space` cycles to the next layout
|
|
input.keybind(
|
|
[mod_key],
|
|
Keysym::space,
|
|
move || {
|
|
let Some(focused_op) = output.get_focused() else { return };
|
|
let Some(first_active_tag) = focused_op.tags().batch_find(
|
|
|tg| Box::pin(tg.active_async()),
|
|
|active| active == &Some(true),
|
|
) else {
|
|
return;
|
|
};
|
|
|
|
layout_requester.cycle_layout_forward(&first_active_tag);
|
|
layout_requester.request_layout_on_output(&focused_op);
|
|
},
|
|
KeybindInfo {
|
|
group: Some("Layout".into()),
|
|
description: Some("Cycle the layout forward on the first active tag".into()),
|
|
},
|
|
);
|
|
|
|
// `mod_key + shift + space` cycles to the previous layout
|
|
input.keybind(
|
|
[mod_key, Mod::Shift],
|
|
Keysym::space,
|
|
move || {
|
|
let Some(focused_op) = output.get_focused() else { return };
|
|
let Some(first_active_tag) = focused_op.tags().batch_find(
|
|
|tg| Box::pin(tg.active_async()),
|
|
|active| active == &Some(true),
|
|
) else {
|
|
return;
|
|
};
|
|
|
|
layout_requester_clone.cycle_layout_backward(&first_active_tag);
|
|
layout_requester_clone.request_layout_on_output(&focused_op);
|
|
},
|
|
KeybindInfo {
|
|
group: Some("Layout".into()),
|
|
description: Some("Cycle the layout backward on the first active tag".into()),
|
|
},
|
|
);
|
|
|
|
//------------------------
|
|
// Tags |
|
|
//------------------------
|
|
|
|
let tag_names = ["1", "2", "3", "4", "5"];
|
|
|
|
// Setup all monitors with tags "1" through "5"
|
|
output.setup([OutputSetup::new_with_matcher(|_| true).with_tags(tag_names)]);
|
|
|
|
for tag_name in tag_names {
|
|
// `mod_key + 1-5` switches to tag "1" to "5"
|
|
input.keybind(
|
|
[mod_key],
|
|
tag_name,
|
|
move || {
|
|
if let Some(tg) = tag.get(tag_name) {
|
|
tg.switch_to();
|
|
}
|
|
},
|
|
KeybindInfo {
|
|
group: Some("Tag".into()),
|
|
description: Some(format!("Switch to tag {tag_name}")),
|
|
},
|
|
);
|
|
|
|
// `mod_key + ctrl + 1-5` toggles tag "1" to "5"
|
|
input.keybind(
|
|
[mod_key, Mod::Ctrl],
|
|
tag_name,
|
|
move || {
|
|
if let Some(tg) = tag.get(tag_name) {
|
|
tg.toggle_active();
|
|
}
|
|
},
|
|
KeybindInfo {
|
|
group: Some("Tag".into()),
|
|
description: Some(format!("Toggle tag {tag_name}")),
|
|
},
|
|
);
|
|
|
|
// `mod_key + shift + 1-5` moves the focused window to tag "1" to "5"
|
|
input.keybind(
|
|
[mod_key, Mod::Shift],
|
|
tag_name,
|
|
move || {
|
|
if let Some(tg) = tag.get(tag_name) {
|
|
if let Some(win) = window.get_focused() {
|
|
win.move_to_tag(&tg);
|
|
}
|
|
}
|
|
},
|
|
KeybindInfo {
|
|
group: Some("Tag".into()),
|
|
description: Some(format!("Move the focused window to tag {tag_name}")),
|
|
},
|
|
);
|
|
|
|
// `mod_key + ctrl + shift + 1-5` toggles tag "1" to "5" on the focused window
|
|
input.keybind(
|
|
[mod_key, Mod::Ctrl, Mod::Shift],
|
|
tag_name,
|
|
move || {
|
|
if let Some(tg) = tag.get(tag_name) {
|
|
if let Some(win) = window.get_focused() {
|
|
win.toggle_tag(&tg);
|
|
}
|
|
}
|
|
},
|
|
KeybindInfo {
|
|
group: Some("Tag".into()),
|
|
description: Some(format!("Toggle tag {tag_name} on the focused window")),
|
|
},
|
|
);
|
|
}
|
|
|
|
input.set_libinput_setting(LibinputSetting::Tap(true));
|
|
|
|
// Request all windows use client-side decorations.
|
|
window.add_window_rule(
|
|
WindowRuleCondition::new().all([]),
|
|
WindowRule::new().decoration_mode(DecorationMode::ClientSide),
|
|
);
|
|
|
|
// Enable sloppy focus
|
|
window.connect_signal(WindowSignal::PointerEnter(Box::new(|win| {
|
|
win.set_focused(true);
|
|
})));
|
|
|
|
process.spawn_once([terminal]);
|
|
}
|