Function slint::spawn_local
source · pub fn spawn_local<F: Future + 'static>(
fut: F,
) -> Result<JoinHandle<F::Output>, EventLoopError>
Expand description
Spawns a Future
to execute in the Slint event loop.
This function is intended to be invoked only from the main Slint thread that runs the event loop.
For spawning a Send
future from a different thread, this function should be called from a closure
passed to invoke_from_event_loop()
.
This function is typically called from a UI callback.
§Example
slint::spawn_local(async move {
// your async code goes here
}).unwrap();
§Compatibility with Tokio and other runtimes
The runtime used to execute the future on the main thread is platform-dependent,
for instance, it could be the winit event loop. Therefore, futures that assume a specific runtime
may not work. This may be an issue if you call .await
on a future created by another
runtime, or pass the future directly to spawn_local
.
Futures from the smol runtime always hand off their work to separate I/O threads that run in parallel to the Slint event loop.
The Tokio runtime is to the following constraints:
- Tokio futures require entering the context of a global Tokio runtime.
- Tokio futures aren’t guaranteed to hand off their work to separate threads and may therefore not complete, because the Slint runtime can’t drive the Tokio runtime.
- Tokio futures require regular yielding to the Tokio runtime for fairness, a constraint that also can’t be met by Slint.
- Tokio’s current-thread schedule cannot be used in Slint main thread, because Slint cannot yield to it.
To address these constraints, use async_compat’s Compat::new() to implicitly allocate a shared, multi-threaded Tokio runtime that will be used for Tokio futures.
The following little example demonstrates the use of Tokio’s TcpStream
to
read from a network socket. The entire future passed to spawn_local()
is wrapped in Compat::new()
to make it run:
// A dummy TCP server that once reports "Hello World"
use std::io::Write;
let listener = std::net::TcpListener::bind("127.0.0.1:0").unwrap();
let local_addr = listener.local_addr().unwrap();
let server = std::thread::spawn(move || {
let mut stream = listener.incoming().next().unwrap().unwrap();
stream.write("Hello World".as_bytes()).unwrap();
});
let slint_future = async move {
use tokio::io::AsyncReadExt;
let mut stream = tokio::net::TcpStream::connect(local_addr).await.unwrap();
let mut data = Vec::new();
stream.read_to_end(&mut data).await.unwrap();
assert_eq!(data, "Hello World".as_bytes());
slint::quit_event_loop().unwrap();
};
// Wrap the future that includes Tokio futures in async_compat's `Compat` to ensure
// presence of a Tokio run-time.
slint::spawn_local(async_compat::Compat::new(slint_future)).unwrap();
slint::run_event_loop_until_quit().unwrap();
server.join().unwrap();
The use of #[tokio::main]
is not recommended. If it’s necessary to use though, wrap the call to enter the Slint
event loop in a call to tokio::task::block_in_place
:
// Wrap the call to run_event_loop to ensure presence of a Tokio run-time.
tokio::task::block_in_place(slint::run_event_loop).unwrap();