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 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146
use crate::keyboard::{KeyPosition, KeyState, VirtualKey};
use crate::systems::System;
use async_trait::async_trait;
use std::sync::Arc;
#[cfg(target_arch = "wasm32")]
mod canvas;
#[cfg(not(target_arch = "wasm32"))]
mod text;
#[cfg(not(target_arch = "wasm32"))]
mod winit;
#[cfg(target_arch = "wasm32")]
pub use self::canvas::{CanvasPlatform, CanvasPlatformProvider};
#[cfg(not(target_arch = "wasm32"))]
pub use self::text::{TextPlatform, TextPlatformProvider};
#[cfg(not(target_arch = "wasm32"))]
pub use self::winit::{WinitPlatform, WinitPlatformProvider};
/// A Platform provides platform-specific functionality to the emulator.
/// It handles starting and ticking the system, and provides a PlatformProvider
/// to the system for screen/keyboard/etc. access.
pub trait Platform {
/// Return a reference to a provider for systems to interact with this platform.
fn provider(&self) -> Arc<dyn PlatformProvider>;
}
/// A platform which can be run synchronously.
pub trait SyncPlatform: Platform {
/// Run the given system within this platform.
fn run(&mut self, system: Box<dyn System>);
}
/// A platform which can be run asynchronously.
#[async_trait(?Send)]
pub trait AsyncPlatform: Platform {
/// Set up this platform for use.
async fn setup(&mut self);
/// Execute one "step" of this platform. A step is implementation-defined.
async fn tick(&mut self, system: &mut Box<dyn System>);
}
/// Represents an RGB color with 8 bits per channel.
#[derive(Copy, Clone, Debug, PartialEq, Eq, Hash)]
pub struct Color {
r: u8,
g: u8,
b: u8,
}
impl Color {
pub fn new(r: u8, g: u8, b: u8) -> Self {
Self { r, g, b }
}
/// Convert the given color to an array of 4 bytes, where the last byte
/// (alpha) is always 255.
pub fn to_rgba(&self) -> [u8; 4] {
[self.r, self.g, self.b, 255]
}
/// Convert the given color to a 32-bit integer, where the top 8 bits are
/// unset (0), the next 8 bits are red, the next 8 bits are green, and the
/// last 8 bits are blue.
pub fn to_rgb(&self) -> u32 {
(self.r as u32) << 16 | (self.g as u32) << 8 | self.b as u32
}
}
/// Represents the current state of the connected joystick.
#[derive(Copy, Clone, Debug, PartialEq)]
pub struct JoystickState {
pub up: bool,
pub down: bool,
pub left: bool,
pub right: bool,
pub fire: bool,
}
impl JoystickState {
/// Create a new JoystickState with all buttons released.
pub fn empty() -> Self {
Self {
up: false,
down: false,
left: false,
right: false,
fire: false,
}
}
}
/// Represents the configuration of a GUI window that the system can request
/// from the platform.
#[derive(Copy, Clone, Debug, PartialEq)]
pub struct WindowConfig {
pub width: u32,
pub height: u32,
pub scale: f64,
}
impl WindowConfig {
pub fn new(width: u32, height: u32, scale: f64) -> Self {
Self {
width,
height,
scale,
}
}
}
pub trait PlatformProvider {
/// Request that the platform create a window of the specified size,
/// with the specified scale factor. If a window already exists, the platform
/// should resize it to the new size.
fn request_window(&self, config: WindowConfig);
/// Get the current state of the user's physical keyboard.
fn get_key_state(&self) -> KeyState<KeyPosition>;
/// Get the state of a virtual keyboard (emulating the target system),
/// if one is available.
fn get_virtual_key_state(&self) -> KeyState<VirtualKey>;
/// Get the current state of the connected joystick.
/// If no joystick is connected, this should return a default state.
fn get_joystick_state(&self) -> JoystickState;
/// Display the given string to the user, "out-of-band" from any other
/// graphics. This is used for text-mode systems. Implementations may choose
/// various ways to display this, such as a terminal message or a pop-up.
fn print(&self, text: &str);
/// Read a string input from the user, "out-of-band" from any other
/// graphics. This is used for text-mode systems. Implementations may choose
/// various ways to prompt for this, such as a terminal prompt or a pop-up
/// dialog.
fn input(&self) -> String;
/// Return a random number between 0 and 255. This exists as some platforms
/// (such as the web) have a different source of randomness.
fn random(&self) -> u8;
}