feat: add helpers for getting and using activation tokens in applets
refactor(applet): connect to privileged socket if available cleanup
This commit is contained in:
parent
c9554a8740
commit
ef5b6fb44a
9 changed files with 310 additions and 8 deletions
|
|
@ -15,6 +15,8 @@ animated-image = ["image", "dep:async-fs", "tokio?/io-util", "tokio?/fs"]
|
||||||
debug = ["iced/debug"]
|
debug = ["iced/debug"]
|
||||||
# Enables pipewire support in ashpd, if ashpd is enabled
|
# Enables pipewire support in ashpd, if ashpd is enabled
|
||||||
pipewire = ["ashpd?/pipewire"]
|
pipewire = ["ashpd?/pipewire"]
|
||||||
|
# Enables process spawning helper
|
||||||
|
process = ["nix"]
|
||||||
# Enables keycode serialization
|
# Enables keycode serialization
|
||||||
serde-keycode = ["iced_core/serde"]
|
serde-keycode = ["iced_core/serde"]
|
||||||
# smol async runtime
|
# smol async runtime
|
||||||
|
|
@ -27,7 +29,7 @@ wayland = [
|
||||||
"iced_runtime/wayland",
|
"iced_runtime/wayland",
|
||||||
"iced/wayland",
|
"iced/wayland",
|
||||||
"iced_sctk",
|
"iced_sctk",
|
||||||
"sctk",
|
"cctk",
|
||||||
]
|
]
|
||||||
# Render with wgpu
|
# Render with wgpu
|
||||||
wgpu = ["iced/wgpu", "iced_wgpu"]
|
wgpu = ["iced/wgpu", "iced_wgpu"]
|
||||||
|
|
@ -40,6 +42,7 @@ winit_wgpu = ["winit", "wgpu"]
|
||||||
xdg-portal = ["ashpd"]
|
xdg-portal = ["ashpd"]
|
||||||
# XXX Use "a11y"; which is causing a panic currently
|
# XXX Use "a11y"; which is causing a panic currently
|
||||||
applet = ["wayland", "tokio", "cosmic-panel-config", "ron"]
|
applet = ["wayland", "tokio", "cosmic-panel-config", "ron"]
|
||||||
|
applet-token = []
|
||||||
zbus = ["dep:zbus", "serde", "ron"]
|
zbus = ["dep:zbus", "serde", "ron"]
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
|
|
@ -48,7 +51,7 @@ derive_setters = "0.1.5"
|
||||||
lazy_static = "1.4.0"
|
lazy_static = "1.4.0"
|
||||||
palette = "0.7.3"
|
palette = "0.7.3"
|
||||||
tokio = { version = "1.24.2", optional = true }
|
tokio = { version = "1.24.2", optional = true }
|
||||||
sctk = { package = "smithay-client-toolkit", git = "https://github.com/smithay/client-toolkit", optional = true, rev = "dc8c4a0" }
|
cctk = { git = "https://github.com/pop-os/cosmic-protocols", package = "cosmic-client-toolkit", rev = "5faec87", optional = true }
|
||||||
slotmap = "1.0.6"
|
slotmap = "1.0.6"
|
||||||
fraction = "0.13.0"
|
fraction = "0.13.0"
|
||||||
cosmic-config = { path = "cosmic-config" }
|
cosmic-config = { path = "cosmic-config" }
|
||||||
|
|
@ -60,6 +63,7 @@ ashpd = { version = "0.5.0", default-features = false, optional = true }
|
||||||
url = "2.4.0"
|
url = "2.4.0"
|
||||||
unicode-segmentation = "1.6"
|
unicode-segmentation = "1.6"
|
||||||
css-color = "0.2.5"
|
css-color = "0.2.5"
|
||||||
|
nix = { version = "0.26", optional = true }
|
||||||
zbus = {version = "3.14.1", default-features = false, optional = true}
|
zbus = {version = "3.14.1", default-features = false, optional = true}
|
||||||
serde = { version = "1.0.180", optional = true }
|
serde = { version = "1.0.180", optional = true }
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -5,6 +5,8 @@ use super::{command, Application, ApplicationExt, Core, Subscription};
|
||||||
use crate::theme::{self, Theme, ThemeType, THEME};
|
use crate::theme::{self, Theme, ThemeType, THEME};
|
||||||
use crate::widget::nav_bar;
|
use crate::widget::nav_bar;
|
||||||
use crate::{keyboard_nav, Element};
|
use crate::{keyboard_nav, Element};
|
||||||
|
#[cfg(feature = "wayland")]
|
||||||
|
use cctk::sctk::reexports::csd_frame::{WindowManagerCapabilities, WindowState};
|
||||||
use cosmic_theme::ThemeMode;
|
use cosmic_theme::ThemeMode;
|
||||||
#[cfg(feature = "wayland")]
|
#[cfg(feature = "wayland")]
|
||||||
use iced::event::wayland::{self, WindowEvent};
|
use iced::event::wayland::{self, WindowEvent};
|
||||||
|
|
@ -15,8 +17,6 @@ use iced::window;
|
||||||
use iced_runtime::command::Action;
|
use iced_runtime::command::Action;
|
||||||
#[cfg(not(feature = "wayland"))]
|
#[cfg(not(feature = "wayland"))]
|
||||||
use iced_runtime::window::Action as WindowAction;
|
use iced_runtime::window::Action as WindowAction;
|
||||||
#[cfg(feature = "wayland")]
|
|
||||||
use sctk::reexports::csd_frame::{WindowManagerCapabilities, WindowState};
|
|
||||||
|
|
||||||
/// A message managed internally by COSMIC.
|
/// A message managed internally by COSMIC.
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
|
|
|
||||||
|
|
@ -1,12 +1,16 @@
|
||||||
|
#[cfg(feature = "applet-token")]
|
||||||
|
pub mod token;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
app::Core,
|
app::Core,
|
||||||
|
cctk::sctk,
|
||||||
iced::{
|
iced::{
|
||||||
self,
|
self,
|
||||||
alignment::{Horizontal, Vertical},
|
alignment::{Horizontal, Vertical},
|
||||||
widget::Container,
|
widget::Container,
|
||||||
window, Color, Length, Limits, Rectangle,
|
window, Color, Length, Limits, Rectangle,
|
||||||
},
|
},
|
||||||
iced_style, iced_widget, sctk,
|
iced_style, iced_widget,
|
||||||
theme::{self, Button, THEME},
|
theme::{self, Button, THEME},
|
||||||
widget, Application, Element, Renderer,
|
widget, Application, Element, Renderer,
|
||||||
};
|
};
|
||||||
|
|
|
||||||
2
src/applet/token/mod.rs
Normal file
2
src/applet/token/mod.rs
Normal file
|
|
@ -0,0 +1,2 @@
|
||||||
|
pub mod subscription;
|
||||||
|
pub mod wayland_handler;
|
||||||
78
src/applet/token/subscription.rs
Normal file
78
src/applet/token/subscription.rs
Normal file
|
|
@ -0,0 +1,78 @@
|
||||||
|
use crate::iced;
|
||||||
|
use crate::iced::subscription;
|
||||||
|
use crate::iced_futures::futures;
|
||||||
|
use cctk::sctk::reexports::calloop;
|
||||||
|
use futures::{
|
||||||
|
channel::mpsc::{unbounded, UnboundedReceiver},
|
||||||
|
SinkExt, StreamExt,
|
||||||
|
};
|
||||||
|
use std::{fmt::Debug, hash::Hash, thread::JoinHandle};
|
||||||
|
|
||||||
|
use super::wayland_handler::wayland_handler;
|
||||||
|
|
||||||
|
pub fn activation_token_subscription<I: 'static + Hash + Copy + Send + Sync + Debug>(
|
||||||
|
id: I,
|
||||||
|
) -> iced::Subscription<TokenUpdate> {
|
||||||
|
subscription::channel(id, 50, move |mut output| async move {
|
||||||
|
let mut state = State::Ready;
|
||||||
|
|
||||||
|
loop {
|
||||||
|
state = start_listening(state, &mut output).await;
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
pub enum State {
|
||||||
|
Ready,
|
||||||
|
Waiting(
|
||||||
|
UnboundedReceiver<TokenUpdate>,
|
||||||
|
calloop::channel::Sender<TokenRequest>,
|
||||||
|
JoinHandle<()>,
|
||||||
|
),
|
||||||
|
Finished,
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn start_listening(
|
||||||
|
state: State,
|
||||||
|
output: &mut futures::channel::mpsc::Sender<TokenUpdate>,
|
||||||
|
) -> State {
|
||||||
|
match state {
|
||||||
|
State::Ready => {
|
||||||
|
let (calloop_tx, calloop_rx) = calloop::channel::channel();
|
||||||
|
let (toplevel_tx, toplevel_rx) = unbounded();
|
||||||
|
let handle = std::thread::spawn(move || {
|
||||||
|
wayland_handler(toplevel_tx, calloop_rx);
|
||||||
|
});
|
||||||
|
let tx = calloop_tx.clone();
|
||||||
|
_ = output.send(TokenUpdate::Init(tx)).await;
|
||||||
|
State::Waiting(toplevel_rx, calloop_tx, handle)
|
||||||
|
}
|
||||||
|
State::Waiting(mut rx, tx, handle) => {
|
||||||
|
if handle.is_finished() {
|
||||||
|
_ = output.send(TokenUpdate::Finished).await;
|
||||||
|
return State::Finished;
|
||||||
|
}
|
||||||
|
if let Some(u) = rx.next().await {
|
||||||
|
_ = output.send(u).await;
|
||||||
|
State::Waiting(rx, tx, handle)
|
||||||
|
} else {
|
||||||
|
_ = output.send(TokenUpdate::Finished).await;
|
||||||
|
State::Finished
|
||||||
|
}
|
||||||
|
}
|
||||||
|
State::Finished => iced::futures::future::pending().await,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
pub enum TokenUpdate {
|
||||||
|
Init(calloop::channel::Sender<TokenRequest>),
|
||||||
|
Finished,
|
||||||
|
ActivationToken { token: Option<String>, exec: String },
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Debug)]
|
||||||
|
pub struct TokenRequest {
|
||||||
|
pub app_id: String,
|
||||||
|
pub exec: String,
|
||||||
|
}
|
||||||
180
src/applet/token/wayland_handler.rs
Normal file
180
src/applet/token/wayland_handler.rs
Normal file
|
|
@ -0,0 +1,180 @@
|
||||||
|
use std::os::{
|
||||||
|
fd::{FromRawFd, RawFd},
|
||||||
|
unix::net::UnixStream,
|
||||||
|
};
|
||||||
|
|
||||||
|
use super::subscription::{TokenRequest, TokenUpdate};
|
||||||
|
use cctk::{
|
||||||
|
sctk::{
|
||||||
|
self,
|
||||||
|
activation::{RequestData, RequestDataExt},
|
||||||
|
reexports::{calloop, calloop_wayland_source::WaylandSource},
|
||||||
|
seat::{SeatHandler, SeatState},
|
||||||
|
},
|
||||||
|
wayland_client::{
|
||||||
|
self,
|
||||||
|
protocol::{wl_seat::WlSeat, wl_surface::WlSurface},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
use iced_futures::futures::channel::mpsc::UnboundedSender;
|
||||||
|
use sctk::{
|
||||||
|
activation::{ActivationHandler, ActivationState},
|
||||||
|
registry::{ProvidesRegistryState, RegistryState},
|
||||||
|
};
|
||||||
|
use wayland_client::{globals::registry_queue_init, Connection, QueueHandle};
|
||||||
|
|
||||||
|
struct AppData {
|
||||||
|
exit: bool,
|
||||||
|
queue_handle: QueueHandle<Self>,
|
||||||
|
registry_state: RegistryState,
|
||||||
|
activation_state: Option<ActivationState>,
|
||||||
|
tx: UnboundedSender<TokenUpdate>,
|
||||||
|
seat_state: SeatState,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ProvidesRegistryState for AppData {
|
||||||
|
fn registry(&mut self) -> &mut RegistryState {
|
||||||
|
&mut self.registry_state
|
||||||
|
}
|
||||||
|
|
||||||
|
sctk::registry_handlers!();
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ExecRequestData {
|
||||||
|
data: RequestData,
|
||||||
|
exec: String,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl RequestDataExt for ExecRequestData {
|
||||||
|
fn app_id(&self) -> Option<&str> {
|
||||||
|
self.data.app_id()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn seat_and_serial(&self) -> Option<(&WlSeat, u32)> {
|
||||||
|
self.data.seat_and_serial()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn surface(&self) -> Option<&WlSurface> {
|
||||||
|
self.data.surface()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ActivationHandler for AppData {
|
||||||
|
type RequestData = ExecRequestData;
|
||||||
|
fn new_token(&mut self, token: String, data: &ExecRequestData) {
|
||||||
|
let _ = self.tx.unbounded_send(TokenUpdate::ActivationToken {
|
||||||
|
token: Some(token),
|
||||||
|
exec: data.exec.clone(),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SeatHandler for AppData {
|
||||||
|
fn seat_state(&mut self) -> &mut sctk::seat::SeatState {
|
||||||
|
&mut self.seat_state
|
||||||
|
}
|
||||||
|
|
||||||
|
fn new_seat(&mut self, _: &Connection, _: &QueueHandle<Self>, _: WlSeat) {}
|
||||||
|
|
||||||
|
fn new_capability(
|
||||||
|
&mut self,
|
||||||
|
_: &Connection,
|
||||||
|
_: &QueueHandle<Self>,
|
||||||
|
_: WlSeat,
|
||||||
|
_: sctk::seat::Capability,
|
||||||
|
) {
|
||||||
|
}
|
||||||
|
|
||||||
|
fn remove_capability(
|
||||||
|
&mut self,
|
||||||
|
_: &Connection,
|
||||||
|
_: &QueueHandle<Self>,
|
||||||
|
_: WlSeat,
|
||||||
|
_: sctk::seat::Capability,
|
||||||
|
) {
|
||||||
|
}
|
||||||
|
|
||||||
|
fn remove_seat(&mut self, _: &Connection, _: &QueueHandle<Self>, _: WlSeat) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn wayland_handler(
|
||||||
|
tx: UnboundedSender<TokenUpdate>,
|
||||||
|
rx: calloop::channel::Channel<TokenRequest>,
|
||||||
|
) {
|
||||||
|
let socket = std::env::var("X_PRIVILEGED_WAYLAND_SOCKET")
|
||||||
|
.ok()
|
||||||
|
.and_then(|fd| {
|
||||||
|
fd.parse::<RawFd>()
|
||||||
|
.ok()
|
||||||
|
.map(|fd| unsafe { UnixStream::from_raw_fd(fd) })
|
||||||
|
});
|
||||||
|
|
||||||
|
let conn = if let Some(socket) = socket {
|
||||||
|
Connection::from_socket(socket).unwrap()
|
||||||
|
} else {
|
||||||
|
Connection::connect_to_env().unwrap()
|
||||||
|
};
|
||||||
|
let (globals, event_queue) = registry_queue_init(&conn).unwrap();
|
||||||
|
|
||||||
|
let mut event_loop = calloop::EventLoop::<AppData>::try_new().unwrap();
|
||||||
|
let qh = event_queue.handle();
|
||||||
|
let wayland_source = WaylandSource::new(conn, event_queue);
|
||||||
|
let handle = event_loop.handle();
|
||||||
|
wayland_source
|
||||||
|
.insert(handle.clone())
|
||||||
|
.expect("Failed to insert wayland source.");
|
||||||
|
|
||||||
|
if handle
|
||||||
|
.insert_source(rx, |event, _, state| match event {
|
||||||
|
calloop::channel::Event::Msg(TokenRequest { app_id, exec }) => {
|
||||||
|
if let Some(activation_state) = state.activation_state.as_ref() {
|
||||||
|
activation_state.request_token_with_data(
|
||||||
|
&state.queue_handle,
|
||||||
|
ExecRequestData {
|
||||||
|
data: RequestData {
|
||||||
|
app_id: Some(app_id),
|
||||||
|
seat_and_serial: state
|
||||||
|
.seat_state
|
||||||
|
.seats()
|
||||||
|
.next()
|
||||||
|
.map(|seat| (seat, 0)),
|
||||||
|
surface: None,
|
||||||
|
},
|
||||||
|
exec,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
let _ = state
|
||||||
|
.tx
|
||||||
|
.unbounded_send(TokenUpdate::ActivationToken { token: None, exec });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
calloop::channel::Event::Closed => {
|
||||||
|
state.exit = true;
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.is_err()
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let registry_state = RegistryState::new(&globals);
|
||||||
|
let mut app_data = AppData {
|
||||||
|
exit: false,
|
||||||
|
tx,
|
||||||
|
seat_state: SeatState::new(&globals, &qh),
|
||||||
|
queue_handle: qh.clone(),
|
||||||
|
activation_state: ActivationState::bind::<AppData>(&globals, &qh).ok(),
|
||||||
|
registry_state,
|
||||||
|
};
|
||||||
|
|
||||||
|
loop {
|
||||||
|
if app_data.exit {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
event_loop.dispatch(None, &mut app_data).unwrap();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sctk::delegate_activation!(AppData, ExecRequestData);
|
||||||
|
sctk::delegate_seat!(AppData);
|
||||||
|
sctk::delegate_registry!(AppData);
|
||||||
|
|
@ -55,8 +55,11 @@ pub use iced_winit;
|
||||||
pub mod icon_theme;
|
pub mod icon_theme;
|
||||||
pub mod keyboard_nav;
|
pub mod keyboard_nav;
|
||||||
|
|
||||||
|
#[cfg(feature = "process")]
|
||||||
|
pub mod process;
|
||||||
|
|
||||||
#[cfg(feature = "wayland")]
|
#[cfg(feature = "wayland")]
|
||||||
pub use sctk;
|
pub use cctk;
|
||||||
|
|
||||||
pub mod theme;
|
pub mod theme;
|
||||||
pub use theme::{style, Theme};
|
pub use theme::{style, Theme};
|
||||||
|
|
|
||||||
31
src/process.rs
Normal file
31
src/process.rs
Normal file
|
|
@ -0,0 +1,31 @@
|
||||||
|
use std::process::{exit, Command, Stdio};
|
||||||
|
|
||||||
|
use nix::sys::wait::waitpid;
|
||||||
|
use nix::unistd::{fork, ForkResult};
|
||||||
|
|
||||||
|
/// Performs a double fork with setsid to spawn and detach a command.
|
||||||
|
pub fn spawn(mut command: Command) {
|
||||||
|
command
|
||||||
|
.stdin(Stdio::null())
|
||||||
|
.stdout(Stdio::null())
|
||||||
|
.stderr(Stdio::null());
|
||||||
|
|
||||||
|
unsafe {
|
||||||
|
match fork() {
|
||||||
|
Ok(ForkResult::Parent { child }) => {
|
||||||
|
let _res = waitpid(Some(child), None);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(ForkResult::Child) => {
|
||||||
|
let _res = nix::unistd::setsid();
|
||||||
|
let _res = command.spawn();
|
||||||
|
|
||||||
|
exit(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
Err(why) => {
|
||||||
|
println!("failed to fork and spawn command: {}", why.desc());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -41,9 +41,9 @@ use iced_runtime::command::platform_specific;
|
||||||
use iced_runtime::Command;
|
use iced_runtime::Command;
|
||||||
|
|
||||||
#[cfg(feature = "wayland")]
|
#[cfg(feature = "wayland")]
|
||||||
use iced_runtime::command::platform_specific::wayland::data_device::{DataFromMimeType, DndIcon};
|
use cctk::sctk::reexports::client::protocol::wl_data_device_manager::DndAction;
|
||||||
#[cfg(feature = "wayland")]
|
#[cfg(feature = "wayland")]
|
||||||
use sctk::reexports::client::protocol::wl_data_device_manager::DndAction;
|
use iced_runtime::command::platform_specific::wayland::data_device::{DataFromMimeType, DndIcon};
|
||||||
|
|
||||||
/// Creates a new [`TextInput`].
|
/// Creates a new [`TextInput`].
|
||||||
///
|
///
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue