wayland_bridge/
subcompositor.rs1use crate::client::Client;
6use crate::compositor::{
7 PlaceSubsurfaceParams, Surface, SurfaceCommand, SurfaceRelation, SurfaceRole,
8};
9use crate::display::Callback;
10use crate::object::{NewObjectExt, ObjectRef, RequestReceiver};
11use anyhow::{format_err, Error};
12use fuchsia_wayland_core as wl;
13use std::mem;
14use wayland_server_protocol::{
15 WlSubcompositor, WlSubcompositorRequest, WlSubsurface, WlSubsurfaceRequest,
16};
17
18pub struct Subcompositor;
34
35impl Subcompositor {
36 pub fn new() -> Self {
38 Subcompositor
39 }
40}
41
42impl RequestReceiver<WlSubcompositor> for Subcompositor {
43 fn receive(
44 this: ObjectRef<Self>,
45 request: WlSubcompositorRequest,
46 client: &mut Client,
47 ) -> Result<(), Error> {
48 match request {
49 WlSubcompositorRequest::Destroy => {
50 client.delete_id(this.id())?;
51 }
52 WlSubcompositorRequest::GetSubsurface { id, surface, parent } => {
53 let subsurface = Subsurface::new(surface, parent);
54 let surface_ref = subsurface.surface_ref;
55 let parent_ref = subsurface.parent_ref;
56 subsurface.attach_to_parent(client)?;
57 let subsurface_ref = id.implement(client, subsurface)?;
58 surface_ref.get_mut(client)?.set_role(SurfaceRole::Subsurface(subsurface_ref))?;
59 parent_ref
60 .get_mut(client)?
61 .enqueue(SurfaceCommand::AddSubsurface(surface_ref, subsurface_ref));
62 }
63 }
64 Ok(())
65 }
66}
67
68#[derive(Debug, Copy, Clone, PartialEq)]
76pub enum SubsurfaceMode {
77 Sync,
78 Desync,
79}
80
81pub struct Subsurface {
82 surface_ref: ObjectRef<Surface>,
84
85 parent_ref: ObjectRef<Surface>,
87
88 mode: SubsurfaceMode,
92
93 pending_commands: Vec<SurfaceCommand>,
98
99 pending_callbacks: Vec<ObjectRef<Callback>>,
104}
105
106impl Subsurface {
107 pub fn new(surface: wl::ObjectId, parent: wl::ObjectId) -> Self {
108 Self {
109 surface_ref: surface.into(),
110 parent_ref: parent.into(),
111 mode: SubsurfaceMode::Sync,
112 pending_commands: Vec::new(),
113 pending_callbacks: Vec::new(),
114 }
115 }
116
117 pub fn surface(&self) -> ObjectRef<Surface> {
119 self.surface_ref
120 }
121
122 pub fn is_sync(&self) -> bool {
124 self.mode == SubsurfaceMode::Sync
125 }
126
127 pub fn add_pending_commands(&mut self, mut commands: Vec<SurfaceCommand>) {
136 assert!(self.is_sync(), "Desync subsurfaces have no pending state");
137 self.pending_commands.append(&mut commands);
138 }
139
140 pub fn take_pending_state(&mut self) -> (Vec<SurfaceCommand>, Vec<ObjectRef<Callback>>) {
143 let commands = mem::replace(&mut self.pending_commands, Vec::new());
144 let callbacks = mem::replace(&mut self.pending_callbacks, Vec::new());
145 (commands, callbacks)
146 }
147
148 pub fn finalize_commit(&mut self, callbacks: &mut Vec<ObjectRef<Callback>>) -> bool {
149 if self.is_sync() {
150 self.pending_callbacks.append(callbacks);
153 false
154 } else {
155 true
156 }
157 }
158
159 fn attach_to_parent(&self, client: &mut Client) -> Result<(), Error> {
160 let flatland = match self.parent_ref.get(client)?.flatland() {
161 Some(s) => s,
162 None => return Err(format_err!("Parent surface has no flatland instance!")),
163 };
164 self.surface_ref.get_mut(client)?.set_flatland(flatland.clone())?;
165
166 let parent_transform = self.parent_ref.get(client)?.transform().unwrap();
169 let child_transform = self.surface_ref.get(client)?.transform().unwrap();
170 flatland
171 .borrow()
172 .proxy()
173 .add_child(&parent_transform, &child_transform)
174 .expect("fidl error");
175
176 Ok(())
177 }
178
179 fn detach_from_parent(&self, client: &Client) -> Result<(), Error> {
180 if let Some(flatland) = self.parent_ref.get(client)?.flatland() {
181 let parent_transform = *self.parent_ref.get(client)?.transform().unwrap();
185 if let Some(child_transform) = self.surface_ref.get(client)?.transform() {
186 flatland
187 .borrow()
188 .proxy()
189 .remove_child(&parent_transform, &child_transform)
190 .expect("fidl error");
191 }
192 }
193 Ok(())
194 }
195}
196
197impl RequestReceiver<WlSubsurface> for Subsurface {
198 fn receive(
199 this: ObjectRef<Self>,
200 request: WlSubsurfaceRequest,
201 client: &mut Client,
202 ) -> Result<(), Error> {
203 match request {
204 WlSubsurfaceRequest::Destroy => {
205 let parent_ref = {
206 let subsurface = this.get(client)?;
207 subsurface.detach_from_parent(client)?;
208 subsurface.parent_ref
209 };
210 parent_ref.get_mut(client)?.detach_subsurface(this);
211 client.delete_id(this.id())?;
212 }
213 WlSubsurfaceRequest::SetPosition { x, y } => {
214 let surface_ref = this.get(client)?.surface_ref;
215 surface_ref.get_mut(client)?.enqueue(SurfaceCommand::SetPosition(x, y));
216 }
217 WlSubsurfaceRequest::PlaceAbove { sibling } => {
218 let parent_ref = this.get(client)?.parent_ref;
219 parent_ref.get_mut(client)?.enqueue(SurfaceCommand::PlaceSubsurface(
220 PlaceSubsurfaceParams {
221 subsurface: this,
222 sibling: sibling.into(),
223 relation: SurfaceRelation::Above,
224 },
225 ));
226 }
227 WlSubsurfaceRequest::PlaceBelow { sibling } => {
228 let parent_ref = this.get(client)?.parent_ref;
229 parent_ref.get_mut(client)?.enqueue(SurfaceCommand::PlaceSubsurface(
230 PlaceSubsurfaceParams {
231 subsurface: this,
232 sibling: sibling.into(),
233 relation: SurfaceRelation::Below,
234 },
235 ));
236 }
237 WlSubsurfaceRequest::SetSync => {
244 this.get_mut(client)?.mode = SubsurfaceMode::Sync;
245 }
246 WlSubsurfaceRequest::SetDesync => {
247 let (commands, callbacks, surface_ref) = {
248 let this = this.get_mut(client)?;
249 let (commands, callbacks) = this.take_pending_state();
250 (commands, callbacks, this.surface_ref)
251 };
252 assert!(callbacks.is_empty());
256 let surface = surface_ref.get_mut(client)?;
257 for command in commands {
258 surface.enqueue(command);
259 }
260 this.get_mut(client)?.mode = SubsurfaceMode::Desync;
261 }
262 }
263 Ok(())
264 }
265}