1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78
/* LICENSE BEGIN
This file is part of the SixtyFPS Project -- https://sixtyfps.io
Copyright (c) 2021 Olivier Goffart <olivier.goffart@sixtyfps.io>
Copyright (c) 2021 Simon Hausmann <simon.hausmann@sixtyfps.io>
SPDX-License-Identifier: GPL-3.0-only
This file is also available under commercial licensing terms.
Please contact info@sixtyfps.io for more information.
LICENSE END */
/*!
The backend is the abstraction for crates that need to do the actual drawing and event loop
*/
use std::path::Path;
use std::rc::Rc;
use crate::graphics::{Image, Size};
use crate::window::Window;
/// Behavior describing how the event loop should terminate.
pub enum EventLoopQuitBehavior {
/// Terminate the event loop when the last window was closed.
QuitOnLastWindowClosed,
/// Keep the event loop running until [`Backend::quit_event_loop()`] is called.
QuitOnlyExplicitly,
}
/// Interface implemented by back-ends
pub trait Backend: Send + Sync {
/// Instantiate a window for a component.
/// FIXME: should return a Box<dyn PlatformWindow>
fn create_window(&'static self) -> Rc<Window>;
/// Spins an event loop and renders the visible windows.
fn run_event_loop(&'static self, behavior: EventLoopQuitBehavior);
/// Exits the event loop.
fn quit_event_loop(&'static self);
/// This function can be used to register a custom TrueType font with SixtyFPS,
/// for use with the `font-family` property. The provided slice must be a valid TrueType
/// font.
fn register_font_from_memory(
&'static self,
data: &[u8],
) -> Result<(), Box<dyn std::error::Error>>;
/// This function can be used to register a custom TrueType font with SixtyFPS,
/// for use with the `font-family` property. The provided path must refer to a valid TrueType
/// font.
fn register_font_from_path(
&'static self,
path: &Path,
) -> Result<(), Box<dyn std::error::Error>>;
fn set_clipboard_text(&'static self, text: String);
fn clipboard_text(&'static self) -> Option<String>;
/// Send an user event to from another thread that should be run in the GUI event loop
fn post_event(&'static self, event: Box<dyn FnOnce() + Send>);
fn image_size(&'static self, image: &Image) -> Size;
}
static PRIVATE_BACKEND_INSTANCE: once_cell::sync::OnceCell<Box<dyn Backend + 'static>> =
once_cell::sync::OnceCell::new();
pub fn instance() -> Option<&'static dyn Backend> {
use std::ops::Deref;
PRIVATE_BACKEND_INSTANCE.get().map(|backend_box| backend_box.deref())
}
pub fn instance_or_init(
factory_fn: impl FnOnce() -> Box<dyn Backend + 'static>,
) -> &'static dyn Backend {
use std::ops::Deref;
PRIVATE_BACKEND_INSTANCE.get_or_init(factory_fn).deref()
}