forked from iced-rs/iced
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
sctk: Add
Subsurface
widget (iced-rs#79)
This adds a widget that attaches an shm or dma buffer to a subsurface, scaled with `wp_viewporter`. By exposing this as a widget, rather than as a type of window, it can be positioned and scaled like any other iced widget. It provides an API that's similar to an iced image. The initial version of this just took a `wl_buffer`. But this makes buffer re-use problematic. In particular, the docs for `wl_surface::attach` note that `wl_buffer::release` events become unreliable if a buffer is attached to multiple surfaces. And indicates that a client should create multiple `wl_buffer` instances, or use `wp_linux_buffer_release`. So we store information about the buffer, and create `wl_buffer`s as needed. `SubsurfaceBuffer::new` also returns a future that's signaled when all references are destroyed, both `wl_buffer`s and any instance of the `SubsurfaceBuffer` that might still be used in the `view`. So this seems like the best solution for now, within the model-view-update architecture. This has two examples: `sctk_subsurface`, showing a single-color shm buffer, and `sctk_subsurface_gst`, which plays an h264 video to a subsurface with vaapi decoding.
- Loading branch information
Showing
14 changed files
with
1,074 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,14 @@ | ||
[package] | ||
name = "sctk_subsurface" | ||
version = "0.1.0" | ||
edition = "2021" | ||
|
||
[dependencies] | ||
sctk = { package = "smithay-client-toolkit", git = "https://github.com/smithay/client-toolkit", rev = "828b1eb" } | ||
iced = { path = "../..", default-features = false, features = ["wayland", "debug", "a11y"] } | ||
iced_runtime = { path = "../../runtime" } | ||
iced_sctk = { path = "../../sctk" } | ||
env_logger = "0.10" | ||
futures-channel = "0.3.29" | ||
calloop = "0.12.3" | ||
rustix = { version = "0.38.30", features = ["fs", "shm"] } |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,100 @@ | ||
// Shows a subsurface with a 1x1 px red buffer, stretch to window size | ||
|
||
use iced::{ | ||
event::wayland::Event as WaylandEvent, wayland::InitialSurface, | ||
widget::text, window, Application, Command, Element, Length, Subscription, | ||
Theme, | ||
}; | ||
use iced_sctk::subsurface_widget::SubsurfaceBuffer; | ||
use sctk::reexports::client::{Connection, Proxy}; | ||
|
||
mod wayland; | ||
|
||
fn main() { | ||
let mut settings = iced::Settings::default(); | ||
settings.initial_surface = InitialSurface::XdgWindow(Default::default()); | ||
SubsurfaceApp::run(settings).unwrap(); | ||
} | ||
|
||
#[derive(Debug, Clone, Default)] | ||
struct SubsurfaceApp { | ||
connection: Option<Connection>, | ||
red_buffer: Option<SubsurfaceBuffer>, | ||
} | ||
|
||
#[derive(Debug, Clone)] | ||
pub enum Message { | ||
WaylandEvent(WaylandEvent), | ||
Wayland(wayland::Event), | ||
} | ||
|
||
impl Application for SubsurfaceApp { | ||
type Executor = iced::executor::Default; | ||
type Message = Message; | ||
type Flags = (); | ||
type Theme = Theme; | ||
|
||
fn new(_flags: ()) -> (SubsurfaceApp, Command<Self::Message>) { | ||
( | ||
SubsurfaceApp { | ||
..SubsurfaceApp::default() | ||
}, | ||
Command::none(), | ||
) | ||
} | ||
|
||
fn title(&self, _id: window::Id) -> String { | ||
String::from("SubsurfaceApp") | ||
} | ||
|
||
fn update(&mut self, message: Self::Message) -> Command<Self::Message> { | ||
match message { | ||
Message::WaylandEvent(evt) => match evt { | ||
WaylandEvent::Output(_evt, output) => { | ||
if self.connection.is_none() { | ||
if let Some(backend) = output.backend().upgrade() { | ||
self.connection = | ||
Some(Connection::from_backend(backend)); | ||
} | ||
} | ||
} | ||
_ => {} | ||
}, | ||
Message::Wayland(evt) => match evt { | ||
wayland::Event::RedBuffer(buffer) => { | ||
self.red_buffer = Some(buffer); | ||
} | ||
}, | ||
} | ||
Command::none() | ||
} | ||
|
||
fn view(&self, _id: window::Id) -> Element<Self::Message> { | ||
if let Some(buffer) = &self.red_buffer { | ||
iced_sctk::subsurface_widget::Subsurface::new(1, 1, buffer) | ||
.width(Length::Fill) | ||
.height(Length::Fill) | ||
.into() | ||
} else { | ||
text("No subsurface").into() | ||
} | ||
} | ||
|
||
fn subscription(&self) -> Subscription<Self::Message> { | ||
let mut subscriptions = vec![iced::event::listen_with(|evt, _| { | ||
if let iced::Event::PlatformSpecific( | ||
iced::event::PlatformSpecific::Wayland(evt), | ||
) = evt | ||
{ | ||
Some(Message::WaylandEvent(evt)) | ||
} else { | ||
None | ||
} | ||
})]; | ||
if let Some(connection) = &self.connection { | ||
subscriptions | ||
.push(wayland::subscription(connection).map(Message::Wayland)); | ||
} | ||
Subscription::batch(subscriptions) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,125 @@ | ||
use futures_channel::mpsc; | ||
use iced::futures::{FutureExt, SinkExt}; | ||
use iced_sctk::subsurface_widget::{Shmbuf, SubsurfaceBuffer}; | ||
use rustix::{io::Errno, shm::ShmOFlags}; | ||
use sctk::{ | ||
reexports::{ | ||
calloop_wayland_source::WaylandSource, | ||
client::{ | ||
delegate_noop, | ||
globals::registry_queue_init, | ||
protocol::{wl_buffer::WlBuffer, wl_shm}, | ||
Connection, | ||
}, | ||
}, | ||
registry::{ProvidesRegistryState, RegistryState}, | ||
shm::{Shm, ShmHandler}, | ||
}; | ||
use std::{ | ||
os::fd::OwnedFd, | ||
sync::Arc, | ||
thread, | ||
time::{SystemTime, UNIX_EPOCH}, | ||
}; | ||
|
||
#[derive(Debug, Clone)] | ||
pub enum Event { | ||
RedBuffer(SubsurfaceBuffer), | ||
} | ||
|
||
struct AppData { | ||
registry_state: RegistryState, | ||
shm_state: Shm, | ||
} | ||
|
||
impl ProvidesRegistryState for AppData { | ||
fn registry(&mut self) -> &mut RegistryState { | ||
&mut self.registry_state | ||
} | ||
|
||
sctk::registry_handlers!(); | ||
} | ||
|
||
impl ShmHandler for AppData { | ||
fn shm_state(&mut self) -> &mut Shm { | ||
&mut self.shm_state | ||
} | ||
} | ||
|
||
pub fn subscription(connection: &Connection) -> iced::Subscription<Event> { | ||
let connection = connection.clone(); | ||
iced::subscription::run_with_id( | ||
"wayland-sub", | ||
async { start(connection).await }.flatten_stream(), | ||
) | ||
} | ||
|
||
async fn start(conn: Connection) -> mpsc::Receiver<Event> { | ||
let (mut sender, receiver) = mpsc::channel(20); | ||
|
||
let (globals, event_queue) = registry_queue_init(&conn).unwrap(); | ||
let qh = event_queue.handle(); | ||
|
||
let mut app_data = AppData { | ||
registry_state: RegistryState::new(&globals), | ||
shm_state: Shm::bind(&globals, &qh).unwrap(), | ||
}; | ||
|
||
let fd = create_memfile().unwrap(); | ||
rustix::io::write(&fd, &[0, 0, 255, 255]).unwrap(); | ||
|
||
let shmbuf = Shmbuf { | ||
fd, | ||
offset: 0, | ||
width: 1, | ||
height: 1, | ||
stride: 4, | ||
format: wl_shm::Format::Xrgb8888, | ||
}; | ||
|
||
let buffer = SubsurfaceBuffer::new(Arc::new(shmbuf.into())).0; | ||
let _ = sender.send(Event::RedBuffer(buffer)).await; | ||
|
||
thread::spawn(move || { | ||
let mut event_loop = calloop::EventLoop::try_new().unwrap(); | ||
WaylandSource::new(conn, event_queue) | ||
.insert(event_loop.handle()) | ||
.unwrap(); | ||
loop { | ||
event_loop.dispatch(None, &mut app_data).unwrap(); | ||
} | ||
}); | ||
|
||
receiver | ||
} | ||
|
||
fn create_memfile() -> rustix::io::Result<OwnedFd> { | ||
loop { | ||
let flags = ShmOFlags::CREATE | ShmOFlags::EXCL | ShmOFlags::RDWR; | ||
|
||
let time = SystemTime::now(); | ||
let name = format!( | ||
"/iced-sctk-{}", | ||
time.duration_since(UNIX_EPOCH).unwrap().subsec_nanos() | ||
); | ||
|
||
match rustix::io::retry_on_intr(|| { | ||
rustix::shm::shm_open(&name, flags, 0600.into()) | ||
}) { | ||
Ok(fd) => match rustix::shm::shm_unlink(&name) { | ||
Ok(_) => return Ok(fd), | ||
Err(errno) => { | ||
return Err(errno.into()); | ||
} | ||
}, | ||
Err(Errno::EXIST) => { | ||
continue; | ||
} | ||
Err(err) => return Err(err.into()), | ||
} | ||
} | ||
} | ||
|
||
delegate_noop!(AppData: ignore WlBuffer); | ||
sctk::delegate_registry!(AppData); | ||
sctk::delegate_shm!(AppData); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
[package] | ||
name = "sctk_subsurface_gst" | ||
version = "0.1.0" | ||
edition = "2021" | ||
|
||
[dependencies] | ||
sctk = { package = "smithay-client-toolkit", git = "https://github.com/smithay/client-toolkit", rev = "828b1eb" } | ||
iced = { path = "../..", default-features = false, features = ["wayland", "debug", "a11y"] } | ||
iced_runtime = { path = "../../runtime" } | ||
iced_sctk = { path = "../../sctk" } | ||
env_logger = "0.10" | ||
futures-channel = "0.3.29" | ||
calloop = "0.12.3" | ||
gst = { package = "gstreamer", version = "0.21.3" } | ||
gst-app = { package = "gstreamer-app", version = "0.21.2" } | ||
gst-video = { package = "gstreamer-video", version = "0.21.2" } | ||
gst-allocators = { package = "gstreamer-allocators", version = "0.21.2" } | ||
drm-fourcc = "2.2.0" |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,84 @@ | ||
// Shows a subsurface with a 1x1 px red buffer, stretch to window size | ||
|
||
use iced::{ | ||
wayland::InitialSurface, widget::text, window, Application, Command, | ||
Element, Length, Subscription, Theme, | ||
}; | ||
use iced_sctk::subsurface_widget::SubsurfaceBuffer; | ||
use std::{env, path::Path}; | ||
|
||
mod pipewire; | ||
|
||
fn main() { | ||
let args = env::args(); | ||
if args.len() != 2 { | ||
eprintln!("usage: sctk_subsurface_gst [h264 mp4 path]"); | ||
return; | ||
} | ||
let path = args.skip(1).next().unwrap(); | ||
if !Path::new(&path).exists() { | ||
eprintln!("File `{path}` not found."); | ||
return; | ||
} | ||
let mut settings = iced::Settings::with_flags(path); | ||
settings.initial_surface = InitialSurface::XdgWindow(Default::default()); | ||
SubsurfaceApp::run(settings).unwrap(); | ||
} | ||
|
||
#[derive(Debug, Clone, Default)] | ||
struct SubsurfaceApp { | ||
path: String, | ||
buffer: Option<SubsurfaceBuffer>, | ||
} | ||
|
||
#[derive(Debug, Clone)] | ||
pub enum Message { | ||
Pipewire(pipewire::Event), | ||
} | ||
|
||
impl Application for SubsurfaceApp { | ||
type Executor = iced::executor::Default; | ||
type Message = Message; | ||
type Flags = String; | ||
type Theme = Theme; | ||
|
||
fn new(flags: String) -> (SubsurfaceApp, Command<Self::Message>) { | ||
( | ||
SubsurfaceApp { | ||
path: flags, | ||
..SubsurfaceApp::default() | ||
}, | ||
Command::none(), | ||
) | ||
} | ||
|
||
fn title(&self, _id: window::Id) -> String { | ||
String::from("SubsurfaceApp") | ||
} | ||
|
||
fn update(&mut self, message: Self::Message) -> Command<Self::Message> { | ||
match message { | ||
Message::Pipewire(evt) => match evt { | ||
pipewire::Event::Frame(subsurface_buffer) => { | ||
self.buffer = Some(subsurface_buffer); | ||
} | ||
}, | ||
} | ||
Command::none() | ||
} | ||
|
||
fn view(&self, _id: window::Id) -> Element<Self::Message> { | ||
if let Some(buffer) = &self.buffer { | ||
iced_sctk::subsurface_widget::Subsurface::new(1, 1, buffer) | ||
.width(Length::Fill) | ||
.height(Length::Fill) | ||
.into() | ||
} else { | ||
text("No subsurface").into() | ||
} | ||
} | ||
|
||
fn subscription(&self) -> Subscription<Self::Message> { | ||
pipewire::subscription(&self.path).map(Message::Pipewire) | ||
} | ||
} |
Oops, something went wrong.