feat(dialog): XDG portal integrations for open and save dialogs

This commit is contained in:
Michael Aaron Murphy 2023-08-15 10:58:46 +02:00 committed by Michael Murphy
parent a5d3814fff
commit 1705b6fe27
10 changed files with 861 additions and 21 deletions

View file

@ -1,6 +1,8 @@
// Copyright 2023 System76 <info@system76.com>
// SPDX-License-Identifier: MPL-2.0
//! Create asynchronous actions to be performed in the background.
#[cfg(feature = "wayland")]
use iced::window;
use iced::Command;
@ -21,7 +23,7 @@ pub fn batch<M>(commands: impl IntoIterator<Item = Command<M>>) -> Command<M> {
Command::batch(commands)
}
/// Yields a command which will run the future on the runtime executor.
/// Yields a command which will run the future on thet runtime executor.
pub fn future<M: Send + 'static>(future: impl Future<Output = M> + Send + 'static) -> Command<M> {
Command::single(Action::Future(Box::pin(future)))
}

9
src/dialog/mod.rs Normal file
View file

@ -0,0 +1,9 @@
// Copyright 2023 System76 <info@system76.com>
// SPDX-License-Identifier: MPL-2.0
//! Create dialogs for retrieving user input.
pub use ashpd::desktop::file_chooser::{Choice, FileFilter, SelectedFiles};
pub use ashpd::WindowIdentifier;
pub mod open_file;

252
src/dialog/open_file.rs Normal file
View file

@ -0,0 +1,252 @@
// Copyright 2023 System76 <info@system76.com>
// SPDX-License-Identifier: MPL-2.0
//! Request to open files and/or directories.
//!
//! Check out the [open-dialog](https://github.com/pop-os/libcosmic/tree/master/examples/open-dialog)
//! example in our repository.
use derive_setters::Setters;
use iced::futures::{channel, SinkExt, StreamExt};
use iced::{Command, Subscription};
use std::cell::Cell;
use thiserror::Error;
thread_local! {
/// Prevents duplicate dialog open requests.
static OPENED: Cell<bool> = Cell::new(false);
}
fn dialog_is_open() -> bool {
OPENED.with(Cell::get)
}
/// Creates a [`Builder`] if no other open file dialog exists.
pub fn builder() -> Option<Builder> {
if dialog_is_open() {
None
} else {
Some(Builder::new())
}
}
/// Creates a subscription for open file dialog events.
pub fn subscription<M: Send + 'static>(handle: fn(Message) -> M) -> Subscription<M> {
let type_id = std::any::TypeId::of::<State<M>>();
iced::subscription::channel(type_id, 1, move |output| async move {
let mut state = State {
active: None,
handle,
output,
};
loop {
let (sender, mut receiver) = channel::mpsc::channel(1);
state.emit(Message::Init(Sender(sender))).await;
while let Some(request) = receiver.next().await {
match request {
Request::Close => state.close().await,
Request::Open(dialog) => {
state.open(dialog).await;
OPENED.with(|last| last.set(false));
}
Request::Response => state.response().await,
}
}
}
})
}
/// Errors that my occur when interacting with an open file dialog subscription
#[derive(Debug, Error)]
pub enum Error {
#[error("dialog close failed")]
Close(#[source] ashpd::Error),
#[error("dialog open failed")]
Open(#[source] ashpd::Error),
#[error("dialog response failed")]
Response(#[source] ashpd::Error),
}
/// Requests for an open file dialog subscription
enum Request {
Close,
Open(Builder),
Response,
}
/// Messages from an open file dialog subscription.
pub enum Message {
Closed,
Err(Error),
Init(Sender),
Opened,
Selected(super::SelectedFiles),
}
/// Sends requests to an open file dialog subscription.
#[derive(Clone, Debug)]
pub struct Sender(channel::mpsc::Sender<Request>);
impl Sender {
/// Creates a [`Command`] that closes an active open file dialog.
pub fn close(&mut self) -> Command<()> {
let mut sender = self.0.clone();
crate::command::future(async move {
let _res = sender.send(Request::Close).await;
()
})
}
/// Creates a [`Command`] that opens a new open file dialog.
pub fn open(&mut self, dialog: Builder) -> Command<()> {
OPENED.with(|opened| opened.set(true));
let mut sender = self.0.clone();
crate::command::future(async move {
let _res = sender.send(Request::Open(dialog)).await;
()
})
}
/// Creates a [`Command`] that requests the response from an active open file dialog.
pub fn response(&mut self) -> Command<()> {
let mut sender = self.0.clone();
crate::command::future(async move {
let _res = sender.send(Request::Response).await;
()
})
}
}
/// A builder for an open file dialog, passed as a request by a [`Sender`]
#[derive(Setters)]
#[must_use]
pub struct Builder {
/// The lab for the dialog's window title.
title: String,
/// The label for the accept button. Mnemonic underlines are allowed.
#[setters(strip_option)]
accept_label: Option<String>,
/// Whether to select for folders instead of files. Default is to select files.
include_directories: bool,
/// Modal dialogs require user input before continuing the program.
modal: bool,
/// Whether to allow selection of multiple files. Default is no.
multiple_files: bool,
/// Adds a list of choices.
choices: Vec<super::Choice>,
/// Specifies the default file filter.
#[setters(into)]
current_filter: Option<super::FileFilter>,
/// A collection of file filters.
filters: Vec<super::FileFilter>,
}
impl Builder {
const fn new() -> Self {
Self {
title: String::new(),
accept_label: None,
include_directories: false,
modal: true,
multiple_files: false,
current_filter: None,
choices: Vec::new(),
filters: Vec::new(),
}
}
/// Creates a [`Command`] which opens the dialog.
pub fn create(self, sender: &mut Sender) -> Command<()> {
sender.open(self)
}
/// Adds a choice.
pub fn choice(mut self, choice: impl Into<super::Choice>) -> Self {
self.choices.push(choice.into());
self
}
/// Adds a files filter.
pub fn filter(mut self, filter: impl Into<super::FileFilter>) -> Self {
self.filters.push(filter.into());
self
}
}
struct State<M> {
active: Option<ashpd::desktop::Request<super::SelectedFiles>>,
handle: fn(Message) -> M,
output: channel::mpsc::Sender<M>,
}
impl<M> State<M> {
/// Emits close request if there is an active dialog request.
async fn close(&mut self) {
if let Some(request) = self.active.take() {
if let Err(why) = request.close().await {
self.emit(Message::Err(Error::Close(why))).await;
}
}
}
async fn emit(&mut self, response: Message) {
let _res = self.output.send((self.handle)(response)).await;
}
/// Creates a new dialog, and closes any prior active dialogs.
async fn open(&mut self, dialog: Builder) {
let response = match create(dialog).await {
Ok(request) => {
self.active = Some(request);
Message::Opened
}
Err(why) => Message::Err(Error::Open(why)),
};
self.emit(response).await;
}
/// Collects selected files from the active dialog.
async fn response(&mut self) {
if let Some(request) = self.active.as_ref() {
let response = match request.response() {
Ok(selected) => Message::Selected(selected),
Err(why) => Message::Err(Error::Response(why)),
};
self.emit(response).await;
}
}
}
/// Creates a new file dialog, and begins to await its responses.
async fn create(dialog: Builder) -> ashpd::Result<ashpd::desktop::Request<super::SelectedFiles>> {
ashpd::desktop::file_chooser::OpenFileRequest::default()
.title(Some(dialog.title.as_str()))
.accept_label(dialog.accept_label.as_deref())
.directory(dialog.include_directories)
.modal(dialog.modal)
.multiple(dialog.multiple_files)
.choices(dialog.choices)
.filters(dialog.filters)
.current_filter(dialog.current_filter)
.send()
.await
}

262
src/dialog/save_file.rs Normal file
View file

@ -0,0 +1,262 @@
// Copyright 2023 System76 <info@system76.com>
// SPDX-License-Identifier: MPL-2.0
//! Choose a location to save a file to.
//!
//! Check out the [open-dialog](https://github.com/pop-os/libcosmic/tree/master/examples/open-dialog)
//! example in our repository.
use derive_setters::Setters;
use iced::{Command, Subscription};
use iced::futures::{channel, SinkExt, StreamExt};
use std::cell::Cell;
use std::path::PathBuf;
use std::time::Instant;
use thiserror::Error;
thread_local! {
/// Prevents duplicate dialog open requests.
static OPENED: Cell<bool> = Cell::new(false);
}
fn dialog_is_open() -> bool {
OPENED.with(Cell::get)
}
/// Creates a [`Builder`] if no other save file dialog exists.
pub fn builder() -> Option<Builder> {
if dialog_is_open() {
None
} else {
Some(Builder::new())
}
}
/// Creates a subscription for save file dialog events.
pub fn subscription<M: Send + 'static>(handle: fn(Message) -> M) -> Subscription<M> {
let type_id = std::any::TypeId::of::<State<M>>();
iced::subscription::channel(type_id, 1, move |output| async move {
let mut state = State {
active: None,
handle,
output,
};
loop {
let (sender, mut receiver) = channel::mpsc::channel(1);
state.emit(Message::Init(Sender(sender))).await;
while let Some(request) = receiver.next().await {
match request {
Request::Close => state.close().await,
Request::Open(dialog) => {
state.open(dialog).await;
OPENED.with(|last| last.set(false));
},
Request::Response => state.response().await,
}
}
}
})
}
/// Errors that my occur when interacting with an save file dialog subscription
#[derive(Debug, Error)]
pub enum Error {
#[error("dialog close failed")]
Close(#[source] ashpd::Error),
#[error("dialog open failed")]
Open(#[source] ashpd::Error),
#[error("dialog response failed")]
Response(#[source] ashpd::Error),
}
/// Requests for an save file dialog subscription
enum Request {
Close,
Open(Builder),
Response,
}
/// Messages from an save file dialog subscription.
pub enum Message {
Closed,
Err(Error),
Init(Sender),
Opened,
Selected(super::SelectedFiles),
}
/// Sends requests to an save file dialog subscription.
#[derive(Clone, Debug)]
pub struct Sender(channel::mpsc::Sender<Request>);
impl Sender {
/// Creates a [`Command`] that closes an active save file dialog.
pub fn close(&mut self) -> Command<()> {
let mut sender = self.0.clone();
crate::command::future(async move {
let _res = sender.send(Request::Close).await;
()
})
}
/// Creates a [`Command`] that opens a new save file dialog.
pub fn open(&mut self, dialog: Builder) -> Command<()> {
OPENED.with(|opened| opened.set(true));
let mut sender = self.0.clone();
crate::command::future(async move {
let _res = sender.send(Request::Open(dialog)).await;
()
})
}
/// Creates a [`Command`] that requests the response from an active save file dialog.
pub fn response(&mut self) -> Command<()> {
let mut sender = self.0.clone();
crate::command::future(async move {
let _res = sender.send(Request::Response).await;
()
})
}
}
/// A builder for an save file dialog, passed as a request by a [`Sender`]
#[derive(Setters)]
#[must_use]
pub struct Builder {
/// The lab for the dialog's window title.
title: String,
/// The label for the accept button. Mnemonic underlines are allowed.
#[setters(strip_option)]
accept_label: Option<String>,
/// Modal dialogs require user input before continuing the program.
modal: bool,
/// Sets the current file name.
#[setters(strip_option)]
current_name: Option<String>,
/// Sets the current folder.
#[setters(strip_option)]
current_folder: Option<PathBuf>,
/// Sets the absolute path of the file
#[setters(strip_option)]
current_file: Option<PathBuf>,
/// Adds a list of choices.
choices: Vec<super::Choice>,
/// Specifies the default file filter.
#[setters(into)]
current_filter: Option<super::FileFilter>,
/// A collection of file filters.
filters: Vec<super::FileFilter>,
}
impl Builder {
const fn new() -> Self {
Self {
title: String::new(),
accept_label: None,
modal: true,
current_name: None,
current_folder: None,
current_file: None,
current_filter: None,
choices: Vec::new(),
filters: Vec::new(),
}
}
/// Creates a [`Command`] which opens the dialog.
pub fn create(self, sender: &mut Sender) -> Command<()> {
sender.open(self)
}
/// Adds a choice.
pub fn choice(mut self, choice: impl Into<super::Choice>) -> Self {
self.choices.push(choice.into());
self
}
/// Adds a files filter.
pub fn filter(mut self, filter: impl Into<super::FileFilter>) -> Self {
self.filters.push(filter.into());
self
}
}
struct State<M> {
active: Option<ashpd::desktop::Request<super::SelectedFiles>>,
handle: fn(Message) -> M,
output: channel::mpsc::Sender<M>,
}
impl<M> State<M> {
/// Emits close request if there is an active dialog request.
async fn close(&mut self) {
if let Some(request) = self.active.take() {
if let Err(why) = request.close().await {
self.emit(Message::Err(Error::Close(why))).await;
}
}
}
async fn emit(&mut self, response: Message) {
let _res = self.output.send((self.handle)(response)).await;
}
/// Creates a new dialog, and closes any prior active dialogs.
async fn open(&mut self, dialog: Builder) {
let response = match create(dialog).await {
Ok(request) => {
self.active = Some(request);
Message::Opened
}
Err(why) => Message::Err(Error::Open(why)),
};
self.emit(response).await;
}
/// Collects selected files from the active dialog.
async fn response(&mut self) {
if let Some(request) = self.active.as_ref() {
let response = match request.response() {
Ok(selected) => Message::Selected(selected),
Err(why) => Message::Err(Error::Message(why)),
};
self.emit(response).await;
}
}
}
/// Creates a new file dialog, and begins to await its responses.
async fn create(dialog: Builder) -> ashpd::Result<ashpd::desktop::Request<super::SelectedFiles>> {
ashpd::desktop::file_chooser::SaveFileRequest::default()
.title(Some(dialog.title.as_str()))
.accept_label(dialog.accept_label.as_deref())
.modal(dialog.modal)
.choices(dialog.choices)
.filters(dialog.filters)
.current_filter(dialog.current_filter)
.current_name(dialog.current_name)
.current_folder(dialog.current_folder)?
.current_file(dialog.current_file)?
.send()
.await
}

View file

@ -10,6 +10,9 @@ pub mod command;
pub use cosmic_config;
pub use cosmic_theme;
#[cfg(feature = "xdg-portal")]
pub mod dialog;
pub mod executor;
#[cfg(feature = "tokio")]
pub use executor::single::Executor as SingleThreadExecutor;