Merge pull request #103 from pinnacle-comp/even_better_rendering

Improve rendering
This commit is contained in:
Ottatop 2023-10-17 23:05:36 -05:00 committed by GitHub
commit 5077369732
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
10 changed files with 472 additions and 509 deletions

View file

@ -1,16 +1,15 @@
// SPDX-License-Identifier: GPL-3.0-or-later // SPDX-License-Identifier: GPL-3.0-or-later
mod utils;
use std::{ use std::{
collections::{HashMap, HashSet}, collections::{HashMap, HashSet},
ffi::OsString, ffi::OsString,
os::fd::FromRawFd, os::fd::FromRawFd,
path::Path, path::Path,
time::Duration, time::{Duration, Instant},
}; };
use anyhow::Context; use anyhow::Context;
use calloop::Idle;
use smithay::{ use smithay::{
backend::{ backend::{
allocator::{ allocator::{
@ -28,10 +27,7 @@ use smithay::{
libinput::{LibinputInputBackend, LibinputSessionInterface}, libinput::{LibinputInputBackend, LibinputSessionInterface},
renderer::{ renderer::{
damage::{self}, damage::{self},
element::{ element::{texture::TextureBuffer, RenderElement, RenderElementStates},
surface::WaylandSurfaceRenderElement, texture::TextureBuffer, RenderElement,
RenderElementStates,
},
gles::{GlesRenderer, GlesTexture}, gles::{GlesRenderer, GlesTexture},
multigpu::{gbm::GbmGlesBackend, GpuManager, MultiRenderer, MultiTexture}, multigpu::{gbm::GbmGlesBackend, GpuManager, MultiRenderer, MultiTexture},
Bind, ExportMem, ImportDma, ImportEgl, ImportMemWl, Offscreen, Renderer, Bind, ExportMem, ImportDma, ImportEgl, ImportMemWl, Offscreen, Renderer,
@ -53,10 +49,7 @@ use smithay::{
output::{Output, PhysicalProperties, Subpixel}, output::{Output, PhysicalProperties, Subpixel},
reexports::{ reexports::{
ash::vk::ExtPhysicalDeviceDrmFn, ash::vk::ExtPhysicalDeviceDrmFn,
calloop::{ calloop::{EventLoop, LoopHandle, RegistrationToken},
timer::{TimeoutAction, Timer},
EventLoop, LoopHandle, RegistrationToken,
},
drm::{ drm::{
self, self,
control::{connector, crtc, ModeTypeFlags}, control::{connector, crtc, ModeTypeFlags},
@ -88,7 +81,7 @@ use crate::{
ConnectorSavedState, ConnectorSavedState,
}, },
output::OutputName, output::OutputName,
render::{pointer::PointerElement, take_presentation_feedback, CustomRenderElements}, render::{pointer::PointerElement, take_presentation_feedback},
state::{CalloopData, State, SurfaceDmabufFeedback, WithState}, state::{CalloopData, State, SurfaceDmabufFeedback, WithState},
window::WindowElement, window::WindowElement,
}; };
@ -128,6 +121,44 @@ pub struct Udev {
pointer_images: Vec<(xcursor::parser::Image, TextureBuffer<MultiTexture>)>, pointer_images: Vec<(xcursor::parser::Image, TextureBuffer<MultiTexture>)>,
pointer_element: PointerElement<MultiTexture>, pointer_element: PointerElement<MultiTexture>,
pointer_image: crate::cursor::Cursor, pointer_image: crate::cursor::Cursor,
last_vblank_time: Instant,
}
impl Backend {
fn udev(&self) -> &Udev {
let Backend::Udev(udev) = self else { unreachable!() };
udev
}
fn udev_mut(&mut self) -> &mut Udev {
let Backend::Udev(udev) = self else { unreachable!() };
udev
}
}
impl Udev {
pub fn schedule_render(&mut self, loop_handle: &LoopHandle<CalloopData>, output: &Output) {
let Some(surface) = render_surface_for_output(output, &mut self.backends) else {
return;
};
// tracing::debug!(state = ?surface.render_state, "scheduling render");
match &surface.render_state {
RenderState::Idle => {
let output = output.clone();
let token = loop_handle.insert_idle(move |data| {
data.state.render_surface(&output);
});
surface.render_state = RenderState::Scheduled(token);
}
RenderState::Scheduled(_) => (),
RenderState::WaitingForVblank { dirty: _ } => {
surface.render_state = RenderState::WaitingForVblank { dirty: true }
}
}
}
} }
impl BackendData for Udev { impl BackendData for Udev {
@ -193,6 +224,8 @@ pub fn run_udev() -> anyhow::Result<()> {
pointer_image: crate::cursor::Cursor::load(), pointer_image: crate::cursor::Cursor::load(),
pointer_images: Vec::new(), pointer_images: Vec::new(),
pointer_element: PointerElement::default(), pointer_element: PointerElement::default(),
last_vblank_time: Instant::now(),
}; };
let display_handle = display.handle(); let display_handle = display.handle();
@ -217,9 +250,7 @@ pub fn run_udev() -> anyhow::Result<()> {
} }
} }
let Backend::Udev(udev) = &mut state.backend else { let udev = state.backend.udev_mut();
unreachable!()
};
event_loop event_loop
.handle() .handle()
@ -272,9 +303,8 @@ pub fn run_udev() -> anyhow::Result<()> {
event_loop event_loop
.handle() .handle()
.insert_source(notifier, move |event, _, data| { .insert_source(notifier, move |event, _, data| {
let Backend::Udev(udev) = &mut data.state.backend else { let udev = data.state.backend.udev_mut();
unreachable!()
};
match event { match event {
session::Event::PauseSession => { session::Event::PauseSession => {
libinput_context.suspend(); libinput_context.suspend();
@ -290,11 +320,7 @@ pub fn run_udev() -> anyhow::Result<()> {
if let Err(err) = libinput_context.resume() { if let Err(err) = libinput_context.resume() {
tracing::error!("Failed to resume libinput context: {:?}", err); tracing::error!("Failed to resume libinput context: {:?}", err);
} }
for (node, backend) in udev for backend in udev.backends.values_mut() {
.backends
.iter_mut()
.map(|(handle, backend)| (*handle, backend))
{
backend.drm.activate(); backend.drm.activate();
for surface in backend.surfaces.values_mut() { for surface in backend.surfaces.values_mut() {
if let Err(err) = surface.compositor.surface().reset_state() { if let Err(err) = surface.compositor.surface().reset_state() {
@ -306,9 +332,13 @@ pub fn run_udev() -> anyhow::Result<()> {
// otherwise // otherwise
surface.compositor.reset_buffers(); surface.compositor.reset_buffers();
} }
data.state }
.loop_handle
.insert_idle(move |data| data.state.render(node, None)); for output in data.state.space.outputs().cloned().collect::<Vec<_>>() {
data.state.schedule_render(&output);
// data.state
// .loop_handle
// .insert_idle(move |data| data.state.render_surface(&output));
} }
} }
} }
@ -429,6 +459,8 @@ pub fn run_udev() -> anyhow::Result<()> {
data.display_handle data.display_handle
.flush_clients() .flush_clients()
.expect("failed to flush_clients"); .expect("failed to flush_clients");
data.state.focus_state.fix_up_focus(&mut data.state.space);
}, },
)?; )?;
@ -436,7 +468,7 @@ pub fn run_udev() -> anyhow::Result<()> {
} }
struct UdevBackendData { struct UdevBackendData {
surfaces: HashMap<crtc::Handle, SurfaceData>, surfaces: HashMap<crtc::Handle, RenderSurface>,
gbm: GbmDevice<DrmDeviceFd>, gbm: GbmDevice<DrmDeviceFd>,
drm: DrmDevice, drm: DrmDevice,
drm_scanner: DrmScanner, drm_scanner: DrmScanner,
@ -525,17 +557,47 @@ struct DrmSurfaceDmabufFeedback {
scanout_feedback: DmabufFeedback, scanout_feedback: DmabufFeedback,
} }
/// Data associated with a crtc. /// The state of a [`RenderSurface`].
struct SurfaceData { #[derive(Debug)]
global: Option<GlobalId>, enum RenderState {
display_handle: DisplayHandle, /// No render is scheduled.
device_id: DrmNode, Idle,
render_node: DrmNode, /// A render has been queued.
compositor: GbmDrmCompositor, Scheduled(
dmabuf_feedback: Option<DrmSurfaceDmabufFeedback>, /// The idle token from a render being scheduled.
/// This is used to cancel renders if, for example,
/// the output being rendered is removed.
Idle<'static>,
),
/// A frame was rendered and scheduled and we are waiting for vblank.
WaitingForVblank {
/// A render was scheduled while waiting for vblank.
/// In this case, another render will be scheduled once vblank happens.
dirty: bool,
},
} }
impl Drop for SurfaceData { /// Render surface for an output.
struct RenderSurface {
/// The output global id.
global: Option<GlobalId>,
/// A display handle used to remove the global on drop.
display_handle: DisplayHandle,
/// The node from `connector_connected`.
device_id: DrmNode,
/// The node rendering to the screen? idk
///
/// If this is equal to the primary gpu node then it does the rendering operations.
/// If it's not it is the node the composited buffer ends up on.
render_node: DrmNode,
/// The thing rendering elements and queueing frames.
compositor: GbmDrmCompositor,
dmabuf_feedback: Option<DrmSurfaceDmabufFeedback>,
render_state: RenderState,
}
impl Drop for RenderSurface {
// Stop advertising this output to clients on drop.
fn drop(&mut self) { fn drop(&mut self) {
if let Some(global) = self.global.take() { if let Some(global) = self.global.take() {
self.display_handle.remove_global::<State>(global); self.display_handle.remove_global::<State>(global);
@ -550,13 +612,15 @@ type GbmDrmCompositor = DrmCompositor<
DrmDeviceFd, DrmDeviceFd,
>; >;
/// The result of a frame render from `GbmDrmCompositor::render_frame`. /// The result of a frame render from `render_frame`.
struct SurfaceCompositorRenderResult { struct SurfaceCompositorRenderResult {
rendered: bool, rendered: bool,
states: RenderElementStates, states: RenderElementStates,
} }
/// Render a frame with the given elements. /// Render a frame with the given elements.
///
/// This frame needs to be queued for scanout afterwards.
fn render_frame<R, E, Target>( fn render_frame<R, E, Target>(
compositor: &mut GbmDrmCompositor, compositor: &mut GbmDrmCompositor,
renderer: &mut R, renderer: &mut R,
@ -569,6 +633,8 @@ where
<R as Renderer>::Error: Into<SwapBuffersError>, <R as Renderer>::Error: Into<SwapBuffersError>,
E: RenderElement<R>, E: RenderElement<R>,
{ {
use smithay::backend::drm::compositor::RenderFrameError;
compositor compositor
.render_frame(renderer, elements, clear_color) .render_frame(renderer, elements, clear_color)
.map(|render_frame_result| { .map(|render_frame_result| {
@ -581,10 +647,8 @@ where
} }
}) })
.map_err(|err| match err { .map_err(|err| match err {
smithay::backend::drm::compositor::RenderFrameError::PrepareFrame(err) => err.into(), RenderFrameError::PrepareFrame(err) => err.into(),
smithay::backend::drm::compositor::RenderFrameError::RenderFrame( RenderFrameError::RenderFrame(damage::Error::Rendering(err)) => err.into(),
damage::Error::Rendering(err),
) => err.into(),
_ => unreachable!(), _ => unreachable!(),
}) })
} }
@ -592,9 +656,7 @@ where
impl State { impl State {
/// A GPU was plugged in. /// A GPU was plugged in.
fn device_added(&mut self, node: DrmNode, path: &Path) -> Result<(), DeviceAddError> { fn device_added(&mut self, node: DrmNode, path: &Path) -> Result<(), DeviceAddError> {
let Backend::Udev(udev) = &mut self.backend else { let udev = self.backend.udev_mut();
unreachable!()
};
// Try to open the device // Try to open the device
let fd = udev let fd = udev
@ -613,17 +675,22 @@ impl State {
let registration_token = self let registration_token = self
.loop_handle .loop_handle
.insert_source( .insert_source(notifier, move |event, metadata, data| match event {
notifier,
move |event, metadata, data: &mut CalloopData| match event {
DrmEvent::VBlank(crtc) => { DrmEvent::VBlank(crtc) => {
data.state.frame_finish(node, crtc, metadata); // { TODO:
// let udev = data.state.backend.udev_mut();
// let then = udev.last_vblank_time;
// let now = Instant::now();
// let diff = now.duration_since(then);
// // tracing::debug!(time = diff.as_secs_f64(), "Time since last vblank");
// udev.last_vblank_time = now;
// }
data.state.on_vblank(node, crtc, metadata);
} }
DrmEvent::Error(error) => { DrmEvent::Error(error) => {
tracing::error!("{:?}", error); tracing::error!("{:?}", error);
} }
}, })
)
.expect("failed to insert drm notifier into event loop"); .expect("failed to insert drm notifier into event loop");
let render_node = EGLDevice::device_for_display( let render_node = EGLDevice::device_for_display(
@ -663,9 +730,7 @@ impl State {
connector: connector::Info, connector: connector::Info,
crtc: crtc::Handle, crtc: crtc::Handle,
) { ) {
let Backend::Udev(udev) = &mut self.backend else { let udev = self.backend.udev_mut();
unreachable!()
};
let device = if let Some(device) = udev.backends.get_mut(&node) { let device = if let Some(device) = udev.backends.get_mut(&node) {
device device
@ -845,19 +910,18 @@ impl State {
&compositor, &compositor,
); );
let surface = SurfaceData { let surface = RenderSurface {
display_handle: udev.display_handle.clone(), display_handle: udev.display_handle.clone(),
device_id: node, device_id: node,
render_node: device.render_node, render_node: device.render_node,
global: Some(global), global: Some(global),
compositor, compositor,
dmabuf_feedback, dmabuf_feedback,
render_state: RenderState::Idle,
}; };
device.surfaces.insert(crtc, surface); device.surfaces.insert(crtc, surface);
self.schedule_initial_render(node, crtc, self.loop_handle.clone());
// If there is saved connector state, the connector was previously plugged in. // If there is saved connector state, the connector was previously plugged in.
// In this case, restore its tags and location. // In this case, restore its tags and location.
// TODO: instead of checking the connector, check the monitor's edid info instead // TODO: instead of checking the connector, check the monitor's edid info instead
@ -911,9 +975,7 @@ impl State {
) { ) {
tracing::debug!(?crtc, "connector_disconnected"); tracing::debug!(?crtc, "connector_disconnected");
let Backend::Udev(udev) = &mut self.backend else { let udev = self.backend.udev_mut();
unreachable!()
};
let device = if let Some(device) = udev.backends.get_mut(&node) { let device = if let Some(device) = udev.backends.get_mut(&node) {
device device
@ -948,9 +1010,7 @@ impl State {
} }
fn device_changed(&mut self, node: DrmNode) { fn device_changed(&mut self, node: DrmNode) {
let Backend::Udev(udev) = &mut self.backend else { let udev = self.backend.udev_mut();
unreachable!()
};
let device = if let Some(device) = udev.backends.get_mut(&node) { let device = if let Some(device) = udev.backends.get_mut(&node) {
device device
@ -980,11 +1040,9 @@ impl State {
/// A GPU was unplugged. /// A GPU was unplugged.
fn device_removed(&mut self, node: DrmNode) { fn device_removed(&mut self, node: DrmNode) {
let crtcs = { let crtcs = {
let Backend::Udev(udev) = &mut self.backend else { let udev = self.backend.udev();
unreachable!()
};
let Some(device) = udev.backends.get_mut(&node) else { let Some(device) = udev.backends.get(&node) else {
return; return;
}; };
@ -1001,9 +1059,7 @@ impl State {
tracing::debug!("Surfaces dropped"); tracing::debug!("Surfaces dropped");
let Backend::Udev(udev) = &mut self.backend else { let udev = self.backend.udev_mut();
unreachable!()
};
// drop the backends on this side // drop the backends on this side
if let Some(backend_data) = udev.backends.remove(&node) { if let Some(backend_data) = udev.backends.remove(&node) {
@ -1018,15 +1074,13 @@ impl State {
} }
/// Mark [`OutputPresentationFeedback`]s as presented and schedule a new render on idle. /// Mark [`OutputPresentationFeedback`]s as presented and schedule a new render on idle.
fn frame_finish( fn on_vblank(
&mut self, &mut self,
dev_id: DrmNode, dev_id: DrmNode,
crtc: crtc::Handle, crtc: crtc::Handle,
metadata: &mut Option<DrmEventMetadata>, metadata: &mut Option<DrmEventMetadata>,
) { ) {
let Backend::Udev(udev) = &mut self.backend else { let udev = self.backend.udev_mut();
unreachable!()
};
let Some(surface) = udev let Some(surface) = udev
.backends .backends
@ -1047,7 +1101,7 @@ impl State {
return; return;
}; };
let schedule_render = match surface match surface
.compositor .compositor
.frame_submitted() .frame_submitted()
.map_err(SwapBuffersError::from) .map_err(SwapBuffersError::from)
@ -1084,83 +1138,43 @@ impl State {
flags, flags,
); );
} }
true
} }
Err(err) => { Err(err) => {
tracing::warn!("Error during rendering: {:?}", err); tracing::warn!("Error during rendering: {:?}", err);
match err { if let SwapBuffersError::ContextLost(err) = err {
SwapBuffersError::AlreadySwapped => true, panic!("Rendering loop lost: {}", err)
// If the device has been deactivated do not reschedule, this will be done
// by session resume
SwapBuffersError::TemporaryFailure(err)
if matches!(
err.downcast_ref::<DrmError>(),
Some(&DrmError::DeviceInactive)
) =>
{
false
}
SwapBuffersError::TemporaryFailure(err) => matches!(
err.downcast_ref::<DrmError>(),
Some(&DrmError::Access {
source: drm::SystemError::PermissionDenied,
..
})
),
SwapBuffersError::ContextLost(err) => panic!("Rendering loop lost: {}", err),
} }
} }
}; };
if schedule_render { let RenderState::WaitingForVblank { dirty } = surface.render_state else {
// Anvil had some stuff here about delaying a render to reduce latency, unreachable!();
// but it introduces visible hitching when scrolling, so I'm removing it here. };
//
// If latency is a problem then future me can deal with it :) surface.render_state = RenderState::Idle;
self.loop_handle.insert_idle(move |data| {
data.state.render(dev_id, Some(crtc)); if dirty {
self.schedule_render(&output);
} else {
for window in self.windows.iter() {
window.send_frame(&output, self.clock.now(), Some(Duration::ZERO), |_, _| {
Some(output.clone())
}); });
} }
} }
/// Render using the gpu on `node` to the provided `crtc`, or all available crtcs if `None`.
fn render(&mut self, node: DrmNode, crtc: Option<crtc::Handle>) {
let Backend::Udev(udev) = &mut self.backend else {
unreachable!()
};
let device_backend = match udev.backends.get_mut(&node) {
Some(backend) => backend,
None => {
tracing::error!("Trying to render on non-existent backend {}", node);
return;
}
};
if let Some(crtc) = crtc {
self.render_surface(node, crtc);
} else {
let crtcs: Vec<_> = device_backend.surfaces.keys().copied().collect();
for crtc in crtcs {
self.render_surface(node, crtc);
}
};
} }
fn render_surface(&mut self, node: DrmNode, crtc: crtc::Handle) { /// Render to the [`RenderSurface`] associated with the given `output`.
let Backend::Udev(udev) = &mut self.backend else { #[tracing::instrument(level = "debug", skip(self), fields(output = output.name()))]
unreachable!() fn render_surface(&mut self, output: &Output) {
}; let udev = self.backend.udev_mut();
let Some(surface) = udev let Some(surface) = render_surface_for_output(output, &mut udev.backends) else {
.backends
.get_mut(&node)
.and_then(|device| device.surfaces.get_mut(&crtc))
else {
return; return;
}; };
assert!(matches!(surface.render_state, RenderState::Scheduled(_)));
// TODO get scale from the rendersurface when supporting HiDPI // TODO get scale from the rendersurface when supporting HiDPI
let frame = udev.pointer_image.get_image( let frame = udev.pointer_image.get_image(
1, /*scale*/ 1, /*scale*/
@ -1219,17 +1233,6 @@ impl State {
texture texture
}); });
let output = if let Some(output) = self.space.outputs().find(|o| {
let udev_op_data = o.user_data().get::<UdevOutputData>();
udev_op_data
.is_some_and(|data| data.device_id == surface.device_id && data.crtc == crtc)
}) {
output.clone()
} else {
// somehow we got called with an invalid output
return;
};
let windows = self let windows = self
.focus_state .focus_state
.focus_stack .focus_stack
@ -1239,137 +1242,65 @@ impl State {
.collect::<Vec<_>>(); .collect::<Vec<_>>();
let result = render_surface( let result = render_surface(
&mut self.cursor_status, surface,
&mut renderer,
output,
&self.space, &self.space,
&windows, &windows,
&self.override_redirect_windows, &self.override_redirect_windows,
self.dnd_icon.as_ref(), self.dnd_icon.as_ref(),
surface, &mut self.cursor_status,
&mut renderer,
&output,
// self.seat.input_method(),
&pointer_image, &pointer_image,
&mut udev.pointer_element, &mut udev.pointer_element,
self.pointer_location, self.pointer_location,
&self.clock, &self.clock,
); );
let reschedule = match &result { match result {
Ok(has_rendered) => !has_rendered, Ok(true) => surface.render_state = RenderState::WaitingForVblank { dirty: false },
Err(err) => { Ok(false) | Err(_) => surface.render_state = RenderState::Idle,
tracing::warn!("Error during rendering: {:?}", err);
match err {
SwapBuffersError::AlreadySwapped => false,
SwapBuffersError::TemporaryFailure(err) => !matches!(
err.downcast_ref::<DrmError>(),
Some(&DrmError::DeviceInactive)
| Some(&DrmError::Access {
source: drm::SystemError::PermissionDenied,
..
})
),
SwapBuffersError::ContextLost(err) => panic!("Rendering loop lost: {}", err),
} }
} }
};
if reschedule {
let Some(data) = output.user_data().get::<UdevOutputData>() else {
unreachable!()
};
// Literally no idea if this refresh time calculation is doing anything, but we're
// gonna keep it here because I already added the stuff for it
let refresh_time = if let Some(mode) = data.mode {
self::utils::refresh_time(mode)
} else {
let output_refresh = match output.current_mode() {
Some(mode) => mode.refresh,
None => {
return;
} }
};
Duration::from_millis((1_000_000f32 / output_refresh as f32) as u64)
};
// If reschedule is true we either hit a temporary failure or more likely rendering fn render_surface_for_output<'a>(
// did not cause any damage on the output. In this case we just re-schedule a repaint output: &Output,
// after approx. one frame to re-test for damage. backends: &'a mut HashMap<DrmNode, UdevBackendData>,
tracing::trace!( ) -> Option<&'a mut RenderSurface> {
"reschedule repaint timer with delay {:?} on {:?}", let UdevOutputData {
refresh_time, device_id,
crtc, crtc,
); mode: _,
let timer = Timer::from_duration(refresh_time); } = output.user_data().get()?;
self.loop_handle
.insert_source(timer, move |_, _, data| { backends
data.state.render(node, Some(crtc)); .get_mut(device_id)
TimeoutAction::Drop .and_then(|device| device.surfaces.get_mut(crtc))
})
.expect("failed to schedule frame timer");
}
}
fn schedule_initial_render(
&mut self,
node: DrmNode,
crtc: crtc::Handle,
evt_handle: LoopHandle<'static, CalloopData>,
) {
let Backend::Udev(udev) = &mut self.backend else {
unreachable!()
};
let Some(surface) = udev
.backends
.get_mut(&node)
.and_then(|device| device.surfaces.get_mut(&crtc))
else {
return;
};
let node = surface.render_node;
let result = {
let mut renderer = udev
.gpu_manager
.single_renderer(&node)
.expect("failed to create MultiRenderer");
initial_render(surface, &mut renderer)
};
if let Err(err) = result {
match err {
SwapBuffersError::AlreadySwapped => {}
SwapBuffersError::TemporaryFailure(err) => {
// TODO dont reschedule after 3(?) retries
tracing::warn!("Failed to submit page_flip: {}", err);
let handle = evt_handle.clone();
evt_handle.insert_idle(move |data| {
data.state.schedule_initial_render(node, crtc, handle)
});
}
SwapBuffersError::ContextLost(err) => panic!("Rendering loop lost: {}", err),
}
}
}
} }
/// Render windows, layers, and everything else needed to the given [`RenderSurface`].
/// Also queues the frame for scanout.
#[allow(clippy::too_many_arguments)] #[allow(clippy::too_many_arguments)]
fn render_surface<'a>( fn render_surface(
cursor_status: &mut CursorImageStatus, surface: &mut RenderSurface,
renderer: &mut UdevRenderer<'_, '_>,
output: &Output,
space: &Space<WindowElement>, space: &Space<WindowElement>,
windows: &[WindowElement], windows: &[WindowElement],
override_redirect_windows: &[X11Surface], override_redirect_windows: &[X11Surface],
dnd_icon: Option<&WlSurface>, dnd_icon: Option<&WlSurface>,
surface: &'a mut SurfaceData, cursor_status: &mut CursorImageStatus,
renderer: &mut UdevRenderer<'a, '_>,
output: &Output,
// input_method: &InputMethodHandle,
pointer_image: &TextureBuffer<MultiTexture>, pointer_image: &TextureBuffer<MultiTexture>,
pointer_element: &mut PointerElement<MultiTexture>, pointer_element: &mut PointerElement<MultiTexture>,
pointer_location: Point<f64, Logical>, pointer_location: Point<f64, Logical>,
clock: &Clock<Monotonic>, clock: &Clock<Monotonic>,
) -> Result<bool, SwapBuffersError> { ) -> Result<bool, SwapBuffersError> {
use smithay::reexports::wayland_protocols::xdg::shell::server::xdg_toplevel;
let pending_wins = windows let pending_wins = windows
.iter() .iter()
.filter(|win| win.alive()) .filter(|win| win.alive())
@ -1383,6 +1314,16 @@ fn render_surface<'a>(
}; };
pending_size || win.with_state(|state| !state.loc_request_state.is_idle()) pending_size || win.with_state(|state| !state.loc_request_state.is_idle())
}) })
.filter(|win| {
if let WindowElement::Wayland(win) = win {
!win.toplevel()
.current_state()
.states
.contains(xdg_toplevel::State::Resizing)
} else {
true
}
})
.map(|win| { .map(|win| {
( (
win.class().unwrap_or("None".to_string()), win.class().unwrap_or("None".to_string()),
@ -1405,7 +1346,10 @@ fn render_surface<'a>(
.queue_frame(None) .queue_frame(None)
.map_err(Into::<SwapBuffersError>::into)?; .map_err(Into::<SwapBuffersError>::into)?;
tracing::debug!("queued no frame");
// TODO: still draw the cursor here // TODO: still draw the cursor here
surface.render_state = RenderState::WaitingForVblank { dirty: false };
return Ok(true); return Ok(true);
} }
@ -1433,7 +1377,6 @@ fn render_surface<'a>(
let time = clock.now(); let time = clock.now();
// Send frames to the cursor surface to get it to update correctly
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);
} }
@ -1462,19 +1405,3 @@ fn render_surface<'a>(
Ok(res.rendered) Ok(res.rendered)
} }
fn initial_render(
surface: &mut SurfaceData,
renderer: &mut UdevRenderer<'_, '_>,
) -> Result<(), SwapBuffersError> {
render_frame::<_, CustomRenderElements<_, WaylandSurfaceRenderElement<_>>, GlesTexture>(
&mut surface.compositor,
renderer,
&[],
[0.6, 0.6, 0.6, 1.0],
)?;
surface.compositor.queue_frame(None)?;
surface.compositor.reset_buffers();
Ok(())
}

View file

@ -1,31 +0,0 @@
// SPDX-License-Identifier: GPL-3.0-or-later
use std::time::Duration;
use smithay::reexports::drm::control::{Mode, ModeFlags};
// From niri:
// https://github.com/YaLTeR/niri/blob/ba0a6d6b8868cc6348ad1b20f683a95d5909df6b/src/backend/tty.rs#L900-L922
pub fn refresh_time(mode: Mode) -> Duration {
let clock = mode.clock() as u64;
let htotal = mode.hsync().2 as u64;
let vtotal = mode.vsync().2 as u64;
let mut numerator = htotal * vtotal * 1_000_000;
let mut denominator = clock;
if mode.flags().contains(ModeFlags::INTERLACE) {
denominator *= 2;
}
if mode.flags().contains(ModeFlags::DBLSCAN) {
numerator *= 2;
}
if mode.vscan() > 1 {
numerator *= mode.vscan() as u64;
}
let refresh_interval = (numerator + denominator / 2) / denominator;
Duration::from_nanos(refresh_interval)
}

View file

@ -23,7 +23,10 @@ use smithay::{
timer::{TimeoutAction, Timer}, timer::{TimeoutAction, Timer},
EventLoop, EventLoop,
}, },
wayland_protocols::wp::presentation_time::server::wp_presentation_feedback, wayland_protocols::{
wp::presentation_time::server::wp_presentation_feedback,
xdg::shell::server::xdg_toplevel,
},
wayland_server::{protocol::wl_surface::WlSurface, Display}, wayland_server::{protocol::wl_surface::WlSurface, Display},
}, },
utils::{IsAlive, Transform}, utils::{IsAlive, Transform},
@ -57,6 +60,13 @@ impl BackendData for Winit {
fn early_import(&mut self, _surface: &WlSurface) {} fn early_import(&mut self, _surface: &WlSurface) {}
} }
impl Backend {
fn winit_mut(&mut self) -> &mut Winit {
let Backend::Winit(winit) = self else { unreachable!() };
winit
}
}
/// Start Pinnacle as a window in a graphical environment. /// Start Pinnacle as a window in a graphical environment.
pub fn run_winit() -> anyhow::Result<()> { pub fn run_winit() -> anyhow::Result<()> {
let mut event_loop: EventLoop<CalloopData> = EventLoop::try_new()?; let mut event_loop: EventLoop<CalloopData> = EventLoop::try_new()?;
@ -163,13 +173,13 @@ pub fn run_winit() -> anyhow::Result<()> {
state.focus_state.focused_output = Some(output.clone()); state.focus_state.focused_output = Some(output.clone());
let Backend::Winit(backend) = &mut state.backend else { let winit = state.backend.winit_mut();
unreachable!()
}; winit.backend.window().set_title("Pinnacle");
state state
.shm_state .shm_state
.update_formats(backend.backend.renderer().shm_formats()); .update_formats(winit.backend.renderer().shm_formats());
state.space.map_output(&output, (0, 0)); state.space.map_output(&output, (0, 0));
@ -183,13 +193,10 @@ pub fn run_winit() -> anyhow::Result<()> {
tracing::error!("Failed to start XWayland: {err}"); tracing::error!("Failed to start XWayland: {err}");
} }
let mut pointer_element = PointerElement::<GlesTexture>::new();
let insert_ret = let insert_ret =
state state
.loop_handle .loop_handle
.insert_source(Timer::immediate(), move |_instant, _metadata, data| { .insert_source(Timer::immediate(), move |_instant, _metadata, data| {
let display_handle = &mut data.display_handle;
let state = &mut data.state; let state = &mut data.state;
let result = winit_evt_loop.dispatch_new_events(|event| match event { let result = winit_evt_loop.dispatch_new_events(|event| match event {
@ -214,7 +221,9 @@ pub fn run_winit() -> anyhow::Result<()> {
WinitEvent::Input(input_evt) => { WinitEvent::Input(input_evt) => {
state.process_input_event(input_evt); state.process_input_event(input_evt);
} }
WinitEvent::Refresh => {} WinitEvent::Refresh => {
state.render_window(&output);
}
}); });
match result { match result {
@ -224,17 +233,37 @@ pub fn run_winit() -> anyhow::Result<()> {
} }
}; };
if let CursorImageStatus::Surface(surface) = &state.cursor_status { state.render_window(&output);
if !surface.alive() {
state.cursor_status = CursorImageStatus::default_named(); TimeoutAction::ToDuration(Duration::from_micros(((1.0 / 144.0) * 1000000.0) as u64))
} });
if let Err(err) = insert_ret {
anyhow::bail!("Failed to insert winit events into event loop: {err}");
} }
let cursor_visible = !matches!(state.cursor_status, CursorImageStatus::Surface(_)); event_loop.run(
Some(Duration::from_micros(((1.0 / 144.0) * 1000000.0) as u64)),
&mut CalloopData {
display_handle,
state,
},
|data| {
data.state.space.refresh();
data.state.popup_manager.cleanup();
data.display_handle
.flush_clients()
.expect("failed to flush client buffers");
},
)?;
pointer_element.set_status(state.cursor_status.clone()); Ok(())
}
let pending_wins = state impl State {
fn render_window(&mut self, output: &Output) {
let winit = self.backend.winit_mut();
let pending_wins = self
.windows .windows
.iter() .iter()
.filter(|win| win.alive()) .filter(|win| win.alive())
@ -248,6 +277,16 @@ pub fn run_winit() -> anyhow::Result<()> {
}; };
pending_size || win.with_state(|state| !state.loc_request_state.is_idle()) pending_size || win.with_state(|state| !state.loc_request_state.is_idle())
}) })
.filter(|win| {
if let WindowElement::Wayland(win) = win {
!win.toplevel()
.current_state()
.states
.contains(xdg_toplevel::State::Resizing)
} else {
true
}
})
.map(|win| { .map(|win| {
( (
win.class().unwrap_or("None".to_string()), win.class().unwrap_or("None".to_string()),
@ -260,7 +299,7 @@ pub fn run_winit() -> anyhow::Result<()> {
if !pending_wins.is_empty() { if !pending_wins.is_empty() {
// tracing::debug!("Skipping frame, waiting on {pending_wins:?}"); // tracing::debug!("Skipping frame, waiting on {pending_wins:?}");
let op_clone = output.clone(); let op_clone = output.clone();
state.loop_handle.insert_idle(move |dt| { self.loop_handle.insert_idle(move |dt| {
for win in dt.state.windows.iter() { for win in dt.state.windows.iter() {
win.send_frame( win.send_frame(
&op_clone, &op_clone,
@ -271,49 +310,50 @@ pub fn run_winit() -> anyhow::Result<()> {
} }
}); });
state.space.refresh();
state.popup_manager.cleanup();
display_handle
.flush_clients()
.expect("failed to flush client buffers");
// TODO: still draw the cursor here // TODO: still draw the cursor here
return TimeoutAction::ToDuration(Duration::from_millis(1)); return;
} }
let full_redraw = &mut winit.full_redraw;
let Backend::Winit(backend) = &mut state.backend else {
unreachable!()
};
let full_redraw = &mut backend.full_redraw;
*full_redraw = full_redraw.saturating_sub(1); *full_redraw = full_redraw.saturating_sub(1);
state.focus_state.fix_up_focus(&mut state.space); self.focus_state.fix_up_focus(&mut self.space);
if let CursorImageStatus::Surface(surface) = &self.cursor_status {
if !surface.alive() {
self.cursor_status = CursorImageStatus::default_named();
}
}
let cursor_visible = !matches!(self.cursor_status, CursorImageStatus::Surface(_));
let mut pointer_element = PointerElement::<GlesTexture>::new();
pointer_element.set_status(self.cursor_status.clone());
let output_render_elements = crate::render::generate_render_elements( let output_render_elements = crate::render::generate_render_elements(
&output, output,
backend.backend.renderer(), winit.backend.renderer(),
&state.space, &self.space,
&state.focus_state.focus_stack, &self.focus_state.focus_stack,
&state.override_redirect_windows, &self.override_redirect_windows,
state.pointer_location, self.pointer_location,
&mut state.cursor_status, &mut self.cursor_status,
state.dnd_icon.as_ref(), self.dnd_icon.as_ref(),
// state.seat.input_method(), // self.seat.input_method(),
&mut pointer_element, &mut pointer_element,
None, None,
); );
let render_res = backend.backend.bind().and_then(|_| { let render_res = winit.backend.bind().and_then(|_| {
let age = if *full_redraw > 0 { let age = if *full_redraw > 0 {
0 0
} else { } else {
backend.backend.buffer_age().unwrap_or(0) winit.backend.buffer_age().unwrap_or(0)
}; };
let renderer = backend.backend.renderer(); let renderer = winit.backend.renderer();
backend winit
.damage_tracker .damage_tracker
.render_output(renderer, age, &output_render_elements, [0.6, 0.6, 0.6, 1.0]) .render_output(renderer, age, &output_render_elements, [0.6, 0.6, 0.6, 1.0])
.map_err(|err| match err { .map_err(|err| match err {
@ -327,49 +367,41 @@ pub fn run_winit() -> anyhow::Result<()> {
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 {
// tracing::debug!("damage rects are {damage:?}"); // tracing::debug!("damage rects are {damage:?}");
if let Err(err) = backend.backend.submit(Some(&damage)) { if let Err(err) = winit.backend.submit(Some(&damage)) {
tracing::warn!("{}", err); tracing::warn!("{}", err);
} }
} }
backend.backend.window().set_cursor_visible(cursor_visible); winit.backend.window().set_cursor_visible(cursor_visible);
let time = state.clock.now(); let time = self.clock.now();
// Send frames to the cursor surface so it updates correctly // Send frames to the cursor surface so it updates correctly
if let CursorImageStatus::Surface(surf) = &state.cursor_status { if let CursorImageStatus::Surface(surf) = &self.cursor_status {
if let Some(op) = state.focus_state.focused_output.as_ref() { if let Some(op) = self.focus_state.focused_output.as_ref() {
send_frames_surface_tree( send_frames_surface_tree(surf, op, time, Some(Duration::ZERO), |_, _| None);
surf,
op,
time,
Some(Duration::ZERO),
|_, _| None,
);
} }
} }
super::post_repaint( super::post_repaint(
&output, output,
&render_output_result.states, &render_output_result.states,
&state.space, &self.space,
None, None,
time.into(), time.into(),
); );
if has_rendered { if has_rendered {
let mut output_presentation_feedback = take_presentation_feedback( let mut output_presentation_feedback = take_presentation_feedback(
&output, output,
&state.space, &self.space,
&render_output_result.states, &render_output_result.states,
); );
output_presentation_feedback.presented( output_presentation_feedback.presented(
time, time,
output output
.current_mode() .current_mode()
.map(|mode| { .map(|mode| Duration::from_secs_f64(1000f64 / mode.refresh as f64))
Duration::from_secs_f64(1000f64 / mode.refresh as f64)
})
.unwrap_or_default(), .unwrap_or_default(),
0, 0,
wp_presentation_feedback::Kind::Vsync, wp_presentation_feedback::Kind::Vsync,
@ -380,30 +412,5 @@ pub fn run_winit() -> anyhow::Result<()> {
tracing::warn!("{}", err); tracing::warn!("{}", err);
} }
} }
state.space.refresh();
state.popup_manager.cleanup();
display_handle
.flush_clients()
.expect("failed to flush client buffers");
TimeoutAction::ToDuration(Duration::from_micros(((1.0 / 144.0) * 1000000.0) as u64))
});
if let Err(err) = insert_ret {
anyhow::bail!("Failed to insert winit events into event loop: {err}");
} }
event_loop.run(
Some(Duration::from_micros(((1.0 / 144.0) * 1000000.0) as u64)),
&mut CalloopData {
display_handle,
state,
},
|_data| {
// println!("{}", _data.state.space.elements().count());
},
)?;
Ok(())
} }

View file

@ -10,7 +10,7 @@ use smithay::{
delegate_compositor, delegate_data_device, delegate_fractional_scale, delegate_layer_shell, delegate_compositor, delegate_data_device, delegate_fractional_scale, delegate_layer_shell,
delegate_output, delegate_presentation, delegate_primary_selection, delegate_relative_pointer, delegate_output, delegate_presentation, delegate_primary_selection, delegate_relative_pointer,
delegate_seat, delegate_shm, delegate_viewporter, delegate_seat, delegate_shm, delegate_viewporter,
desktop::{self, layer_map_for_output, PopupKind, WindowSurfaceType}, desktop::{self, find_popup_root_surface, layer_map_for_output, PopupKind, WindowSurfaceType},
input::{pointer::CursorImageStatus, Seat, SeatHandler, SeatState}, input::{pointer::CursorImageStatus, Seat, SeatHandler, SeatState},
output::Output, output::Output,
reexports::{ reexports::{
@ -105,11 +105,12 @@ impl CompositorHandler for State {
utils::on_commit_buffer_handler::<Self>(surface); utils::on_commit_buffer_handler::<Self>(surface);
self.backend.early_import(surface); self.backend.early_import(surface);
if !compositor::is_sync_subsurface(surface) {
let mut root = surface.clone(); let mut root = surface.clone();
while let Some(parent) = compositor::get_parent(&root) { while let Some(parent) = compositor::get_parent(&root) {
root = parent; root = parent;
} }
if !compositor::is_sync_subsurface(surface) {
if let Some(win @ WindowElement::Wayland(window)) = &self.window_for_surface(&root) { if let Some(win @ WindowElement::Wayland(window)) = &self.window_for_surface(&root) {
// tracing::debug!("window commit thing {:?}", win.class()); // tracing::debug!("window commit thing {:?}", win.class());
window.on_commit(); window.on_commit();
@ -129,9 +130,41 @@ impl CompositorHandler for State {
crate::grab::resize_grab::handle_commit(self, surface); crate::grab::resize_grab::handle_commit(self, surface);
// if let Some(window) = self.window_for_surface(surface) { // `surface` is a root window
// tracing::debug!("commit on window {:?}", window.class()); let Some(output) = self
// } .window_for_surface(surface)
.and_then(|win| win.output(self))
.or_else(|| {
// `surface` is a descendant of a root window
self.window_for_surface(&root)
.and_then(|win| win.output(self))
})
.or_else(|| {
// `surface` is a popup
self.popup_manager
.find_popup(surface)
.and_then(|popup| find_popup_root_surface(&popup).ok())
.and_then(|surf| self.window_for_surface(&surf))
.and_then(|win| win.output(self))
})
.or_else(|| {
// `surface` is a layer surface
self.space
.outputs()
.find(|op| {
let layer_map = layer_map_for_output(op);
layer_map
.layer_for_surface(surface, WindowSurfaceType::ALL)
.is_some()
})
.cloned()
})
// TODO: cursor surface and dnd icon
else {
return;
};
self.schedule_render(&output);
} }
fn client_compositor_state<'a>(&self, client: &'a Client) -> &'a CompositorClientState { fn client_compositor_state<'a>(&self, client: &'a Client) -> &'a CompositorClientState {

View file

@ -1,5 +1,3 @@
use std::time::Duration;
use smithay::{ use smithay::{
delegate_xdg_shell, delegate_xdg_shell,
desktop::{ desktop::{
@ -150,6 +148,8 @@ impl XdgShellHandler for State {
.get_keyboard() .get_keyboard()
.expect("Seat had no keyboard") .expect("Seat had no keyboard")
.set_focus(self, focus, SERIAL_COUNTER.next_serial()); .set_focus(self, focus, SERIAL_COUNTER.next_serial());
self.schedule_render(&output);
} }
} }
@ -749,14 +749,6 @@ impl XdgShellHandler for State {
state.loc_request_state = state.loc_request_state =
LocationRequestState::Acknowledged(new_loc); LocationRequestState::Acknowledged(new_loc);
}); });
if let Some(op) = window.output(self) {
window.send_frame(
&op,
self.clock.now(),
Some(Duration::ZERO),
|_, _| Some(op.clone()),
);
}
} }
} }
Configure::Popup(_) => todo!(), Configure::Popup(_) => todo!(),

View file

@ -218,6 +218,8 @@ impl XwmHandler for CalloopData {
.get_keyboard() .get_keyboard()
.expect("Seat had no keyboard") .expect("Seat had no keyboard")
.set_focus(&mut self.state, focus, SERIAL_COUNTER.next_serial()); .set_focus(&mut self.state, focus, SERIAL_COUNTER.next_serial());
self.state.schedule_render(&output);
} }
} }
if !window.is_override_redirect() { if !window.is_override_redirect() {
@ -267,6 +269,8 @@ impl XwmHandler for CalloopData {
.get_keyboard() .get_keyboard()
.expect("Seat had no keyboard") .expect("Seat had no keyboard")
.set_focus(&mut self.state, focus, SERIAL_COUNTER.next_serial()); .set_focus(&mut self.state, focus, SERIAL_COUNTER.next_serial());
self.state.schedule_render(&output);
} }
} }
tracing::debug!("destroyed x11 window"); tracing::debug!("destroyed x11 window");
@ -313,7 +317,6 @@ impl XwmHandler for CalloopData {
}; };
tracing::debug!("configure notify with geo: {geometry:?}"); tracing::debug!("configure notify with geo: {geometry:?}");
self.state.space.map_element(win, geometry.loc, true); self.state.space.map_element(win, geometry.loc, true);
// TODO: anvil has a TODO here
} }
fn maximize_request(&mut self, _xwm: XwmId, window: X11Surface) { fn maximize_request(&mut self, _xwm: XwmId, window: X11Surface) {

View file

@ -657,8 +657,8 @@ impl State {
let surface_under = self.surface_under(self.pointer_location); let surface_under = self.surface_under(self.pointer_location);
// tracing::info!("{:?}", self.pointer_location); // tracing::info!("{:?}", self.pointer_location);
if let Some(ptr) = self.seat.get_pointer() { if let Some(pointer) = self.seat.get_pointer() {
ptr.motion( pointer.motion(
self, self,
surface_under.clone(), surface_under.clone(),
&MotionEvent { &MotionEvent {
@ -668,7 +668,7 @@ impl State {
}, },
); );
ptr.relative_motion( pointer.relative_motion(
self, self,
surface_under, surface_under,
&RelativeMotionEvent { &RelativeMotionEvent {
@ -678,7 +678,15 @@ impl State {
}, },
); );
ptr.frame(self); pointer.frame(self);
self.schedule_render(
&self
.focus_state
.focused_output
.clone()
.expect("no focused output"),
);
} }
} }
} }

View file

@ -31,7 +31,11 @@ use smithay::{
xwayland::X11Surface, xwayland::X11Surface,
}; };
use crate::{state::WithState, window::WindowElement}; use crate::{
backend::Backend,
state::{State, WithState},
window::WindowElement,
};
use self::pointer::{PointerElement, PointerRenderElement}; use self::pointer::{PointerElement, PointerRenderElement};
@ -400,3 +404,12 @@ pub fn take_presentation_feedback(
output_presentation_feedback output_presentation_feedback
} }
impl State {
pub fn schedule_render(&mut self, output: &Output) {
// I'm relegating winit to render every frame because it's not my priority right now
if let Backend::Udev(udev) = &mut self.backend {
udev.schedule_render(&self.loop_handle, output);
}
}
}

View file

@ -10,7 +10,7 @@ use smithay::{
}, },
ImportAll, Renderer, Texture, ImportAll, Renderer, Texture,
}, },
input::pointer::{CursorIcon, CursorImageStatus}, input::pointer::CursorImageStatus,
render_elements, render_elements,
utils::{Physical, Point, Scale}, utils::{Physical, Point, Scale},
}; };

View file

@ -113,6 +113,7 @@ impl State {
window.change_geometry(Rectangle::from_loc_and_size(window_loc, window_size)); window.change_geometry(Rectangle::from_loc_and_size(window_loc, window_size));
if let Some(output) = window.output(self) { if let Some(output) = window.output(self) {
self.update_windows(&output); self.update_windows(&output);
self.schedule_render(&output);
} }
} }
Msg::MoveWindowToTag { window_id, tag_id } => { Msg::MoveWindowToTag { window_id, tag_id } => {
@ -123,7 +124,7 @@ impl State {
}); });
let Some(output) = tag.output(self) else { return }; let Some(output) = tag.output(self) else { return };
self.update_windows(&output); self.update_windows(&output);
// self.re_layout(&output); self.schedule_render(&output);
} }
Msg::ToggleTagOnWindow { window_id, tag_id } => { Msg::ToggleTagOnWindow { window_id, tag_id } => {
let Some(window) = window_id.window(self) else { return }; let Some(window) = window_id.window(self) else { return };
@ -139,7 +140,7 @@ impl State {
let Some(output) = tag.output(self) else { return }; let Some(output) = tag.output(self) else { return };
self.update_windows(&output); self.update_windows(&output);
// self.re_layout(&output); self.schedule_render(&output);
} }
Msg::ToggleFloating { window_id } => { Msg::ToggleFloating { window_id } => {
let Some(window) = window_id.window(self) else { return }; let Some(window) = window_id.window(self) else { return };
@ -147,6 +148,12 @@ impl State {
let Some(output) = window.output(self) else { return }; let Some(output) = window.output(self) else { return };
self.update_windows(&output); self.update_windows(&output);
// Sometimes toggling won't change the window size,
// causing no commit.
//
// Schedule a render in case the window moves.
self.schedule_render(&output);
} }
Msg::ToggleFullscreen { window_id } => { Msg::ToggleFullscreen { window_id } => {
let Some(window) = window_id.window(self) else { return }; let Some(window) = window_id.window(self) else { return };
@ -154,6 +161,7 @@ impl State {
let Some(output) = window.output(self) else { return }; let Some(output) = window.output(self) else { return };
self.update_windows(&output); self.update_windows(&output);
self.schedule_render(&output);
} }
Msg::ToggleMaximized { window_id } => { Msg::ToggleMaximized { window_id } => {
let Some(window) = window_id.window(self) else { return }; let Some(window) = window_id.window(self) else { return };
@ -161,6 +169,7 @@ impl State {
let Some(output) = window.output(self) else { return }; let Some(output) = window.output(self) else { return };
self.update_windows(&output); self.update_windows(&output);
self.schedule_render(&output);
} }
Msg::AddWindowRule { cond, rule } => { Msg::AddWindowRule { cond, rule } => {
self.config.window_rules.push((cond, rule)); self.config.window_rules.push((cond, rule));
@ -251,6 +260,7 @@ impl State {
if let Some(output) = tag.output(self) { if let Some(output) = tag.output(self) {
self.update_windows(&output); self.update_windows(&output);
self.update_focus(&output); self.update_focus(&output);
self.schedule_render(&output);
} }
} }
} }
@ -265,6 +275,7 @@ impl State {
}); });
self.update_windows(&output); self.update_windows(&output);
self.update_focus(&output); self.update_focus(&output);
self.schedule_render(&output);
} }
Msg::AddTags { Msg::AddTags {
output_name, output_name,