mirror of
https://github.com/pinnacle-comp/pinnacle.git
synced 2025-01-15 15:42:06 +01:00
Merge pull request #238 from pinnacle-comp/session_lock
Implement session lock
This commit is contained in:
commit
6e3b1b4960
14 changed files with 501 additions and 114 deletions
6
Cargo.lock
generated
6
Cargo.lock
generated
|
@ -1410,7 +1410,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "0c2a198fb6b0eada2a8df47933734e6d35d350665a33a3593d7164fa52c75c19"
|
checksum = "0c2a198fb6b0eada2a8df47933734e6d35d350665a33a3593d7164fa52c75c19"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"cfg-if",
|
"cfg-if",
|
||||||
"windows-targets 0.52.5",
|
"windows-targets 0.48.5",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -2381,7 +2381,7 @@ checksum = "b7c388c1b5e93756d0c740965c41e8822f866621d41acbdf6336a6a168f8840c"
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "smithay"
|
name = "smithay"
|
||||||
version = "0.3.0"
|
version = "0.3.0"
|
||||||
source = "git+https://github.com/Smithay/smithay?rev=95d6e79#95d6e79ed84d6e58ad75e3f1ead154dcd04719f9"
|
source = "git+https://github.com/Smithay/smithay?rev=0398269#0398269eab5e0e1be212d960f7010dfb53f07978"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"appendlist",
|
"appendlist",
|
||||||
"ash",
|
"ash",
|
||||||
|
@ -2456,7 +2456,7 @@ dependencies = [
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "smithay-drm-extras"
|
name = "smithay-drm-extras"
|
||||||
version = "0.1.0"
|
version = "0.1.0"
|
||||||
source = "git+https://github.com/Smithay/smithay?rev=95d6e79#95d6e79ed84d6e58ad75e3f1ead154dcd04719f9"
|
source = "git+https://github.com/Smithay/smithay?rev=0398269#0398269eab5e0e1be212d960f7010dfb53f07978"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"drm",
|
"drm",
|
||||||
"edid-rs",
|
"edid-rs",
|
||||||
|
|
|
@ -35,7 +35,7 @@ tempfile = "3.10.1"
|
||||||
|
|
||||||
[workspace.dependencies.smithay]
|
[workspace.dependencies.smithay]
|
||||||
git = "https://github.com/Smithay/smithay"
|
git = "https://github.com/Smithay/smithay"
|
||||||
rev = "95d6e79"
|
rev = "0398269"
|
||||||
default-features = false
|
default-features = false
|
||||||
features = [
|
features = [
|
||||||
"desktop",
|
"desktop",
|
||||||
|
@ -77,7 +77,7 @@ keywords = ["wayland", "compositor", "smithay", "lua"]
|
||||||
[dependencies]
|
[dependencies]
|
||||||
# Smithay
|
# Smithay
|
||||||
smithay = { workspace = true }
|
smithay = { workspace = true }
|
||||||
smithay-drm-extras = { git = "https://github.com/Smithay/smithay", rev = "95d6e79" }
|
smithay-drm-extras = { git = "https://github.com/Smithay/smithay", rev = "0398269" }
|
||||||
# Tracing
|
# Tracing
|
||||||
tracing = { workspace = true }
|
tracing = { workspace = true }
|
||||||
tracing-subscriber = { workspace = true }
|
tracing-subscriber = { workspace = true }
|
||||||
|
@ -115,7 +115,6 @@ bytemuck = "1.15.0"
|
||||||
pinnacle-api = { path = "./api/rust" }
|
pinnacle-api = { path = "./api/rust" }
|
||||||
gag = "1.0.0"
|
gag = "1.0.0"
|
||||||
|
|
||||||
|
|
||||||
[build-dependencies]
|
[build-dependencies]
|
||||||
vergen = { version = "8.3.1", features = ["git", "gitcl", "rustc", "cargo", "si"] }
|
vergen = { version = "8.3.1", features = ["git", "gitcl", "rustc", "cargo", "si"] }
|
||||||
|
|
||||||
|
|
|
@ -33,7 +33,7 @@ use smithay::{
|
||||||
use tracing::error;
|
use tracing::error;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
state::{Pinnacle, State, SurfaceDmabufFeedback},
|
state::{Pinnacle, State, SurfaceDmabufFeedback, WithState},
|
||||||
window::WindowElement,
|
window::WindowElement,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -248,6 +248,16 @@ pub fn post_repaint(
|
||||||
if let CursorImageStatus::Surface(surf) = cursor_status {
|
if let CursorImageStatus::Surface(surf) = cursor_status {
|
||||||
send_frames_surface_tree(surf, output, time, Some(Duration::ZERO), |_, _| None);
|
send_frames_surface_tree(surf, output, time, Some(Duration::ZERO), |_, _| None);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let Some(lock_surface) = output.with_state(|state| state.lock_surface.clone()) {
|
||||||
|
send_frames_surface_tree(
|
||||||
|
lock_surface.wl_surface(),
|
||||||
|
output,
|
||||||
|
time,
|
||||||
|
Some(Duration::ZERO),
|
||||||
|
|_, _| None,
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DmabufHandler for State {
|
impl DmabufHandler for State {
|
||||||
|
|
|
@ -32,7 +32,11 @@ use smithay::{
|
||||||
renderer::{
|
renderer::{
|
||||||
self, damage,
|
self, damage,
|
||||||
element::{
|
element::{
|
||||||
self, surface::WaylandSurfaceRenderElement, texture::TextureBuffer, Element,
|
self,
|
||||||
|
solid::{SolidColorBuffer, SolidColorRenderElement},
|
||||||
|
surface::{render_elements_from_surface_tree, WaylandSurfaceRenderElement},
|
||||||
|
texture::TextureBuffer,
|
||||||
|
Element,
|
||||||
},
|
},
|
||||||
gles::{GlesRenderbuffer, GlesRenderer},
|
gles::{GlesRenderbuffer, GlesRenderer},
|
||||||
multigpu::{gbm::GbmGlesBackend, GpuManager, MultiRenderer, MultiTexture},
|
multigpu::{gbm::GbmGlesBackend, GpuManager, MultiRenderer, MultiTexture},
|
||||||
|
@ -88,7 +92,7 @@ use tracing::{debug, error, info, trace, warn};
|
||||||
use crate::{
|
use crate::{
|
||||||
backend::Backend,
|
backend::Backend,
|
||||||
config::ConnectorSavedState,
|
config::ConnectorSavedState,
|
||||||
output::OutputName,
|
output::{BlankingState, OutputName},
|
||||||
render::{
|
render::{
|
||||||
pointer::PointerElement, pointer_render_elements, take_presentation_feedback,
|
pointer::PointerElement, pointer_render_elements, take_presentation_feedback,
|
||||||
OutputRenderElement,
|
OutputRenderElement,
|
||||||
|
@ -1312,6 +1316,13 @@ impl Udev {
|
||||||
flags,
|
flags,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
output.with_state_mut(|state| {
|
||||||
|
if let BlankingState::Blanking = state.blanking_state {
|
||||||
|
debug!("Output {} blanked", output.name());
|
||||||
|
state.blanking_state = BlankingState::Blanked;
|
||||||
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
Err(err) => {
|
Err(err) => {
|
||||||
warn!("Error during rendering: {:?}", err);
|
warn!("Error during rendering: {:?}", err);
|
||||||
|
@ -1399,8 +1410,6 @@ impl Udev {
|
||||||
texture
|
texture
|
||||||
});
|
});
|
||||||
|
|
||||||
let windows = pinnacle.space.elements().cloned().collect::<Vec<_>>();
|
|
||||||
|
|
||||||
let pointer_location = pinnacle
|
let pointer_location = pinnacle
|
||||||
.seat
|
.seat
|
||||||
.get_pointer()
|
.get_pointer()
|
||||||
|
@ -1434,10 +1443,14 @@ impl Udev {
|
||||||
|
|
||||||
let mut output_render_elements = Vec::new();
|
let mut output_render_elements = Vec::new();
|
||||||
|
|
||||||
|
let should_blank = pinnacle.lock_state.is_locking()
|
||||||
|
|| (pinnacle.lock_state.is_locked()
|
||||||
|
&& output.with_state(|state| state.lock_surface.is_none()));
|
||||||
|
|
||||||
// If there isn't a pending screencopy that doesn't want to overlay the cursor,
|
// If there isn't a pending screencopy that doesn't want to overlay the cursor,
|
||||||
// render it.
|
// render it.
|
||||||
match pending_screencopy_with_cursor {
|
match pending_screencopy_with_cursor {
|
||||||
Some(include_cursor) => {
|
Some(include_cursor) if pinnacle.lock_state.is_unlocked() => {
|
||||||
if include_cursor {
|
if include_cursor {
|
||||||
// HACK: Doing `RenderFrameResult::blit_frame_result` with something on the
|
// HACK: Doing `RenderFrameResult::blit_frame_result` with something on the
|
||||||
// | cursor plane causes the cursor to overwrite the pixels underneath it,
|
// | cursor plane causes the cursor to overwrite the pixels underneath it,
|
||||||
|
@ -1461,7 +1474,7 @@ impl Udev {
|
||||||
output_render_elements.extend(pointer_render_elements);
|
output_render_elements.extend(pointer_render_elements);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
None => {
|
_ => {
|
||||||
let pointer_render_elements = pointer_render_elements(
|
let pointer_render_elements = pointer_render_elements(
|
||||||
output,
|
output,
|
||||||
&mut renderer,
|
&mut renderer,
|
||||||
|
@ -1475,12 +1488,53 @@ impl Udev {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if should_blank {
|
||||||
|
let output_size = pinnacle
|
||||||
|
.space
|
||||||
|
.output_geometry(output)
|
||||||
|
.map(|geo| geo.size)
|
||||||
|
.unwrap_or((99999, 99999).into());
|
||||||
|
|
||||||
|
let solid_color_buffer = SolidColorBuffer::new(output_size, [0.2, 0.0, 0.3, 1.0]);
|
||||||
|
let solid_color_element = SolidColorRenderElement::from_buffer(
|
||||||
|
&solid_color_buffer,
|
||||||
|
(0, 0),
|
||||||
|
output.current_scale().fractional_scale(),
|
||||||
|
1.0,
|
||||||
|
element::Kind::Unspecified,
|
||||||
|
);
|
||||||
|
|
||||||
|
output_render_elements.push(OutputRenderElement::from(solid_color_element));
|
||||||
|
|
||||||
|
output.with_state_mut(|state| {
|
||||||
|
if let BlankingState::NotBlanked = state.blanking_state {
|
||||||
|
debug!("Blanking output {} for session lock", output.name());
|
||||||
|
state.blanking_state = BlankingState::Blanking;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else if pinnacle.lock_state.is_locked() {
|
||||||
|
if let Some(lock_surface) = output.with_state(|state| state.lock_surface.clone()) {
|
||||||
|
let elems = render_elements_from_surface_tree(
|
||||||
|
&mut renderer,
|
||||||
|
lock_surface.wl_surface(),
|
||||||
|
(0, 0),
|
||||||
|
output.current_scale().fractional_scale(),
|
||||||
|
1.0,
|
||||||
|
element::Kind::Unspecified,
|
||||||
|
);
|
||||||
|
|
||||||
|
output_render_elements.extend(elems);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
let windows = pinnacle.space.elements().cloned().collect::<Vec<_>>();
|
||||||
|
|
||||||
output_render_elements.extend(crate::render::output_render_elements(
|
output_render_elements.extend(crate::render::output_render_elements(
|
||||||
output,
|
output,
|
||||||
&mut renderer,
|
&mut renderer,
|
||||||
&pinnacle.space,
|
&pinnacle.space,
|
||||||
&windows,
|
&windows,
|
||||||
));
|
));
|
||||||
|
}
|
||||||
|
|
||||||
let result = (|| -> Result<bool, SwapBuffersError> {
|
let result = (|| -> Result<bool, SwapBuffersError> {
|
||||||
let render_frame_result = render_frame(
|
let render_frame_result = render_frame(
|
||||||
|
@ -1496,6 +1550,7 @@ impl Udev {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if pinnacle.lock_state.is_unlocked() {
|
||||||
handle_pending_screencopy(
|
handle_pending_screencopy(
|
||||||
&mut renderer,
|
&mut renderer,
|
||||||
output,
|
output,
|
||||||
|
@ -1503,6 +1558,7 @@ impl Udev {
|
||||||
&render_frame_result,
|
&render_frame_result,
|
||||||
&pinnacle.loop_handle,
|
&pinnacle.loop_handle,
|
||||||
);
|
);
|
||||||
|
}
|
||||||
|
|
||||||
super::post_repaint(
|
super::post_repaint(
|
||||||
output,
|
output,
|
||||||
|
|
|
@ -9,6 +9,11 @@ use smithay::{
|
||||||
renderer::{
|
renderer::{
|
||||||
self, buffer_type,
|
self, buffer_type,
|
||||||
damage::{self, OutputDamageTracker, RenderOutputResult},
|
damage::{self, OutputDamageTracker, RenderOutputResult},
|
||||||
|
element::{
|
||||||
|
self,
|
||||||
|
solid::{SolidColorBuffer, SolidColorRenderElement},
|
||||||
|
surface::render_elements_from_surface_tree,
|
||||||
|
},
|
||||||
gles::{GlesRenderbuffer, GlesRenderer, GlesTexture},
|
gles::{GlesRenderbuffer, GlesRenderer, GlesTexture},
|
||||||
Bind, Blit, BufferType, ExportMem, ImportDma, ImportEgl, ImportMemWl, Offscreen,
|
Bind, Blit, BufferType, ExportMem, ImportDma, ImportEgl, ImportMemWl, Offscreen,
|
||||||
TextureFilter,
|
TextureFilter,
|
||||||
|
@ -37,10 +42,14 @@ use smithay::{
|
||||||
utils::{IsAlive, Point, Rectangle, Transform},
|
utils::{IsAlive, Point, Rectangle, Transform},
|
||||||
wayland::dmabuf::{self, DmabufFeedback, DmabufFeedbackBuilder, DmabufGlobal, DmabufState},
|
wayland::dmabuf::{self, DmabufFeedback, DmabufFeedbackBuilder, DmabufGlobal, DmabufState},
|
||||||
};
|
};
|
||||||
use tracing::{error, trace, warn};
|
use tracing::{debug, error, trace, warn};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
render::{pointer::PointerElement, pointer_render_elements, take_presentation_feedback},
|
output::BlankingState,
|
||||||
|
render::{
|
||||||
|
pointer::PointerElement, pointer_render_elements, take_presentation_feedback,
|
||||||
|
OutputRenderElement,
|
||||||
|
},
|
||||||
state::{Pinnacle, State, WithState},
|
state::{Pinnacle, State, WithState},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -200,6 +209,7 @@ impl Winit {
|
||||||
Some(mode),
|
Some(mode),
|
||||||
None,
|
None,
|
||||||
Some(Scale::Fractional(scale_factor)),
|
Some(Scale::Fractional(scale_factor)),
|
||||||
|
// None,
|
||||||
None,
|
None,
|
||||||
);
|
);
|
||||||
state.pinnacle.request_layout(&output);
|
state.pinnacle.request_layout(&output);
|
||||||
|
@ -266,19 +276,16 @@ impl State {
|
||||||
|
|
||||||
let mut output_render_elements = Vec::new();
|
let mut output_render_elements = Vec::new();
|
||||||
|
|
||||||
let pending_screencopy_without_cursor = output.with_state(|state| {
|
let should_draw_cursor = !self.pinnacle.lock_state.is_unlocked()
|
||||||
state
|
|| output.with_state(|state| {
|
||||||
|
// Don't draw cursor when screencopy without cursor is pending
|
||||||
|
!state
|
||||||
.screencopy
|
.screencopy
|
||||||
.as_ref()
|
.as_ref()
|
||||||
.is_some_and(|sc| !sc.overlay_cursor())
|
.is_some_and(|sc| !sc.overlay_cursor())
|
||||||
});
|
});
|
||||||
|
|
||||||
// If there isn't a pending screencopy that doesn't want to overlay the cursor,
|
if should_draw_cursor {
|
||||||
// render it.
|
|
||||||
//
|
|
||||||
// This will cause the cursor to disappear for a frame if there is one though,
|
|
||||||
// but it shouldn't meaningfully affect anything.
|
|
||||||
if !pending_screencopy_without_cursor {
|
|
||||||
let pointer_location = self
|
let pointer_location = self
|
||||||
.pinnacle
|
.pinnacle
|
||||||
.seat
|
.seat
|
||||||
|
@ -298,12 +305,56 @@ impl State {
|
||||||
output_render_elements.extend(pointer_render_elements);
|
output_render_elements.extend(pointer_render_elements);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let should_blank = self.pinnacle.lock_state.is_locking()
|
||||||
|
|| (self.pinnacle.lock_state.is_locked()
|
||||||
|
&& output.with_state(|state| state.lock_surface.is_none()));
|
||||||
|
|
||||||
|
if should_blank {
|
||||||
|
let output_size = self
|
||||||
|
.pinnacle
|
||||||
|
.space
|
||||||
|
.output_geometry(output)
|
||||||
|
.map(|geo| geo.size)
|
||||||
|
.unwrap_or((99999, 99999).into());
|
||||||
|
|
||||||
|
let solid_color_buffer = SolidColorBuffer::new(output_size, [0.2, 0.0, 0.3, 1.0]);
|
||||||
|
let solid_color_element = SolidColorRenderElement::from_buffer(
|
||||||
|
&solid_color_buffer,
|
||||||
|
(0, 0),
|
||||||
|
output.current_scale().fractional_scale(),
|
||||||
|
1.0,
|
||||||
|
element::Kind::Unspecified,
|
||||||
|
);
|
||||||
|
|
||||||
|
output_render_elements.push(OutputRenderElement::from(solid_color_element));
|
||||||
|
|
||||||
|
output.with_state_mut(|state| {
|
||||||
|
if let BlankingState::NotBlanked = state.blanking_state {
|
||||||
|
debug!("Blanking output {} for session lock", output.name());
|
||||||
|
state.blanking_state = BlankingState::Blanking;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else if self.pinnacle.lock_state.is_locked() {
|
||||||
|
if let Some(lock_surface) = output.with_state(|state| state.lock_surface.clone()) {
|
||||||
|
let elems = render_elements_from_surface_tree(
|
||||||
|
winit.backend.renderer(),
|
||||||
|
lock_surface.wl_surface(),
|
||||||
|
(0, 0),
|
||||||
|
output.current_scale().fractional_scale(),
|
||||||
|
1.0,
|
||||||
|
element::Kind::Unspecified,
|
||||||
|
);
|
||||||
|
|
||||||
|
output_render_elements.extend(elems);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
output_render_elements.extend(crate::render::output_render_elements(
|
output_render_elements.extend(crate::render::output_render_elements(
|
||||||
output,
|
output,
|
||||||
winit.backend.renderer(),
|
winit.backend.renderer(),
|
||||||
&self.pinnacle.space,
|
&self.pinnacle.space,
|
||||||
&windows,
|
&windows,
|
||||||
));
|
));
|
||||||
|
}
|
||||||
|
|
||||||
let render_res = winit.backend.bind().and_then(|_| {
|
let render_res = winit.backend.bind().and_then(|_| {
|
||||||
let age = if *full_redraw > 0 {
|
let age = if *full_redraw > 0 {
|
||||||
|
@ -325,19 +376,32 @@ impl State {
|
||||||
|
|
||||||
match render_res {
|
match render_res {
|
||||||
Ok(render_output_result) => {
|
Ok(render_output_result) => {
|
||||||
|
if self.pinnacle.lock_state.is_unlocked() {
|
||||||
Winit::handle_pending_screencopy(
|
Winit::handle_pending_screencopy(
|
||||||
&mut winit.backend,
|
&mut winit.backend,
|
||||||
output,
|
output,
|
||||||
&render_output_result,
|
&render_output_result,
|
||||||
&self.pinnacle.loop_handle,
|
&self.pinnacle.loop_handle,
|
||||||
);
|
);
|
||||||
|
}
|
||||||
|
|
||||||
let has_rendered = render_output_result.damage.is_some();
|
let has_rendered = render_output_result.damage.is_some();
|
||||||
if let Some(damage) = render_output_result.damage {
|
if let Some(damage) = render_output_result.damage {
|
||||||
if let Err(err) = winit.backend.submit(Some(damage)) {
|
match winit.backend.submit(Some(damage)) {
|
||||||
|
Ok(()) => {
|
||||||
|
output.with_state_mut(|state| {
|
||||||
|
if matches!(state.blanking_state, BlankingState::Blanking) {
|
||||||
|
// TODO: this is probably wrong
|
||||||
|
debug!("Output {} blanked", output.name());
|
||||||
|
state.blanking_state = BlankingState::Blanked;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
Err(err) => {
|
||||||
error!("Failed to submit buffer: {}", err);
|
error!("Failed to submit buffer: {}", err);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
winit.backend.window().set_cursor_visible(cursor_visible);
|
winit.backend.window().set_cursor_visible(cursor_visible);
|
||||||
|
|
||||||
|
|
|
@ -7,7 +7,7 @@ use smithay::{
|
||||||
},
|
},
|
||||||
reexports::wayland_server::{protocol::wl_surface::WlSurface, Resource},
|
reexports::wayland_server::{protocol::wl_surface::WlSurface, Resource},
|
||||||
utils::{IsAlive, Serial},
|
utils::{IsAlive, Serial},
|
||||||
wayland::seat::WaylandFocus,
|
wayland::{seat::WaylandFocus, session_lock::LockSurface},
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::{state::State, window::WindowElement};
|
use crate::{state::State, window::WindowElement};
|
||||||
|
@ -18,6 +18,7 @@ pub enum KeyboardFocusTarget {
|
||||||
Window(WindowElement),
|
Window(WindowElement),
|
||||||
Popup(PopupKind),
|
Popup(PopupKind),
|
||||||
LayerSurface(LayerSurface),
|
LayerSurface(LayerSurface),
|
||||||
|
LockSurface(LockSurface),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl KeyboardTarget<State> for KeyboardFocusTarget {
|
impl KeyboardTarget<State> for KeyboardFocusTarget {
|
||||||
|
@ -38,6 +39,9 @@ impl KeyboardTarget<State> for KeyboardFocusTarget {
|
||||||
KeyboardFocusTarget::LayerSurface(surf) => {
|
KeyboardFocusTarget::LayerSurface(surf) => {
|
||||||
KeyboardTarget::enter(surf.wl_surface(), seat, data, keys, serial);
|
KeyboardTarget::enter(surf.wl_surface(), seat, data, keys, serial);
|
||||||
}
|
}
|
||||||
|
KeyboardFocusTarget::LockSurface(lock) => {
|
||||||
|
KeyboardTarget::enter(lock.wl_surface(), seat, data, keys, serial);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -52,6 +56,9 @@ impl KeyboardTarget<State> for KeyboardFocusTarget {
|
||||||
KeyboardFocusTarget::LayerSurface(surf) => {
|
KeyboardFocusTarget::LayerSurface(surf) => {
|
||||||
KeyboardTarget::leave(surf.wl_surface(), seat, data, serial)
|
KeyboardTarget::leave(surf.wl_surface(), seat, data, serial)
|
||||||
}
|
}
|
||||||
|
KeyboardFocusTarget::LockSurface(lock) => {
|
||||||
|
KeyboardTarget::leave(lock.wl_surface(), seat, data, serial);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -60,7 +67,7 @@ impl KeyboardTarget<State> for KeyboardFocusTarget {
|
||||||
seat: &Seat<State>,
|
seat: &Seat<State>,
|
||||||
data: &mut State,
|
data: &mut State,
|
||||||
key: KeysymHandle<'_>,
|
key: KeysymHandle<'_>,
|
||||||
state: smithay::backend::input::KeyState,
|
state: KeyState,
|
||||||
serial: Serial,
|
serial: Serial,
|
||||||
time: u32,
|
time: u32,
|
||||||
) {
|
) {
|
||||||
|
@ -74,6 +81,9 @@ impl KeyboardTarget<State> for KeyboardFocusTarget {
|
||||||
KeyboardFocusTarget::LayerSurface(surf) => {
|
KeyboardFocusTarget::LayerSurface(surf) => {
|
||||||
KeyboardTarget::key(surf.wl_surface(), seat, data, key, state, serial, time);
|
KeyboardTarget::key(surf.wl_surface(), seat, data, key, state, serial, time);
|
||||||
}
|
}
|
||||||
|
KeyboardFocusTarget::LockSurface(lock) => {
|
||||||
|
KeyboardTarget::key(lock.wl_surface(), seat, data, key, state, serial, time);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -81,7 +91,7 @@ impl KeyboardTarget<State> for KeyboardFocusTarget {
|
||||||
&self,
|
&self,
|
||||||
seat: &Seat<State>,
|
seat: &Seat<State>,
|
||||||
data: &mut State,
|
data: &mut State,
|
||||||
modifiers: smithay::input::keyboard::ModifiersState,
|
modifiers: ModifiersState,
|
||||||
serial: Serial,
|
serial: Serial,
|
||||||
) {
|
) {
|
||||||
match self {
|
match self {
|
||||||
|
@ -94,6 +104,9 @@ impl KeyboardTarget<State> for KeyboardFocusTarget {
|
||||||
KeyboardFocusTarget::LayerSurface(surf) => {
|
KeyboardFocusTarget::LayerSurface(surf) => {
|
||||||
KeyboardTarget::modifiers(surf.wl_surface(), seat, data, modifiers, serial);
|
KeyboardTarget::modifiers(surf.wl_surface(), seat, data, modifiers, serial);
|
||||||
}
|
}
|
||||||
|
KeyboardFocusTarget::LockSurface(lock) => {
|
||||||
|
KeyboardTarget::modifiers(lock.wl_surface(), seat, data, modifiers, serial);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -104,6 +117,7 @@ impl IsAlive for KeyboardFocusTarget {
|
||||||
KeyboardFocusTarget::Window(window) => window.alive(),
|
KeyboardFocusTarget::Window(window) => window.alive(),
|
||||||
KeyboardFocusTarget::Popup(popup) => popup.alive(),
|
KeyboardFocusTarget::Popup(popup) => popup.alive(),
|
||||||
KeyboardFocusTarget::LayerSurface(surf) => surf.alive(),
|
KeyboardFocusTarget::LayerSurface(surf) => surf.alive(),
|
||||||
|
KeyboardFocusTarget::LockSurface(lock) => lock.alive(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -114,6 +128,7 @@ impl WaylandFocus for KeyboardFocusTarget {
|
||||||
KeyboardFocusTarget::Window(window) => window.wl_surface(),
|
KeyboardFocusTarget::Window(window) => window.wl_surface(),
|
||||||
KeyboardFocusTarget::Popup(popup) => Some(popup.wl_surface().clone()),
|
KeyboardFocusTarget::Popup(popup) => Some(popup.wl_surface().clone()),
|
||||||
KeyboardFocusTarget::LayerSurface(surf) => Some(surf.wl_surface().clone()),
|
KeyboardFocusTarget::LayerSurface(surf) => Some(surf.wl_surface().clone()),
|
||||||
|
KeyboardFocusTarget::LockSurface(lock) => Some(lock.wl_surface().clone()),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -127,6 +142,9 @@ impl WaylandFocus for KeyboardFocusTarget {
|
||||||
KeyboardFocusTarget::LayerSurface(surf) => {
|
KeyboardFocusTarget::LayerSurface(surf) => {
|
||||||
surf.wl_surface().id().same_client_as(object_id)
|
surf.wl_surface().id().same_client_as(object_id)
|
||||||
}
|
}
|
||||||
|
KeyboardFocusTarget::LockSurface(lock) => {
|
||||||
|
lock.wl_surface().id().same_client_as(object_id)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -414,6 +414,9 @@ impl From<KeyboardFocusTarget> for PointerFocusTarget {
|
||||||
KeyboardFocusTarget::LayerSurface(layer) => {
|
KeyboardFocusTarget::LayerSurface(layer) => {
|
||||||
PointerFocusTarget::WlSurface(layer.wl_surface().clone())
|
PointerFocusTarget::WlSurface(layer.wl_surface().clone())
|
||||||
}
|
}
|
||||||
|
KeyboardFocusTarget::LockSurface(lock) => {
|
||||||
|
PointerFocusTarget::WlSurface(lock.wl_surface().clone())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
// SPDX-License-Identifier: GPL-3.0-or-later
|
// SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
pub mod session_lock;
|
||||||
mod xdg_shell;
|
mod xdg_shell;
|
||||||
mod xwayland;
|
mod xwayland;
|
||||||
|
|
||||||
|
@ -10,7 +11,7 @@ use smithay::{
|
||||||
delegate_compositor, delegate_data_control, delegate_data_device, delegate_fractional_scale,
|
delegate_compositor, delegate_data_control, delegate_data_device, delegate_fractional_scale,
|
||||||
delegate_layer_shell, delegate_output, delegate_pointer_constraints, delegate_presentation,
|
delegate_layer_shell, delegate_output, delegate_pointer_constraints, delegate_presentation,
|
||||||
delegate_primary_selection, delegate_relative_pointer, delegate_seat,
|
delegate_primary_selection, delegate_relative_pointer, delegate_seat,
|
||||||
delegate_security_context, delegate_shm, delegate_viewporter,
|
delegate_security_context, delegate_shm, delegate_viewporter, delegate_xwayland_shell,
|
||||||
desktop::{
|
desktop::{
|
||||||
self, find_popup_root_surface, get_popup_toplevel_coords, layer_map_for_output,
|
self, find_popup_root_surface, get_popup_toplevel_coords, layer_map_for_output,
|
||||||
utils::surface_primary_scanout_output, PopupKind, WindowSurfaceType,
|
utils::surface_primary_scanout_output, PopupKind, WindowSurfaceType,
|
||||||
|
@ -62,6 +63,7 @@ use smithay::{
|
||||||
xdg::{PopupSurface, XdgPopupSurfaceData, XdgToplevelSurfaceData},
|
xdg::{PopupSurface, XdgPopupSurfaceData, XdgToplevelSurfaceData},
|
||||||
},
|
},
|
||||||
shm::{ShmHandler, ShmState},
|
shm::{ShmHandler, ShmState},
|
||||||
|
xwayland_shell::{XWaylandShellHandler, XWaylandShellState},
|
||||||
},
|
},
|
||||||
xwayland::{X11Wm, XWaylandClientData},
|
xwayland::{X11Wm, XWaylandClientData},
|
||||||
};
|
};
|
||||||
|
@ -129,7 +131,7 @@ impl CompositorHandler for State {
|
||||||
|
|
||||||
utils::on_commit_buffer_handler::<State>(surface);
|
utils::on_commit_buffer_handler::<State>(surface);
|
||||||
|
|
||||||
X11Wm::commit_hook::<State>(surface);
|
X11Wm::commit_hook::<State>(self, surface);
|
||||||
|
|
||||||
self.backend.early_import(surface);
|
self.backend.early_import(surface);
|
||||||
|
|
||||||
|
@ -260,6 +262,21 @@ impl CompositorHandler for State {
|
||||||
.cloned()
|
.cloned()
|
||||||
{
|
{
|
||||||
vec![output] // surface is a layer surface
|
vec![output] // surface is a layer surface
|
||||||
|
} else if let Some(output) = self
|
||||||
|
.pinnacle
|
||||||
|
.space
|
||||||
|
.outputs()
|
||||||
|
.find(|op| {
|
||||||
|
op.with_state(|state| {
|
||||||
|
state
|
||||||
|
.lock_surface
|
||||||
|
.as_ref()
|
||||||
|
.is_some_and(|lock| lock.wl_surface() == surface)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.cloned()
|
||||||
|
{
|
||||||
|
vec![output]
|
||||||
} else {
|
} else {
|
||||||
return;
|
return;
|
||||||
};
|
};
|
||||||
|
@ -830,6 +847,13 @@ impl ForeignToplevelHandler for State {
|
||||||
}
|
}
|
||||||
delegate_foreign_toplevel!(State);
|
delegate_foreign_toplevel!(State);
|
||||||
|
|
||||||
|
impl XWaylandShellHandler for State {
|
||||||
|
fn xwayland_shell_state(&mut self) -> &mut XWaylandShellState {
|
||||||
|
&mut self.pinnacle.xwayland_shell_state
|
||||||
|
}
|
||||||
|
}
|
||||||
|
delegate_xwayland_shell!(State);
|
||||||
|
|
||||||
impl Pinnacle {
|
impl Pinnacle {
|
||||||
fn position_popup(&self, popup: &PopupSurface) {
|
fn position_popup(&self, popup: &PopupSurface) {
|
||||||
trace!("State::position_popup");
|
trace!("State::position_popup");
|
||||||
|
|
129
src/handlers/session_lock.rs
Normal file
129
src/handlers/session_lock.rs
Normal file
|
@ -0,0 +1,129 @@
|
||||||
|
use smithay::{
|
||||||
|
delegate_session_lock,
|
||||||
|
output::Output,
|
||||||
|
reexports::wayland_server::protocol::wl_output::WlOutput,
|
||||||
|
utils::SERIAL_COUNTER,
|
||||||
|
wayland::session_lock::{
|
||||||
|
LockSurface, SessionLockHandler, SessionLockManagerState, SessionLocker,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
use tracing::{debug, warn};
|
||||||
|
|
||||||
|
use crate::{
|
||||||
|
focus::keyboard::KeyboardFocusTarget,
|
||||||
|
output::BlankingState,
|
||||||
|
state::{State, WithState},
|
||||||
|
};
|
||||||
|
|
||||||
|
/// State of a session lock.
|
||||||
|
#[derive(Default, Debug)]
|
||||||
|
pub enum LockState {
|
||||||
|
/// There is no session lock.
|
||||||
|
#[default]
|
||||||
|
Unlocked,
|
||||||
|
/// A session lock request came in and we are in the process of blanking outputs.
|
||||||
|
Locking(SessionLocker),
|
||||||
|
/// The session is locked.
|
||||||
|
Locked,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl LockState {
|
||||||
|
/// Returns `true` if the lock state is [`Locking`].
|
||||||
|
///
|
||||||
|
/// [`Locking`]: LockState::Locking
|
||||||
|
#[must_use]
|
||||||
|
pub fn is_locking(&self) -> bool {
|
||||||
|
matches!(self, Self::Locking(..))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns `true` if the lock state is [`Unlocked`].
|
||||||
|
///
|
||||||
|
/// [`Unlocked`]: LockState::Unlocked
|
||||||
|
#[must_use]
|
||||||
|
pub fn is_unlocked(&self) -> bool {
|
||||||
|
matches!(self, Self::Unlocked)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns `true` if the lock state is [`Locked`].
|
||||||
|
///
|
||||||
|
/// [`Locked`]: LockState::Locked
|
||||||
|
#[must_use]
|
||||||
|
pub fn is_locked(&self) -> bool {
|
||||||
|
matches!(self, Self::Locked)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SessionLockHandler for State {
|
||||||
|
fn lock_state(&mut self) -> &mut SessionLockManagerState {
|
||||||
|
&mut self.pinnacle.session_lock_manager_state
|
||||||
|
}
|
||||||
|
|
||||||
|
fn lock(&mut self, confirmation: SessionLocker) {
|
||||||
|
debug!("Received session lock request");
|
||||||
|
self.pinnacle.lock_state = LockState::Locking(confirmation);
|
||||||
|
self.pinnacle.schedule(
|
||||||
|
|state| {
|
||||||
|
let all_outputs_blanked = state.pinnacle.space.outputs().all(|op| {
|
||||||
|
op.with_state(|st| matches!(st.blanking_state, BlankingState::Blanked))
|
||||||
|
});
|
||||||
|
!state.pinnacle.lock_state.is_locking() || all_outputs_blanked
|
||||||
|
},
|
||||||
|
|state| match std::mem::take(&mut state.pinnacle.lock_state) {
|
||||||
|
LockState::Unlocked => (),
|
||||||
|
LockState::Locking(locker) => {
|
||||||
|
debug!("Locking session");
|
||||||
|
locker.lock();
|
||||||
|
state.pinnacle.lock_state = LockState::Locked;
|
||||||
|
for output in state.pinnacle.space.outputs().cloned().collect::<Vec<_>>() {
|
||||||
|
state.schedule_render(&output);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
LockState::Locked => state.pinnacle.lock_state = LockState::Locked,
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn unlock(&mut self) {
|
||||||
|
debug!("Session lock unlocked");
|
||||||
|
for output in self.pinnacle.space.outputs() {
|
||||||
|
output.with_state_mut(|state| {
|
||||||
|
state.lock_surface.take();
|
||||||
|
state.blanking_state = BlankingState::NotBlanked;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
self.pinnacle.lock_state = LockState::Unlocked;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn new_surface(&mut self, surface: LockSurface, output: WlOutput) {
|
||||||
|
let Some(output) = Output::from_resource(&output) else {
|
||||||
|
warn!(
|
||||||
|
"Session lock surface received but output doesn't exist for wl_output {output:?}"
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
debug!("Session lock surface received for output {}", output.name());
|
||||||
|
|
||||||
|
let Some(geo) = self.pinnacle.space.output_geometry(&output) else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
surface.with_pending_state(|state| {
|
||||||
|
state.size = Some((geo.size.w as u32, geo.size.h as u32).into())
|
||||||
|
});
|
||||||
|
surface.send_configure();
|
||||||
|
|
||||||
|
if let Some(keyboard) = self.pinnacle.seat.get_keyboard() {
|
||||||
|
keyboard.set_focus(
|
||||||
|
self,
|
||||||
|
Some(KeyboardFocusTarget::LockSurface(surface.clone())),
|
||||||
|
SERIAL_COUNTER.next_serial(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
output.with_state_mut(|state| state.lock_surface.replace(surface));
|
||||||
|
|
||||||
|
self.schedule_render(&output);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
delegate_session_lock!(State);
|
|
@ -531,8 +531,6 @@ impl Pinnacle {
|
||||||
|_| (),
|
|_| (),
|
||||||
)?;
|
)?;
|
||||||
|
|
||||||
let display_handle = self.display_handle.clone();
|
|
||||||
|
|
||||||
self.loop_handle
|
self.loop_handle
|
||||||
.insert_source(xwayland, move |event, _, state| match event {
|
.insert_source(xwayland, move |event, _, state| match event {
|
||||||
XWaylandEvent::Ready {
|
XWaylandEvent::Ready {
|
||||||
|
@ -541,7 +539,6 @@ impl Pinnacle {
|
||||||
} => {
|
} => {
|
||||||
let mut wm = X11Wm::start_wm(
|
let mut wm = X11Wm::start_wm(
|
||||||
state.pinnacle.loop_handle.clone(),
|
state.pinnacle.loop_handle.clone(),
|
||||||
display_handle.clone(),
|
|
||||||
x11_socket,
|
x11_socket,
|
||||||
client.clone(),
|
client.clone(),
|
||||||
)
|
)
|
||||||
|
|
53
src/input.rs
53
src/input.rs
|
@ -155,10 +155,15 @@ impl InputState {
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
enum KeyAction {
|
enum KeyAction {
|
||||||
|
/// Call a config callback.
|
||||||
CallCallback(UnboundedSender<Result<SetKeybindResponse, tonic::Status>>),
|
CallCallback(UnboundedSender<Result<SetKeybindResponse, tonic::Status>>),
|
||||||
|
/// Quit the compositor.
|
||||||
Quit,
|
Quit,
|
||||||
|
/// Switch ttys.
|
||||||
SwitchVt(i32),
|
SwitchVt(i32),
|
||||||
|
/// Reload the config.
|
||||||
ReloadConfig,
|
ReloadConfig,
|
||||||
|
/// Prevent the key from being sent to clients.
|
||||||
Suppress,
|
Suppress,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -185,6 +190,17 @@ impl Pinnacle {
|
||||||
.output_geometry(output)
|
.output_geometry(output)
|
||||||
.expect("called output_geometry on unmapped output");
|
.expect("called output_geometry on unmapped output");
|
||||||
|
|
||||||
|
if !self.lock_state.is_unlocked() {
|
||||||
|
return output
|
||||||
|
.with_state(|state| state.lock_surface.clone())
|
||||||
|
.map(|lock_surface| {
|
||||||
|
(
|
||||||
|
PointerFocusTarget::WlSurface(lock_surface.wl_surface().clone()),
|
||||||
|
output_geo.loc,
|
||||||
|
)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
let mut fullscreen_and_up_split_at = 0;
|
let mut fullscreen_and_up_split_at = 0;
|
||||||
|
|
||||||
for (i, win) in self
|
for (i, win) in self
|
||||||
|
@ -412,6 +428,8 @@ impl State {
|
||||||
device.led_update(leds);
|
device.led_update(leds);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if self.pinnacle.lock_state.is_unlocked() {
|
||||||
|
// Handle exclusive layers
|
||||||
for layer in self.pinnacle.layer_shell_state.layer_surfaces().rev() {
|
for layer in self.pinnacle.layer_shell_state.layer_surfaces().rev() {
|
||||||
let data = compositor::with_states(layer.wl_surface(), |states| {
|
let data = compositor::with_states(layer.wl_surface(), |states| {
|
||||||
*states.cached_state.current::<LayerSurfaceCachedState>()
|
*states.cached_state.current::<LayerSurfaceCachedState>()
|
||||||
|
@ -476,6 +494,25 @@ impl State {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
// We don't want anything but lock surfaces getting keyboard input when locked
|
||||||
|
let lock_surface = self
|
||||||
|
.pinnacle
|
||||||
|
.space
|
||||||
|
.outputs()
|
||||||
|
.find_map(|op| op.with_state(|state| state.lock_surface.clone()));
|
||||||
|
|
||||||
|
if !matches!(
|
||||||
|
keyboard.current_focus(),
|
||||||
|
Some(KeyboardFocusTarget::LockSurface(_))
|
||||||
|
) {
|
||||||
|
keyboard.set_focus(
|
||||||
|
self,
|
||||||
|
lock_surface.map(KeyboardFocusTarget::LockSurface),
|
||||||
|
serial,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let action = keyboard.input(
|
let action = keyboard.input(
|
||||||
self,
|
self,
|
||||||
|
@ -515,15 +552,23 @@ impl State {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
{
|
{
|
||||||
return FilterResult::Intercept(KeyAction::CallCallback(sender.clone()));
|
if state.pinnacle.lock_state.is_unlocked() {
|
||||||
|
return FilterResult::Intercept(KeyAction::CallCallback(
|
||||||
|
sender.clone(),
|
||||||
|
));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if kill_keybind == Some((mod_mask, mod_sym)) {
|
if kill_keybind == Some((mod_mask, mod_sym)) {
|
||||||
return FilterResult::Intercept(KeyAction::Quit);
|
return FilterResult::Intercept(KeyAction::Quit);
|
||||||
} else if reload_keybind == Some((mod_mask, mod_sym)) {
|
}
|
||||||
|
|
||||||
|
if reload_keybind == Some((mod_mask, mod_sym)) {
|
||||||
return FilterResult::Intercept(KeyAction::ReloadConfig);
|
return FilterResult::Intercept(KeyAction::ReloadConfig);
|
||||||
} else if let mut vt @ keysyms::KEY_XF86Switch_VT_1
|
}
|
||||||
..=keysyms::KEY_XF86Switch_VT_12 = keysym.modified_sym().raw()
|
|
||||||
|
if let mut vt @ keysyms::KEY_XF86Switch_VT_1..=keysyms::KEY_XF86Switch_VT_12 =
|
||||||
|
keysym.modified_sym().raw()
|
||||||
{
|
{
|
||||||
vt = vt - keysyms::KEY_XF86Switch_VT_1 + 1;
|
vt = vt - keysyms::KEY_XF86Switch_VT_1 + 1;
|
||||||
tracing::info!("Switching to vt {vt}");
|
tracing::info!("Switching to vt {vt}");
|
||||||
|
|
|
@ -7,6 +7,7 @@ use smithay::{
|
||||||
desktop::layer_map_for_output,
|
desktop::layer_map_for_output,
|
||||||
output::{Mode, Output, Scale},
|
output::{Mode, Output, Scale},
|
||||||
utils::{Logical, Point, Transform},
|
utils::{Logical, Point, Transform},
|
||||||
|
wayland::session_lock::LockSurface,
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
@ -34,6 +35,18 @@ impl OutputName {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// State of an output's blanking status for session lock.
|
||||||
|
#[derive(Debug, Default, Copy, Clone)]
|
||||||
|
pub enum BlankingState {
|
||||||
|
/// The output is not blanked and is displaying normal content.
|
||||||
|
#[default]
|
||||||
|
NotBlanked,
|
||||||
|
/// A blank frame has been queued up.
|
||||||
|
Blanking,
|
||||||
|
/// A blank frame has been displayed.
|
||||||
|
Blanked,
|
||||||
|
}
|
||||||
|
|
||||||
/// The state of an output
|
/// The state of an output
|
||||||
#[derive(Default, Debug)]
|
#[derive(Default, Debug)]
|
||||||
pub struct OutputState {
|
pub struct OutputState {
|
||||||
|
@ -42,6 +55,8 @@ pub struct OutputState {
|
||||||
pub screencopy: Option<Screencopy>,
|
pub screencopy: Option<Screencopy>,
|
||||||
pub serial: Option<NonZeroU32>,
|
pub serial: Option<NonZeroU32>,
|
||||||
pub modes: Vec<Mode>,
|
pub modes: Vec<Mode>,
|
||||||
|
pub lock_surface: Option<LockSurface>,
|
||||||
|
pub blanking_state: BlankingState,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl WithState for Output {
|
impl WithState for Output {
|
||||||
|
@ -113,5 +128,16 @@ impl Pinnacle {
|
||||||
output.set_preferred(mode);
|
output.set_preferred(mode);
|
||||||
output.with_state_mut(|state| state.modes.push(mode));
|
output.with_state_mut(|state| state.modes.push(mode));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let Some(lock_surface) = output.with_state(|state| state.lock_surface.clone()) {
|
||||||
|
lock_surface.with_pending_state(|state| {
|
||||||
|
let Some(new_geo) = self.space.output_geometry(output) else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
state.size = Some((new_geo.size.w as u32, new_geo.size.h as u32).into());
|
||||||
|
});
|
||||||
|
|
||||||
|
lock_surface.send_configure();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,6 +5,7 @@ use std::{ops::Deref, sync::Mutex};
|
||||||
use smithay::{
|
use smithay::{
|
||||||
backend::renderer::{
|
backend::renderer::{
|
||||||
element::{
|
element::{
|
||||||
|
solid::SolidColorRenderElement,
|
||||||
surface::WaylandSurfaceRenderElement,
|
surface::WaylandSurfaceRenderElement,
|
||||||
utils::{CropRenderElement, RelocateRenderElement, RescaleRenderElement},
|
utils::{CropRenderElement, RelocateRenderElement, RescaleRenderElement},
|
||||||
AsRenderElements, RenderElementStates, Wrap,
|
AsRenderElements, RenderElementStates, Wrap,
|
||||||
|
@ -51,6 +52,7 @@ render_elements! {
|
||||||
Surface = WaylandSurfaceRenderElement<R>,
|
Surface = WaylandSurfaceRenderElement<R>,
|
||||||
Pointer = PointerRenderElement<R>,
|
Pointer = PointerRenderElement<R>,
|
||||||
Transform = TransformRenderElement<R, E>,
|
Transform = TransformRenderElement<R, E>,
|
||||||
|
Color = SolidColorRenderElement,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<R> AsRenderElements<R> for WindowElement
|
impl<R> AsRenderElements<R> for WindowElement
|
||||||
|
|
14
src/state.rs
14
src/state.rs
|
@ -7,6 +7,7 @@ use crate::{
|
||||||
config::Config,
|
config::Config,
|
||||||
focus::OutputFocusStack,
|
focus::OutputFocusStack,
|
||||||
grab::resize_grab::ResizeSurfaceState,
|
grab::resize_grab::ResizeSurfaceState,
|
||||||
|
handlers::session_lock::LockState,
|
||||||
layout::LayoutState,
|
layout::LayoutState,
|
||||||
protocol::{
|
protocol::{
|
||||||
foreign_toplevel::{self, ForeignToplevelManagerState},
|
foreign_toplevel::{self, ForeignToplevelManagerState},
|
||||||
|
@ -41,10 +42,12 @@ use smithay::{
|
||||||
data_device::DataDeviceState, primary_selection::PrimarySelectionState,
|
data_device::DataDeviceState, primary_selection::PrimarySelectionState,
|
||||||
wlr_data_control::DataControlState,
|
wlr_data_control::DataControlState,
|
||||||
},
|
},
|
||||||
|
session_lock::SessionLockManagerState,
|
||||||
shell::{wlr_layer::WlrLayerShellState, xdg::XdgShellState},
|
shell::{wlr_layer::WlrLayerShellState, xdg::XdgShellState},
|
||||||
shm::ShmState,
|
shm::ShmState,
|
||||||
socket::ListeningSocketSource,
|
socket::ListeningSocketSource,
|
||||||
viewporter::ViewporterState,
|
viewporter::ViewporterState,
|
||||||
|
xwayland_shell::XWaylandShellState,
|
||||||
},
|
},
|
||||||
xwayland::{X11Wm, XWaylandClientData},
|
xwayland::{X11Wm, XWaylandClientData},
|
||||||
};
|
};
|
||||||
|
@ -94,6 +97,10 @@ pub struct Pinnacle {
|
||||||
pub relative_pointer_manager_state: RelativePointerManagerState,
|
pub relative_pointer_manager_state: RelativePointerManagerState,
|
||||||
pub pointer_constraints_state: PointerConstraintsState,
|
pub pointer_constraints_state: PointerConstraintsState,
|
||||||
pub foreign_toplevel_manager_state: ForeignToplevelManagerState,
|
pub foreign_toplevel_manager_state: ForeignToplevelManagerState,
|
||||||
|
pub session_lock_manager_state: SessionLockManagerState,
|
||||||
|
pub xwayland_shell_state: XWaylandShellState,
|
||||||
|
|
||||||
|
pub lock_state: LockState,
|
||||||
|
|
||||||
/// The state of key and mousebinds along with libinput settings
|
/// The state of key and mousebinds along with libinput settings
|
||||||
pub input_state: InputState,
|
pub input_state: InputState,
|
||||||
|
@ -262,6 +269,13 @@ impl Pinnacle {
|
||||||
&display_handle,
|
&display_handle,
|
||||||
filter_restricted_client,
|
filter_restricted_client,
|
||||||
),
|
),
|
||||||
|
session_lock_manager_state: SessionLockManagerState::new::<State, _>(
|
||||||
|
&display_handle,
|
||||||
|
filter_restricted_client,
|
||||||
|
),
|
||||||
|
xwayland_shell_state: XWaylandShellState::new::<State>(&display_handle),
|
||||||
|
|
||||||
|
lock_state: LockState::default(),
|
||||||
|
|
||||||
input_state: InputState::new(),
|
input_state: InputState::new(),
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue