1
0
Fork 0
forked from wry/wry

config: move handlers to State

This commit is contained in:
Julian Orth 2026-03-07 14:40:26 +01:00
parent 97f7b68369
commit 6e25180d92
6 changed files with 204 additions and 109 deletions

View file

@ -9,7 +9,6 @@ use {
client::{CAP_JAY_COMPOSITOR, Client, ClientCaps, ClientId}, client::{CAP_JAY_COMPOSITOR, Client, ClientCaps, ClientId},
cmm::cmm_eotf::Eotf, cmm::cmm_eotf::Eotf,
compositor::{MAX_EXTENTS, WAYLAND_DISPLAY}, compositor::{MAX_EXTENTS, WAYLAND_DISPLAY},
config::ConfigProxy,
criteria::{ criteria::{
CritLiteralOrRegex, CritMgrExt, CritTarget, CritUpstreamNode, CritLiteralOrRegex, CritMgrExt, CritTarget, CritUpstreamNode,
clm::ClmLeafMatcher, clm::ClmLeafMatcher,
@ -29,10 +28,9 @@ use {
tagged_acceptor::TaggedAcceptorError, tagged_acceptor::TaggedAcceptorError,
theme::{ThemeColor, ThemeSized}, theme::{ThemeColor, ThemeSized},
tree::{ tree::{
ContainerNode, ContainerSplit, FloatNode, Node, NodeVisitorBase, OutputNode, ContainerSplit, OutputNode, TearingMode, TileState, ToplevelData, ToplevelNode,
TearingMode, TileState, ToplevelData, ToplevelNode, VrrMode, WorkspaceNode, VrrMode, WorkspaceNode, toplevel_create_split, toplevel_parent_container,
toplevel_create_split, toplevel_parent_container, toplevel_set_floating, toplevel_set_floating, toplevel_set_workspace,
toplevel_set_workspace,
}, },
utils::{ utils::{
asyncevent::AsyncEvent, asyncevent::AsyncEvent,
@ -88,7 +86,6 @@ use {
hash::Hash, hash::Hash,
ops::Deref, ops::Deref,
rc::{Rc, Weak}, rc::{Rc, Weak},
sync::Arc,
time::Duration, time::Duration,
}, },
thiserror::Error, thiserror::Error,
@ -424,22 +421,7 @@ impl ConfigProxyHandler {
} }
fn handle_reload(&self) { fn handle_reload(&self) {
log::info!("Reloading config"); self.state.reload_config();
let config = match ConfigProxy::from_config_dir(&self.state) {
Ok(c) => c,
Err(e) => {
log::error!("Cannot reload config: {}", ErrorFmt(e));
return;
}
};
if let Some(config) = self.state.config.take() {
config.destroy();
for seat in self.state.globals.seats.lock().values() {
seat.clear_shortcuts();
}
}
config.configure(true);
self.state.config.set(Some(Rc::new(config)));
} }
fn handle_get_seat_fullscreen(&self, seat: Seat) -> Result<(), CphError> { fn handle_get_seat_fullscreen(&self, seat: Seat) -> Result<(), CphError> {
@ -926,8 +908,7 @@ impl ConfigProxyHandler {
} }
fn handle_set_ei_socket_enabled(&self, enabled: bool) { fn handle_set_ei_socket_enabled(&self, enabled: bool) {
self.state.enable_ei_acceptor.set(enabled); self.state.set_ei_socket_enabled(enabled);
self.state.update_ei_acceptor();
} }
fn handle_get_workspace(&self, name: &str) { fn handle_get_workspace(&self, name: &str) {
@ -971,7 +952,6 @@ impl ConfigProxyHandler {
fn handle_set_flip_margin(&self, device: DrmDevice, margin: Duration) -> Result<(), CphError> { fn handle_set_flip_margin(&self, device: DrmDevice, margin: Duration) -> Result<(), CphError> {
self.get_drm_device(device)? self.get_drm_device(device)?
.dev
.set_flip_margin(margin.as_nanos().try_into().unwrap_or(u64::MAX)); .set_flip_margin(margin.as_nanos().try_into().unwrap_or(u64::MAX));
Ok(()) Ok(())
} }
@ -982,8 +962,7 @@ impl ConfigProxyHandler {
XScalingMode::DOWNSCALED => true, XScalingMode::DOWNSCALED => true,
_ => return Err(CphError::UnknownXScalingMode(mode)), _ => return Err(CphError::UnknownXScalingMode(mode)),
}; };
self.state.xwayland.use_wire_scale.set(use_wire_scale); self.state.set_xwayland_use_wire_scale(use_wire_scale);
self.state.update_xwayland_wire_scale();
Ok(()) Ok(())
} }
@ -993,13 +972,11 @@ impl ConfigProxyHandler {
} }
fn handle_set_ui_drag_enabled(&self, enabled: bool) { fn handle_set_ui_drag_enabled(&self, enabled: bool) {
self.state.ui_drag_enabled.set(enabled); self.state.set_ui_drag_enabled(enabled);
} }
fn handle_set_ui_drag_threshold(&self, threshold: i32) { fn handle_set_ui_drag_threshold(&self, threshold: i32) {
let threshold = threshold.max(1); self.state.set_ui_drag_threshold(threshold.max(1));
let squared = threshold.saturating_mul(threshold);
self.state.ui_drag_threshold_squared.set(squared);
} }
fn handle_set_direct_scanout_enabled( fn handle_set_direct_scanout_enabled(
@ -1010,7 +987,6 @@ impl ConfigProxyHandler {
match device { match device {
Some(dev) => self Some(dev) => self
.get_drm_device(dev)? .get_drm_device(dev)?
.dev
.set_direct_scanout_enabled(enabled), .set_direct_scanout_enabled(enabled),
_ => self.state.direct_scanout_enabled.set(enabled), _ => self.state.direct_scanout_enabled.set(enabled),
} }
@ -1403,12 +1379,7 @@ impl ConfigProxyHandler {
} }
fn handle_set_float_above_fullscreen(&self, above: bool) { fn handle_set_float_above_fullscreen(&self, above: bool) {
self.state.float_above_fullscreen.set(above); self.state.set_float_above_fullscreen(above);
for seat in self.state.globals.seats.lock().values() {
seat.emulate_cursor_moved();
seat.trigger_tree_changed(false);
}
self.state.root.update_visible(&self.state);
} }
fn handle_get_float_above_fullscreen(&self) { fn handle_get_float_above_fullscreen(&self) {
@ -1418,10 +1389,7 @@ impl ConfigProxyHandler {
} }
fn handle_set_show_bar(&self, show: bool) { fn handle_set_show_bar(&self, show: bool) {
self.state.show_bar.set(show); self.state.set_show_bar(show);
for output in self.state.root.outputs.lock().values() {
output.on_spaces_changed();
}
} }
fn handle_get_show_bar(&self) { fn handle_get_show_bar(&self) {
@ -1431,8 +1399,7 @@ impl ConfigProxyHandler {
} }
fn handle_set_show_titles(&self, show: bool) { fn handle_set_show_titles(&self, show: bool) {
self.state.theme.show_titles.set(show); self.state.set_show_titles(show);
self.spaces_change();
} }
fn handle_get_show_titles(&self) { fn handle_get_show_titles(&self) {
@ -1445,8 +1412,7 @@ impl ConfigProxyHandler {
let Ok(position) = position.try_into() else { let Ok(position) = position.try_into() else {
return Err(CphError::UnknownBarPosition(position)); return Err(CphError::UnknownBarPosition(position));
}; };
self.state.theme.bar_position.set(position); self.state.set_bar_position(position);
self.spaces_change();
Ok(()) Ok(())
} }
@ -1457,19 +1423,11 @@ impl ConfigProxyHandler {
} }
fn handle_set_show_float_pin_icon(&self, show: bool) { fn handle_set_show_float_pin_icon(&self, show: bool) {
self.state.show_pin_icon.set(show); self.state.set_show_pin_icon(show);
for stacked in self.state.root.stacked.iter() {
if let Some(float) = stacked.deref().clone().node_into_float() {
float.schedule_render_titles();
}
}
} }
fn handle_set_workspace_display_order(&self, order: WorkspaceDisplayOrder) { fn handle_set_workspace_display_order(&self, order: WorkspaceDisplayOrder) {
self.state.workspace_display_order.set(order.into()); self.state.set_workspace_display_order(order.into());
for output in self.state.root.outputs.lock().values() {
output.handle_workspace_display_order_update();
}
} }
fn handle_get_seat_float_pinned(&self, seat: Seat) -> Result<(), CphError> { fn handle_get_seat_float_pinned(&self, seat: Seat) -> Result<(), CphError> {
@ -2412,27 +2370,6 @@ impl ConfigProxyHandler {
Ok(()) Ok(())
} }
fn spaces_change(&self) {
struct V;
impl NodeVisitorBase for V {
fn visit_output(&mut self, node: &Rc<OutputNode>) {
node.on_spaces_changed();
node.node_visit_children(self);
}
fn visit_container(&mut self, node: &Rc<ContainerNode>) {
node.on_spaces_changed();
node.node_visit_children(self);
}
fn visit_float(&mut self, node: &Rc<FloatNode>) {
node.on_spaces_changed();
node.node_visit_children(self);
}
}
self.state.root.clone().node_visit(&mut V);
self.state.damage(self.state.root.extents.get());
self.state.icons.update_sizes(&self.state);
}
fn get_sized(&self, sized: Resizable) -> Result<ThemeSized, CphError> { fn get_sized(&self, sized: Resizable) -> Result<ThemeSized, CphError> {
use jay_config::theme::sized::*; use jay_config::theme::sized::*;
let sized = match sized { let sized = match sized {
@ -2460,10 +2397,7 @@ impl ConfigProxyHandler {
if size > sized.max() { if size > sized.max() {
return Err(CphError::InvalidSize(size, sized)); return Err(CphError::InvalidSize(size, sized));
} }
let field = sized.field(&self.state.theme); self.state.set_size(sized, size);
field.val.set(size);
field.set.set(true);
self.spaces_change();
Ok(()) Ok(())
} }
@ -2472,33 +2406,23 @@ impl ConfigProxyHandler {
} }
fn handle_reset_sizes(&self) { fn handle_reset_sizes(&self) {
self.state.theme.sizes.reset(); self.state.reset_sizes();
self.spaces_change();
} }
fn handle_reset_font(&self) { fn handle_reset_font(&self) {
let theme = &self.state.theme; self.state.reset_fonts();
theme.font.set(self.state.theme.default_font.clone());
theme.bar_font.set(None);
theme.title_font.set(None);
} }
fn handle_set_font(&self, font: &str) { fn handle_set_font(&self, font: &str) {
self.state.theme.font.set(Arc::new(font.to_string())); self.state.set_font(font);
} }
fn handle_set_bar_font(&self, font: &str) { fn handle_set_bar_font(&self, font: &str) {
self.state self.state.set_bar_font(Some(font));
.theme
.bar_font
.set(Some(Arc::new(font.to_string())));
} }
fn handle_set_title_font(&self, font: &str) { fn handle_set_title_font(&self, font: &str) {
self.state self.state.set_title_font(Some(font));
.theme
.title_font
.set(Some(Arc::new(font.to_string())));
} }
fn handle_get_font(&self) { fn handle_get_font(&self) {

View file

@ -179,8 +179,7 @@ impl JayCompositorRequestHandler for JayCompositor {
} }
fn quit(&self, _req: Quit, _slf: &Rc<Self>) -> Result<(), Self::Error> { fn quit(&self, _req: Quit, _slf: &Rc<Self>) -> Result<(), Self::Error> {
log::info!("Quitting"); self.client.state.quit();
self.client.state.ring.stop();
Ok(()) Ok(())
} }

View file

@ -67,13 +67,15 @@ impl JayIdleRequestHandler for JayIdle {
fn set_interval(&self, req: SetInterval, _slf: &Rc<Self>) -> Result<(), Self::Error> { fn set_interval(&self, req: SetInterval, _slf: &Rc<Self>) -> Result<(), Self::Error> {
let interval = Duration::from_secs(req.interval); let interval = Duration::from_secs(req.interval);
self.client.state.idle.set_timeout(interval); let state = &self.client.state;
state.idle.set_timeout(interval);
Ok(()) Ok(())
} }
fn set_grace_period(&self, req: SetGracePeriod, _slf: &Rc<Self>) -> Result<(), Self::Error> { fn set_grace_period(&self, req: SetGracePeriod, _slf: &Rc<Self>) -> Result<(), Self::Error> {
let period = Duration::from_secs(req.period); let period = Duration::from_secs(req.period);
self.client.state.idle.set_grace_period(period); let state = &self.client.state;
state.idle.set_grace_period(period);
Ok(()) Ok(())
} }
} }

View file

@ -350,7 +350,7 @@ impl JayRandrRequestHandler for JayRandr {
let Some(dev) = self.get_device(req.dev) else { let Some(dev) = self.get_device(req.dev) else {
return Ok(()); return Ok(());
}; };
dev.dev.set_direct_scanout_enabled(req.enabled != 0); dev.set_direct_scanout_enabled(req.enabled != 0);
Ok(()) Ok(())
} }
@ -493,7 +493,7 @@ impl JayRandrRequestHandler for JayRandr {
let Some(dev) = self.get_device(req.dev) else { let Some(dev) = self.get_device(req.dev) else {
return Ok(()); return Ok(());
}; };
dev.dev.set_flip_margin(req.margin_ns); dev.set_flip_margin(req.margin_ns);
Ok(()) Ok(())
} }

View file

@ -57,10 +57,7 @@ impl JayXwaylandRequestHandler for JayXwayland {
}; };
self.client self.client
.state .state
.xwayland .set_xwayland_use_wire_scale(use_wire_scale);
.use_wire_scale
.set(use_wire_scale);
self.client.state.update_xwayland_wire_scale();
Ok(()) Ok(())
} }
} }

View file

@ -93,7 +93,7 @@ use {
scale::Scale, scale::Scale,
security_context_acceptor::SecurityContextAcceptors, security_context_acceptor::SecurityContextAcceptors,
tagged_acceptor::TaggedAcceptors, tagged_acceptor::TaggedAcceptors,
theme::{Color, Theme, ThemeColor}, theme::{BarPosition, Color, Theme, ThemeColor, ThemeSized},
time::Time, time::Time,
tree::{ tree::{
ContainerNode, ContainerSplit, Direction, DisplayNode, FindTreeUsecase, FloatNode, ContainerNode, ContainerSplit, Direction, DisplayNode, FindTreeUsecase, FloatNode,
@ -341,12 +341,15 @@ pub struct IdleState {
impl IdleState { impl IdleState {
pub fn set_timeout(&self, timeout: Duration) { pub fn set_timeout(&self, timeout: Duration) {
self.timeout.set(timeout); self.timeout.set(timeout);
self.timeout_changed.set(true); self.timeout_changed();
self.change.trigger();
} }
pub fn set_grace_period(&self, grace_period: Duration) { pub fn set_grace_period(&self, grace_period: Duration) {
self.grace_period.set(grace_period); self.grace_period.set(grace_period);
self.timeout_changed();
}
fn timeout_changed(&self) {
self.timeout_changed.set(true); self.timeout_changed.set(true);
self.change.trigger(); self.change.trigger();
} }
@ -520,6 +523,14 @@ impl DrmDevData {
); );
self.dev.clone().make_render_device(); self.dev.clone().make_render_device();
} }
pub fn set_direct_scanout_enabled(&self, enabled: bool) {
self.dev.set_direct_scanout_enabled(enabled);
}
pub fn set_flip_margin(&self, margin: u64) {
self.dev.set_flip_margin(margin);
}
} }
struct UpdateTextTexturesVisitor; struct UpdateTextTexturesVisitor;
@ -987,6 +998,13 @@ impl State {
} }
} }
pub fn set_xwayland_use_wire_scale(&self, use_wire_scale: bool) {
if self.xwayland.use_wire_scale.replace(use_wire_scale) == use_wire_scale {
return;
}
self.update_xwayland_wire_scale();
}
pub fn next_serial(&self, client: Option<&Client>) -> u64 { pub fn next_serial(&self, client: Option<&Client>) -> u64 {
let serial = self.serial.fetch_add(1); let serial = self.serial.fetch_add(1);
if let Some(client) = client { if let Some(client) = client {
@ -1727,6 +1745,161 @@ impl State {
self.colors_changed(); self.colors_changed();
} }
pub fn quit(&self) {
log::info!("Quitting");
self.ring.stop();
}
pub fn reload_config(self: &Rc<Self>) {
log::info!("Reloading config");
let config = match ConfigProxy::from_config_dir(self) {
Ok(c) => c,
Err(e) => {
log::error!("Cannot reload config: {}", ErrorFmt(e));
return;
}
};
if let Some(config) = self.config.take() {
config.destroy();
for seat in self.globals.seats.lock().values() {
seat.clear_shortcuts();
}
}
config.configure(true);
self.config.set(Some(Rc::new(config)));
}
pub fn set_ei_socket_enabled(self: &Rc<Self>, enabled: bool) {
self.enable_ei_acceptor.set(enabled);
self.update_ei_acceptor();
}
pub fn set_workspace_display_order(&self, order: WorkspaceDisplayOrder) {
self.workspace_display_order.set(order);
for output in self.root.outputs.lock().values() {
output.handle_workspace_display_order_update();
}
}
fn spaces_changed(&self) {
struct V;
impl NodeVisitorBase for V {
fn visit_container(&mut self, node: &Rc<ContainerNode>) {
node.on_spaces_changed();
node.node_visit_children(self);
}
fn visit_output(&mut self, node: &Rc<OutputNode>) {
node.on_spaces_changed();
node.node_visit_children(self);
}
fn visit_float(&mut self, node: &Rc<FloatNode>) {
node.on_spaces_changed();
node.node_visit_children(self);
}
}
self.root.clone().node_visit(&mut V);
self.damage(self.root.extents.get());
self.icons.update_sizes(self);
}
pub fn set_show_bar(&self, show: bool) {
self.show_bar.set(show);
self.spaces_changed();
}
pub fn set_show_titles(&self, show: bool) {
self.theme.show_titles.set(show);
self.spaces_changed();
}
pub fn set_ui_drag_enabled(&self, enabled: bool) {
self.ui_drag_enabled.set(enabled);
}
pub fn set_ui_drag_threshold(&self, threshold: i32) {
self.ui_drag_threshold_squared
.set(threshold.saturating_mul(threshold));
}
pub fn set_show_pin_icon(&self, show: bool) {
self.show_pin_icon.set(show);
for stacked in self.root.stacked.iter() {
if let Some(float) = stacked.deref().clone().node_into_float() {
float.schedule_render_titles();
}
}
}
pub fn set_float_above_fullscreen(&self, v: bool) {
self.float_above_fullscreen.set(v);
for seat in self.globals.seats.lock().values() {
seat.emulate_cursor_moved();
seat.trigger_tree_changed(false);
}
self.root.update_visible(self);
}
pub fn reset_sizes(&self) {
self.theme.sizes.reset();
self.spaces_changed();
}
fn fonts_changed(&self) {
struct V;
impl NodeVisitorBase for V {
fn visit_container(&mut self, node: &Rc<ContainerNode>) {
node.schedule_render_titles();
node.node_visit_children(self);
}
fn visit_output(&mut self, node: &Rc<OutputNode>) {
node.schedule_update_render_data();
node.node_visit_children(self);
}
fn visit_float(&mut self, node: &Rc<FloatNode>) {
node.schedule_render_titles();
node.node_visit_children(self);
}
}
self.root.clone().node_visit(&mut V);
}
pub fn reset_fonts(&self) {
let theme = &self.theme;
theme.font.set(self.theme.default_font.clone());
theme.bar_font.set(None);
theme.title_font.set(None);
self.fonts_changed();
}
pub fn set_font(&self, font: &str) {
self.theme.font.set(Arc::new(font.to_string()));
self.fonts_changed();
}
pub fn set_bar_font(&self, font: Option<&str>) {
let font = font.map(|font| Arc::new(font.to_string()));
self.theme.bar_font.set(font);
self.fonts_changed();
}
pub fn set_title_font(&self, font: Option<&str>) {
let font = font.map(|font| Arc::new(font.to_string()));
self.theme.title_font.set(font);
self.fonts_changed();
}
pub fn set_bar_position(&self, p: BarPosition) {
self.theme.bar_position.set(p);
self.spaces_changed();
}
pub fn set_size(&self, sized: ThemeSized, size: i32) {
let field = sized.field(&self.theme);
field.val.set(size);
field.set.set(true);
self.spaces_changed();
}
pub fn set_color(&self, colored: ThemeColor, v: Color) { pub fn set_color(&self, colored: ThemeColor, v: Color) {
colored.field(&self.theme).set(v); colored.field(&self.theme).set(v);
self.colors_changed(); self.colors_changed();