Persist window state across runs

This commit is contained in:
Simon Gellis 2026-04-04 23:05:53 -04:00
parent f7b37caccb
commit e75bef634b
No known key found for this signature in database
GPG Key ID: DA576912FED9577B
10 changed files with 342 additions and 155 deletions

21
Cargo.lock generated
View File

@ -1010,6 +1010,7 @@ dependencies = [
"log", "log",
"nohash-hasher", "nohash-hasher",
"profiling", "profiling",
"ron",
"serde", "serde",
"smallvec", "smallvec",
"unicode-segmentation", "unicode-segmentation",
@ -3766,6 +3767,20 @@ dependencies = [
"windows-sys 0.52.0", "windows-sys 0.52.0",
] ]
[[package]]
name = "ron"
version = "0.12.1"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "4147b952f3f819eca0e99527022f7d6a8d05f111aeb0a62960c74eb283bec8fc"
dependencies = [
"bitflags 2.11.0",
"once_cell",
"serde",
"serde_derive",
"typeid",
"unicode-ident",
]
[[package]] [[package]]
name = "rtrb" name = "rtrb"
version = "0.3.3" version = "0.3.3"
@ -4753,6 +4768,12 @@ dependencies = [
"rustc-hash 2.1.2", "rustc-hash 2.1.2",
] ]
[[package]]
name = "typeid"
version = "1.0.3"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "bc7d623258602320d5c55d1bc22793b57daff0ec7efc270ea7d55ce1d5f5471c"
[[package]] [[package]]
name = "typenum" name = "typenum"
version = "1.19.0" version = "1.19.0"

View File

@ -17,7 +17,7 @@ bytemuck = { version = "1", features = ["derive"] }
clap = { version = "4", features = ["derive"] } clap = { version = "4", features = ["derive"] }
cpal = { git = "https://github.com/sidit77/cpal.git", rev = "66ed6be" } cpal = { git = "https://github.com/sidit77/cpal.git", rev = "66ed6be" }
directories = "6" directories = "6"
egui = { version = "0.34", features = ["serde"] } egui = { version = "0.34", features = ["persistence", "serde"] }
egui_extras = { version = "0.34", features = ["image"] } egui_extras = { version = "0.34", features = ["image"] }
egui-notify = "0.22" egui-notify = "0.22"
egui-winit = "0.34" egui-winit = "0.34"

View File

@ -33,6 +33,8 @@ use crate::{
window::{AppWindow, ChildWindow, GameScreen, GameWindow, InitArgs}, window::{AppWindow, ChildWindow, GameScreen, GameWindow, InitArgs},
}; };
const EGUI_FILENAME: &str = "egui";
fn load_icon() -> anyhow::Result<IconData> { fn load_icon() -> anyhow::Result<IconData> {
let bytes = include_bytes!("../assets/lemur-256x256.png"); let bytes = include_bytes!("../assets/lemur-256x256.png");
let img = image::load_from_memory_with_format(bytes, image::ImageFormat::Png)?; let img = image::load_from_memory_with_format(bytes, image::ImageFormat::Png)?;
@ -52,6 +54,7 @@ struct SharedViewportState {
pub struct Application { pub struct Application {
client: EmulatorClient, client: EmulatorClient,
persistence: Persistence,
ctx: Context, ctx: Context,
shared: SharedViewportState, shared: SharedViewportState,
icon: Option<Arc<IconData>>, icon: Option<Arc<IconData>>,
@ -95,6 +98,8 @@ impl Application {
); );
let ctx = Context::default(); let ctx = Context::default();
let data = persistence.load_config(EGUI_FILENAME).unwrap_or_default();
ctx.data_mut(|d| *d = data);
let mut fonts = FontDefinitions::default(); let mut fonts = FontDefinitions::default();
fonts.font_data.insert( fonts.font_data.insert(
"Selawik".into(), "Selawik".into(),
@ -175,6 +180,7 @@ impl Application {
Self { Self {
client, client,
persistence,
ctx, ctx,
shared: SharedViewportState { shared: SharedViewportState {
viewport_info: ViewportIdMap::default(), viewport_info: ViewportIdMap::default(),
@ -454,6 +460,12 @@ impl ApplicationHandler<UserEvent> for Application {
} }
fn exiting(&mut self, _event_loop: &ActiveEventLoop) { fn exiting(&mut self, _event_loop: &ActiveEventLoop) {
if let Err(error) = self
.ctx
.data(|d| self.persistence.save_config(EGUI_FILENAME, d))
{
error!(%error, "could not save egui state.");
}
let (sender, receiver) = oneshot::channel(); let (sender, receiver) = oneshot::channel();
if self.client.send_command(EmulatorCommand::Exit(sender)) if self.client.send_command(EmulatorCommand::Exit(sender))
&& let Err(error) = receiver.recv_timeout(Duration::from_secs(5)) && let Err(error) = receiver.recv_timeout(Duration::from_secs(5))

View File

@ -1,5 +1,4 @@
use std::{ use std::{
collections::HashMap,
ops::{Deref, DerefMut}, ops::{Deref, DerefMut},
sync::{Arc, Mutex, atomic::AtomicBool, mpsc}, sync::{Arc, Mutex, atomic::AtomicBool, mpsc},
time::Duration, time::Duration,
@ -22,7 +21,7 @@ use crate::{
use anyhow::Context as _; use anyhow::Context as _;
use egui::{ use egui::{
Align2, Button, CentralPanel, Color32, Context, Frame, MenuBar, Panel, Ui, Vec2, Align2, Button, CentralPanel, Color32, Context, Frame, MenuBar, Panel, Ui, Vec2,
ViewportBuilder, ViewportCommand, ViewportId, Window, ViewportBuilder, ViewportCommand, ViewportId, ViewportIdMap, Window,
}; };
use egui_notify::{Anchor, Toast, Toasts}; use egui_notify::{Anchor, Toast, Toasts};
use winit::{event::KeyEvent, event_loop::EventLoopProxy}; use winit::{event::KeyEvent, event_loop::EventLoopProxy};
@ -50,7 +49,7 @@ pub struct GameWindow {
memory: Arc<MemoryClient>, memory: Arc<MemoryClient>,
images: Arc<ImageTextureLoader>, images: Arc<ImageTextureLoader>,
mappings: MappingProvider, mappings: MappingProvider,
children: HashMap<ViewportId, ChildWindowWrapper>, children: ViewportIdMap<ChildWindowWrapper>,
} }
impl GameWindow { impl GameWindow {
@ -89,7 +88,7 @@ impl GameWindow {
memory: memory.clone(), memory: memory.clone(),
images: images.clone(), images: images.clone(),
mappings: mappings.clone(), mappings: mappings.clone(),
children: HashMap::new(), children: ViewportIdMap::default(),
} }
} }

View File

@ -1,6 +1,6 @@
use std::{ use std::{
fmt::{Display, UpperHex}, fmt::{Display, UpperHex},
ops::{Bound, RangeBounds}, ops::{Bound, Deref, DerefMut, RangeBounds},
str::FromStr, str::FromStr,
}; };
@ -8,7 +8,7 @@ use atoi::FromRadix16;
use egui::{ use egui::{
Align, Color32, CornerRadius, CursorIcon, Event, Frame, Key, Layout, Margin, Rect, Response, Align, Color32, CornerRadius, CursorIcon, Event, Frame, Key, Layout, Margin, Rect, Response,
RichText, Sense, Shape, Stroke, StrokeKind, TextEdit, Ui, UiBuilder, Vec2, Widget, WidgetText, RichText, Sense, Shape, Stroke, StrokeKind, TextEdit, Ui, UiBuilder, Vec2, Widget, WidgetText,
ecolor::HexColor, ecolor::HexColor, util::id_type_map::SerializableAny,
}; };
use num_traits::{CheckedAdd, CheckedSub, One}; use num_traits::{CheckedAdd, CheckedSub, One};
@ -409,3 +409,51 @@ impl ResponseExt for Response {
self.clicked() || self.dragged() self.clicked() || self.dragged()
} }
} }
pub struct UiData<T> {
current: T,
prev: T,
loaded: bool,
}
impl<T> UiData<T>
where
T: Default + PartialEq + SerializableAny,
{
pub fn new() -> Self {
Self {
current: T::default(),
prev: T::default(),
loaded: false,
}
}
pub fn load(&mut self, ui: &Ui) {
if !self.loaded {
self.current = ui
.data_mut(|d| d.get_persisted(ui.id()))
.unwrap_or_default();
self.loaded = true;
}
}
pub fn save(&mut self, ui: &Ui) {
if self.current != self.prev {
ui.data_mut(|d| d.insert_persisted(ui.id(), self.current.clone()));
self.prev = self.current.clone();
}
}
}
impl<T> Deref for UiData<T> {
type Target = T;
fn deref(&self) -> &Self::Target {
&self.current
}
}
impl<T> DerefMut for UiData<T> {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.current
}
}

View File

@ -5,6 +5,7 @@ use egui::{
TextureOptions, Ui, ViewportBuilder, TextureOptions, Ui, ViewportBuilder,
}; };
use egui_extras::{Column, Size, StripBuilder, TableBuilder}; use egui_extras::{Column, Size, StripBuilder, TableBuilder};
use serde::{Deserialize, Serialize};
use crate::{ use crate::{
emulator::SimId, emulator::SimId,
@ -12,36 +13,56 @@ use crate::{
memory::{MemoryClient, MemoryView}, memory::{MemoryClient, MemoryView},
window::{ window::{
AppWindow, AppWindow,
utils::{NumberEdit, UiExt}, utils::{NumberEdit, UiData, UiExt},
}, },
}; };
use super::utils::{self, CellData, CharacterGrid}; use super::utils::{self, CellData, CharacterGrid};
#[derive(Clone, PartialEq, Serialize, Deserialize)]
struct State {
cell_index: usize,
generic_palette: bool,
scale: f32,
show_grid: bool,
}
impl Default for State {
fn default() -> Self {
Self {
cell_index: 0,
generic_palette: false,
scale: 1.0,
show_grid: false,
}
}
}
pub struct BgMapWindow { pub struct BgMapWindow {
sim_id: SimId, sim_id: SimId,
memory: Arc<MemoryClient>, memory: Arc<MemoryClient>,
bgmaps: MemoryView, bgmaps: MemoryView,
cell_index: usize,
generic_palette: bool,
params: ImageParams<BgMapParams>, params: ImageParams<BgMapParams>,
scale: f32, state: UiData<State>,
show_grid: bool,
} }
impl BgMapWindow { impl BgMapWindow {
pub fn new(sim_id: SimId, memory: &Arc<MemoryClient>, images: &ImageTextureLoader) -> Self { pub fn new(sim_id: SimId, memory: &Arc<MemoryClient>, images: &ImageTextureLoader) -> Self {
let state: UiData<State> = UiData::new();
let renderer = BgMapRenderer::new(sim_id, memory); let renderer = BgMapRenderer::new(sim_id, memory);
let params = images.add(sim_id, renderer, BgMapParams::default()); let params = images.add(
sim_id,
renderer,
BgMapParams {
cell_index: state.cell_index,
generic_palette: state.generic_palette,
},
);
Self { Self {
sim_id, sim_id,
memory: memory.clone(), memory: memory.clone(),
bgmaps: memory.watch(sim_id, 0x00020000, 0x20000), bgmaps: memory.watch(sim_id, 0x00020000, 0x20000),
cell_index: params.cell_index,
generic_palette: params.generic_palette,
params, params,
scale: 1.0, state,
show_grid: false,
} }
} }
@ -57,10 +78,11 @@ impl BgMapWindow {
ui.label("Map"); ui.label("Map");
}); });
row.col(|ui| { row.col(|ui| {
let mut bgmap_index = self.cell_index / 4096; let mut bgmap_index = self.state.cell_index / 4096;
ui.add(NumberEdit::new(&mut bgmap_index).range(0..16)); ui.add(NumberEdit::new(&mut bgmap_index).range(0..16));
if bgmap_index != self.cell_index / 4096 { if bgmap_index != self.state.cell_index / 4096 {
self.cell_index = (bgmap_index * 4096) + (self.cell_index % 4096); self.state.cell_index =
(bgmap_index * 4096) + (self.state.cell_index % 4096);
} }
}); });
}); });
@ -69,7 +91,7 @@ impl BgMapWindow {
ui.label("Cell"); ui.label("Cell");
}); });
row.col(|ui| { row.col(|ui| {
ui.add(NumberEdit::new(&mut self.cell_index).range(0..16 * 4096)); ui.add(NumberEdit::new(&mut self.state.cell_index).range(0..16 * 4096));
}); });
}); });
body.row(row_height, |mut row| { body.row(row_height, |mut row| {
@ -77,7 +99,7 @@ impl BgMapWindow {
ui.label("Address"); ui.label("Address");
}); });
row.col(|ui| { row.col(|ui| {
let address = 0x00020000 + (self.cell_index * 2); let address = 0x00020000 + (self.state.cell_index * 2);
let mut address_str = format!("{address:08x}"); let mut address_str = format!("{address:08x}");
ui.add_enabled( ui.add_enabled(
false, false,
@ -91,7 +113,7 @@ impl BgMapWindow {
.texture_options(TextureOptions::NEAREST); .texture_options(TextureOptions::NEAREST);
ui.add(image); ui.add(image);
ui.section("Cell", |ui| { ui.section("Cell", |ui| {
let mut data = self.bgmaps.borrow().read::<u16>(self.cell_index); let mut data = self.bgmaps.borrow().read::<u16>(self.state.cell_index);
let mut cell = CellData::parse(data); let mut cell = CellData::parse(data);
TableBuilder::new(ui) TableBuilder::new(ui)
.column(Column::remainder()) .column(Column::remainder())
@ -134,7 +156,7 @@ impl BgMapWindow {
}); });
}); });
if cell.update(&mut data) { if cell.update(&mut data) {
let address = 0x00020000 + (self.cell_index * 2); let address = 0x00020000 + (self.state.cell_index * 2);
self.memory.write(self.sim_id, address as u32, &data); self.memory.write(self.sim_id, address as u32, &data);
} }
}); });
@ -142,28 +164,28 @@ impl BgMapWindow {
ui.horizontal(|ui| { ui.horizontal(|ui| {
ui.label("Scale"); ui.label("Scale");
ui.spacing_mut().slider_width = ui.available_width(); ui.spacing_mut().slider_width = ui.available_width();
let slider = Slider::new(&mut self.scale, 1.0..=10.0) let slider = Slider::new(&mut self.state.scale, 1.0..=10.0)
.step_by(1.0) .step_by(1.0)
.show_value(false); .show_value(false);
ui.add(slider); ui.add(slider);
}); });
ui.checkbox(&mut self.show_grid, "Show grid"); ui.checkbox(&mut self.state.show_grid, "Show grid");
ui.checkbox(&mut self.generic_palette, "Generic palette"); ui.checkbox(&mut self.state.generic_palette, "Generic palette");
}); });
}); });
self.params.write(BgMapParams { self.params.write(BgMapParams {
cell_index: self.cell_index, cell_index: self.state.cell_index,
generic_palette: self.generic_palette, generic_palette: self.state.generic_palette,
}); });
} }
fn show_bgmap(&mut self, ui: &mut Ui) { fn show_bgmap(&mut self, ui: &mut Ui) {
let grid = CharacterGrid::new(self.image_url("bgmap")) let grid = CharacterGrid::new(self.image_url("bgmap"))
.with_scale(self.scale) .with_scale(self.state.scale)
.with_grid(self.show_grid) .with_grid(self.state.show_grid)
.with_selected(self.cell_index % 4096); .with_selected(self.state.cell_index % 4096);
if let Some(selected) = grid.show(ui) { if let Some(selected) = grid.show(ui) {
self.cell_index = (self.cell_index / 4096 * 4096) + selected; self.state.cell_index = (self.state.cell_index / 4096 * 4096) + selected;
} }
} }
} }
@ -180,6 +202,7 @@ impl AppWindow for BgMapWindow {
} }
fn show(&mut self, ui: &mut Ui) { fn show(&mut self, ui: &mut Ui) {
self.state.load(ui);
CentralPanel::default().show_inside(ui, |ui| { CentralPanel::default().show_inside(ui, |ui| {
ui.horizontal_top(|ui| { ui.horizontal_top(|ui| {
StripBuilder::new(ui) StripBuilder::new(ui)
@ -195,6 +218,7 @@ impl AppWindow for BgMapWindow {
}) })
}); });
}); });
self.state.save(ui);
} }
} }

View File

@ -13,7 +13,7 @@ use crate::{
memory::{MemoryClient, MemoryView}, memory::{MemoryClient, MemoryView},
window::{ window::{
AppWindow, AppWindow,
utils::{NumberEdit, UiExt as _}, utils::{NumberEdit, UiData, UiExt as _},
}, },
}; };
@ -79,30 +79,50 @@ impl Display for Palette {
} }
} }
#[derive(Clone, PartialEq, Serialize, Deserialize)]
struct State {
palette: Palette,
index: usize,
scale: f32,
show_grid: bool,
}
impl Default for State {
fn default() -> Self {
Self {
palette: Palette::default(),
index: 0,
scale: 4.0,
show_grid: true,
}
}
}
pub struct CharacterDataWindow { pub struct CharacterDataWindow {
sim_id: SimId, sim_id: SimId,
brightness: MemoryView, brightness: MemoryView,
palettes: MemoryView, palettes: MemoryView,
palette: Palette,
index: usize,
params: ImageParams<CharDataParams>, params: ImageParams<CharDataParams>,
scale: f32, state: UiData<State>,
show_grid: bool,
} }
impl CharacterDataWindow { impl CharacterDataWindow {
pub fn new(sim_id: SimId, memory: &MemoryClient, images: &ImageTextureLoader) -> Self { pub fn new(sim_id: SimId, memory: &MemoryClient, images: &ImageTextureLoader) -> Self {
let state: UiData<State> = UiData::new();
let renderer = CharDataRenderer::new(sim_id, memory); let renderer = CharDataRenderer::new(sim_id, memory);
let params = images.add(sim_id, renderer, CharDataParams::default()); let params = images.add(
sim_id,
renderer,
CharDataParams {
palette: state.palette,
index: state.index,
},
);
Self { Self {
sim_id, sim_id,
brightness: memory.watch(sim_id, 0x0005f824, 8), brightness: memory.watch(sim_id, 0x0005f824, 8),
palettes: memory.watch(sim_id, 0x0005f860, 16), palettes: memory.watch(sim_id, 0x0005f860, 16),
palette: params.palette,
index: params.index,
params, params,
scale: 4.0, state,
show_grid: true,
} }
} }
@ -118,7 +138,7 @@ impl CharacterDataWindow {
ui.label("Index"); ui.label("Index");
}); });
row.col(|ui| { row.col(|ui| {
ui.add(NumberEdit::new(&mut self.index).range(0..2048)); ui.add(NumberEdit::new(&mut self.state.index).range(0..2048));
}); });
}); });
body.row(row_height, |mut row| { body.row(row_height, |mut row| {
@ -126,11 +146,11 @@ impl CharacterDataWindow {
ui.label("Address"); ui.label("Address");
}); });
row.col(|ui| { row.col(|ui| {
let address = match self.index { let address = match self.state.index {
0x000..0x200 => 0x00060000 + self.index * 16, 0x000..0x200 => 0x00060000 + self.state.index * 16,
0x200..0x400 => 0x000e0000 + (self.index - 0x200) * 16, 0x200..0x400 => 0x000e0000 + (self.state.index - 0x200) * 16,
0x400..0x600 => 0x00160000 + (self.index - 0x400) * 16, 0x400..0x600 => 0x00160000 + (self.state.index - 0x400) * 16,
0x600..0x800 => 0x001e0000 + (self.index - 0x600) * 16, 0x600..0x800 => 0x001e0000 + (self.state.index - 0x600) * 16,
_ => unreachable!("can't happen"), _ => unreachable!("can't happen"),
}; };
let mut address_str = format!("{address:08x}"); let mut address_str = format!("{address:08x}");
@ -145,7 +165,7 @@ impl CharacterDataWindow {
ui.label("Mirror"); ui.label("Mirror");
}); });
row.col(|ui| { row.col(|ui| {
let mirror = 0x00078000 + (self.index * 16); let mirror = 0x00078000 + (self.state.index * 16);
let mut mirror_str = format!("{mirror:08x}"); let mut mirror_str = format!("{mirror:08x}");
ui.add_enabled( ui.add_enabled(
false, false,
@ -162,12 +182,12 @@ impl CharacterDataWindow {
ui.horizontal(|ui| { ui.horizontal(|ui| {
ui.label("Palette"); ui.label("Palette");
ComboBox::from_id_salt("palette") ComboBox::from_id_salt("palette")
.selected_text(self.palette.to_string()) .selected_text(self.state.palette.to_string())
.width(ui.available_width()) .width(ui.available_width())
.show_ui(ui, |ui| { .show_ui(ui, |ui| {
for palette in Palette::values() { for palette in Palette::values() {
ui.selectable_value( ui.selectable_value(
&mut self.palette, &mut self.state.palette,
palette, palette,
palette.to_string(), palette.to_string(),
); );
@ -194,23 +214,23 @@ impl CharacterDataWindow {
ui.horizontal(|ui| { ui.horizontal(|ui| {
ui.label("Scale"); ui.label("Scale");
ui.spacing_mut().slider_width = ui.available_width(); ui.spacing_mut().slider_width = ui.available_width();
let slider = Slider::new(&mut self.scale, 1.0..=10.0) let slider = Slider::new(&mut self.state.scale, 1.0..=10.0)
.step_by(1.0) .step_by(1.0)
.show_value(false); .show_value(false);
ui.add(slider); ui.add(slider);
}); });
ui.checkbox(&mut self.show_grid, "Show grid"); ui.checkbox(&mut self.state.show_grid, "Show grid");
}); });
}); });
self.params.write(CharDataParams { self.params.write(CharDataParams {
palette: self.palette, palette: self.state.palette,
index: self.index, index: self.state.index,
}); });
} }
fn load_palette_colors(&self) -> [Color32; 4] { fn load_palette_colors(&self) -> [Color32; 4] {
let Some(offset) = self.palette.offset() else { let Some(offset) = self.state.palette.offset() else {
return utils::generic_palette(Color32::RED); return utils::generic_palette(Color32::RED);
}; };
let palette = self.palettes.borrow().read(offset); let palette = self.palettes.borrow().read(offset);
@ -221,11 +241,11 @@ impl CharacterDataWindow {
fn show_chardata(&mut self, ui: &mut Ui) { fn show_chardata(&mut self, ui: &mut Ui) {
let grid = CharacterGrid::new(self.image_url("chardata")) let grid = CharacterGrid::new(self.image_url("chardata"))
.with_scale(self.scale) .with_scale(self.state.scale)
.with_grid(self.show_grid) .with_grid(self.state.show_grid)
.with_selected(self.index); .with_selected(self.state.index);
if let Some(selected) = grid.show(ui) { if let Some(selected) = grid.show(ui) {
self.index = selected; self.state.index = selected;
} }
} }
} }
@ -242,6 +262,7 @@ impl AppWindow for CharacterDataWindow {
} }
fn show(&mut self, ui: &mut Ui) { fn show(&mut self, ui: &mut Ui) {
self.state.load(ui);
CentralPanel::default().show_inside(ui, |ui| { CentralPanel::default().show_inside(ui, |ui| {
ui.horizontal_top(|ui| { ui.horizontal_top(|ui| {
StripBuilder::new(ui) StripBuilder::new(ui)
@ -257,10 +278,11 @@ impl AppWindow for CharacterDataWindow {
}); });
}); });
}); });
self.state.save(ui);
} }
} }
#[derive(Clone, Default, PartialEq, Eq)] #[derive(Clone, PartialEq, Eq)]
struct CharDataParams { struct CharDataParams {
palette: Palette, palette: Palette,
index: usize, index: usize,

View File

@ -3,6 +3,7 @@ use egui::{
ViewportBuilder, ViewportBuilder,
}; };
use egui_extras::{Column, Size, StripBuilder, TableBuilder}; use egui_extras::{Column, Size, StripBuilder, TableBuilder};
use serde::{Deserialize, Serialize};
use crate::{ use crate::{
emulator::SimId, emulator::SimId,
@ -10,42 +11,58 @@ use crate::{
memory::{MemoryClient, MemoryView}, memory::{MemoryClient, MemoryView},
window::{ window::{
AppWindow, AppWindow,
utils::{NumberEdit, UiExt as _}, utils::{NumberEdit, UiData, UiExt as _},
}, },
}; };
use super::utils; use super::utils;
pub struct FrameBufferWindow { #[derive(Clone, PartialEq, Serialize, Deserialize)]
sim_id: SimId, struct State {
index: usize, index: usize,
left: bool, left: bool,
right: bool, right: bool,
generic_palette: bool, generic_palette: bool,
params: ImageParams<FrameBufferParams>,
scale: f32, scale: f32,
} }
impl Default for State {
impl FrameBufferWindow { fn default() -> Self {
pub fn new(sim_id: SimId, memory: &MemoryClient, images: &ImageTextureLoader) -> Self { Self {
let initial_params = FrameBufferParams {
index: 0, index: 0,
left: true, left: true,
right: true, right: true,
generic_palette: false, generic_palette: false,
left_color: Color32::from_rgb(0xff, 0x00, 0x00), scale: 2.0,
right_color: Color32::from_rgb(0x00, 0xc6, 0xf0), }
}; }
}
pub struct FrameBufferWindow {
sim_id: SimId,
params: ImageParams<FrameBufferParams>,
state: UiData<State>,
}
impl FrameBufferWindow {
pub fn new(sim_id: SimId, memory: &MemoryClient, images: &ImageTextureLoader) -> Self {
let state: UiData<State> = UiData::new();
let renderer = FrameBufferRenderer::new(sim_id, memory); let renderer = FrameBufferRenderer::new(sim_id, memory);
let params = images.add(sim_id, renderer, initial_params); let params = images.add(
sim_id,
renderer,
FrameBufferParams {
index: state.index,
left: state.left,
right: state.right,
generic_palette: state.generic_palette,
left_color: Color32::from_rgb(0xff, 0x00, 0x00),
right_color: Color32::from_rgb(0x00, 0xc6, 0xf0),
},
);
Self { Self {
sim_id, sim_id,
index: params.index,
left: params.left,
right: params.right,
generic_palette: params.generic_palette,
params, params,
scale: 2.0, state,
} }
} }
@ -61,7 +78,7 @@ impl FrameBufferWindow {
ui.label("Index"); ui.label("Index");
}); });
row.col(|ui| { row.col(|ui| {
ui.add(NumberEdit::new(&mut self.index).range(0..2)); ui.add(NumberEdit::new(&mut self.state.index).range(0..2));
}); });
}); });
body.row(row_height, |mut row| { body.row(row_height, |mut row| {
@ -69,7 +86,7 @@ impl FrameBufferWindow {
ui.label("Left"); ui.label("Left");
}); });
row.col(|ui| { row.col(|ui| {
let address = self.index * 0x00008000; let address = self.state.index * 0x00008000;
let mut address_str = format!("{address:08x}"); let mut address_str = format!("{address:08x}");
ui.add_enabled( ui.add_enabled(
false, false,
@ -82,7 +99,7 @@ impl FrameBufferWindow {
ui.label("Right"); ui.label("Right");
}); });
row.col(|ui| { row.col(|ui| {
let address = self.index * 0x00008000 + 0x00010000; let address = self.state.index * 0x00008000 + 0x00010000;
let mut address_str = format!("{address:08x}"); let mut address_str = format!("{address:08x}");
ui.add_enabled( ui.add_enabled(
false, false,
@ -95,7 +112,7 @@ impl FrameBufferWindow {
ui.horizontal(|ui| { ui.horizontal(|ui| {
ui.label("Scale"); ui.label("Scale");
ui.spacing_mut().slider_width = ui.available_width(); ui.spacing_mut().slider_width = ui.available_width();
let slider = Slider::new(&mut self.scale, 1.0..=10.0) let slider = Slider::new(&mut self.state.scale, 1.0..=10.0)
.step_by(1.0) .step_by(1.0)
.show_value(false); .show_value(false);
ui.add(slider); ui.add(slider);
@ -105,29 +122,29 @@ impl FrameBufferWindow {
.body(|mut body| { .body(|mut body| {
body.row(row_height, |mut row| { body.row(row_height, |mut row| {
row.col(|ui| { row.col(|ui| {
ui.checkbox(&mut self.left, "Left"); ui.checkbox(&mut self.state.left, "Left");
}); });
row.col(|ui| { row.col(|ui| {
ui.checkbox(&mut self.right, "Right"); ui.checkbox(&mut self.state.right, "Right");
}); });
}); });
}); });
ui.checkbox(&mut self.generic_palette, "Generic colors"); ui.checkbox(&mut self.state.generic_palette, "Generic colors");
}); });
}); });
self.params.write(FrameBufferParams { self.params.write(FrameBufferParams {
index: self.index, index: self.state.index,
left: self.left, left: self.state.left,
right: self.right, right: self.state.right,
generic_palette: self.generic_palette, generic_palette: self.state.generic_palette,
..*self.params ..*self.params
}); });
} }
fn show_buffers(&mut self, ui: &mut Ui) { fn show_buffers(&mut self, ui: &mut Ui) {
let image = Image::new(self.image_url("buffer")) let image = Image::new(self.image_url("buffer"))
.fit_to_original_size(self.scale) .fit_to_original_size(self.state.scale)
.texture_options(TextureOptions::NEAREST); .texture_options(TextureOptions::NEAREST);
ui.add(image); ui.add(image);
} }
@ -145,6 +162,7 @@ impl AppWindow for FrameBufferWindow {
} }
fn show(&mut self, ui: &mut Ui) { fn show(&mut self, ui: &mut Ui) {
self.state.load(ui);
CentralPanel::default().show_inside(ui, |ui| { CentralPanel::default().show_inside(ui, |ui| {
ui.horizontal_top(|ui| { ui.horizontal_top(|ui| {
StripBuilder::new(ui) StripBuilder::new(ui)
@ -160,6 +178,7 @@ impl AppWindow for FrameBufferWindow {
}); });
}); });
}); });
self.state.save(ui);
} }
} }

View File

@ -5,6 +5,7 @@ use egui::{
TextureOptions, Ui, ViewportBuilder, TextureOptions, Ui, ViewportBuilder,
}; };
use egui_extras::{Column, Size, StripBuilder, TableBuilder}; use egui_extras::{Column, Size, StripBuilder, TableBuilder};
use serde::{Deserialize, Serialize};
use crate::{ use crate::{
emulator::SimId, emulator::SimId,
@ -12,40 +13,56 @@ use crate::{
memory::{MemoryClient, MemoryView}, memory::{MemoryClient, MemoryView},
window::{ window::{
AppWindow, AppWindow,
utils::{NumberEdit, UiExt as _}, utils::{NumberEdit, UiData, UiExt as _},
}, },
}; };
use super::utils::{self, Object}; use super::utils::{self, Object};
#[derive(Clone, PartialEq, Serialize, Deserialize)]
struct State {
index: usize,
generic_palette: bool,
scale: f32,
}
impl Default for State {
fn default() -> Self {
Self {
index: 0,
generic_palette: false,
scale: 1.0,
}
}
}
pub struct ObjectWindow { pub struct ObjectWindow {
sim_id: SimId, sim_id: SimId,
memory: Arc<MemoryClient>, memory: Arc<MemoryClient>,
objects: MemoryView, objects: MemoryView,
index: usize,
generic_palette: bool,
params: ImageParams<ObjectParams>, params: ImageParams<ObjectParams>,
scale: f32, state: UiData<State>,
} }
impl ObjectWindow { impl ObjectWindow {
pub fn new(sim_id: SimId, memory: &Arc<MemoryClient>, images: &ImageTextureLoader) -> Self { pub fn new(sim_id: SimId, memory: &Arc<MemoryClient>, images: &ImageTextureLoader) -> Self {
let initial_params = ObjectParams { let state: UiData<State> = UiData::new();
index: 0,
generic_palette: false,
left_color: Color32::from_rgb(0xff, 0x00, 0x00),
right_color: Color32::from_rgb(0x00, 0xc6, 0xf0),
};
let renderer = ObjectRenderer::new(sim_id, memory); let renderer = ObjectRenderer::new(sim_id, memory);
let params = images.add(sim_id, renderer, initial_params); let params = images.add(
sim_id,
renderer,
ObjectParams {
index: state.index,
generic_palette: state.generic_palette,
left_color: Color32::from_rgb(0xff, 0x00, 0x00),
right_color: Color32::from_rgb(0x00, 0xc6, 0xf0),
},
);
Self { Self {
sim_id, sim_id,
memory: memory.clone(), memory: memory.clone(),
objects: memory.watch(sim_id, 0x0003e000, 0x2000), objects: memory.watch(sim_id, 0x0003e000, 0x2000),
index: params.index,
generic_palette: params.generic_palette,
params, params,
scale: 1.0, state,
} }
} }
@ -61,7 +78,7 @@ impl ObjectWindow {
ui.label("Index"); ui.label("Index");
}); });
row.col(|ui| { row.col(|ui| {
ui.add(NumberEdit::new(&mut self.index).range(0..1024)); ui.add(NumberEdit::new(&mut self.state.index).range(0..1024));
}); });
}); });
body.row(row_height, |mut row| { body.row(row_height, |mut row| {
@ -69,7 +86,7 @@ impl ObjectWindow {
ui.label("Address"); ui.label("Address");
}); });
row.col(|ui| { row.col(|ui| {
let address = 0x3e000 + self.index * 8; let address = 0x3e000 + self.state.index * 8;
let mut address_str = format!("{address:08x}"); let mut address_str = format!("{address:08x}");
ui.add_enabled( ui.add_enabled(
false, false,
@ -83,7 +100,7 @@ impl ObjectWindow {
.texture_options(TextureOptions::NEAREST); .texture_options(TextureOptions::NEAREST);
ui.add(image); ui.add(image);
ui.section("Properties", |ui| { ui.section("Properties", |ui| {
let mut object = self.objects.borrow().read::<[u16; 4]>(self.index); let mut object = self.objects.borrow().read::<[u16; 4]>(self.state.index);
let mut obj = Object::parse(object); let mut obj = Object::parse(object);
TableBuilder::new(ui) TableBuilder::new(ui)
.column(Column::remainder()) .column(Column::remainder())
@ -158,7 +175,7 @@ impl ObjectWindow {
}); });
}); });
if obj.update(&mut object) { if obj.update(&mut object) {
let address = 0x3e000 + self.index * 8; let address = 0x3e000 + self.state.index * 8;
self.memory.write(self.sim_id, address as u32, &object); self.memory.write(self.sim_id, address as u32, &object);
} }
}); });
@ -166,24 +183,24 @@ impl ObjectWindow {
ui.horizontal(|ui| { ui.horizontal(|ui| {
ui.label("Scale"); ui.label("Scale");
ui.spacing_mut().slider_width = ui.available_width(); ui.spacing_mut().slider_width = ui.available_width();
let slider = Slider::new(&mut self.scale, 1.0..=10.0) let slider = Slider::new(&mut self.state.scale, 1.0..=10.0)
.step_by(1.0) .step_by(1.0)
.show_value(false); .show_value(false);
ui.add(slider); ui.add(slider);
}); });
ui.checkbox(&mut self.generic_palette, "Generic palette"); ui.checkbox(&mut self.state.generic_palette, "Generic palette");
}); });
}); });
self.params.write(ObjectParams { self.params.write(ObjectParams {
index: self.index, index: self.state.index,
generic_palette: self.generic_palette, generic_palette: self.state.generic_palette,
..*self.params ..*self.params
}); });
} }
fn show_object(&mut self, ui: &mut Ui) { fn show_object(&mut self, ui: &mut Ui) {
let image = Image::new(self.image_url("object-full")) let image = Image::new(self.image_url("object-full"))
.fit_to_original_size(self.scale) .fit_to_original_size(self.state.scale)
.texture_options(TextureOptions::NEAREST); .texture_options(TextureOptions::NEAREST);
ui.add(image); ui.add(image);
} }
@ -201,6 +218,7 @@ impl AppWindow for ObjectWindow {
} }
fn show(&mut self, ui: &mut Ui) { fn show(&mut self, ui: &mut Ui) {
self.state.load(ui);
CentralPanel::default().show_inside(ui, |ui| { CentralPanel::default().show_inside(ui, |ui| {
ui.horizontal_top(|ui| { ui.horizontal_top(|ui| {
StripBuilder::new(ui) StripBuilder::new(ui)
@ -216,6 +234,7 @@ impl AppWindow for ObjectWindow {
}); });
}); });
}); });
self.state.save(ui);
} }
} }

View File

@ -11,6 +11,7 @@ use fixed::{
}; };
use num_derive::{FromPrimitive, ToPrimitive}; use num_derive::{FromPrimitive, ToPrimitive};
use num_traits::{FromPrimitive, ToPrimitive}; use num_traits::{FromPrimitive, ToPrimitive};
use serde::{Deserialize, Serialize};
use crate::{ use crate::{
emulator::SimId, emulator::SimId,
@ -18,46 +19,62 @@ use crate::{
memory::{MemoryClient, MemoryRef, MemoryView}, memory::{MemoryClient, MemoryRef, MemoryView},
window::{ window::{
AppWindow, AppWindow,
utils::{NumberEdit, UiExt as _}, utils::{NumberEdit, UiData, UiExt as _},
}, },
}; };
use super::utils::{self, CellData, Object, shade}; use super::utils::{self, CellData, Object, shade};
#[derive(Clone, PartialEq, Serialize, Deserialize)]
struct State {
index: usize,
param_index: usize,
generic_palette: bool,
show_extents: bool,
scale: f32,
}
impl Default for State {
fn default() -> Self {
Self {
index: 31,
param_index: 0,
generic_palette: false,
show_extents: false,
scale: 1.0,
}
}
}
pub struct WorldWindow { pub struct WorldWindow {
sim_id: SimId, sim_id: SimId,
memory: Arc<MemoryClient>, memory: Arc<MemoryClient>,
worlds: MemoryView, worlds: MemoryView,
bgmaps: MemoryView, bgmaps: MemoryView,
index: usize,
param_index: usize,
generic_palette: bool,
show_extents: bool,
params: ImageParams<WorldParams>, params: ImageParams<WorldParams>,
scale: f32, state: UiData<State>,
} }
impl WorldWindow { impl WorldWindow {
pub fn new(sim_id: SimId, memory: &Arc<MemoryClient>, images: &ImageTextureLoader) -> Self { pub fn new(sim_id: SimId, memory: &Arc<MemoryClient>, images: &ImageTextureLoader) -> Self {
let initial_params = WorldParams { let state: UiData<State> = UiData::new();
index: 31,
generic_palette: false,
left_color: Color32::from_rgb(0xff, 0x00, 0x00),
right_color: Color32::from_rgb(0x00, 0xc6, 0xf0),
};
let renderer = WorldRenderer::new(sim_id, memory); let renderer = WorldRenderer::new(sim_id, memory);
let params = images.add(sim_id, renderer, initial_params); let params = images.add(
sim_id,
renderer,
WorldParams {
index: state.index,
generic_palette: state.generic_palette,
left_color: Color32::from_rgb(0xff, 0x00, 0x00),
right_color: Color32::from_rgb(0x00, 0xc6, 0xf0),
},
);
Self { Self {
sim_id, sim_id,
memory: memory.clone(), memory: memory.clone(),
worlds: memory.watch(sim_id, 0x0003d800, 0x400), worlds: memory.watch(sim_id, 0x0003d800, 0x400),
bgmaps: memory.watch(sim_id, 0x00020000, 0x20000), bgmaps: memory.watch(sim_id, 0x00020000, 0x20000),
index: params.index,
param_index: 0,
generic_palette: params.generic_palette,
show_extents: false,
params, params,
scale: 1.0, state,
} }
} }
@ -73,7 +90,7 @@ impl WorldWindow {
ui.label("Index"); ui.label("Index");
}); });
row.col(|ui| { row.col(|ui| {
ui.add(NumberEdit::new(&mut self.index).range(0..32)); ui.add(NumberEdit::new(&mut self.state.index).range(0..32));
}); });
}); });
body.row(row_height, |mut row| { body.row(row_height, |mut row| {
@ -81,7 +98,7 @@ impl WorldWindow {
ui.label("Address"); ui.label("Address");
}); });
row.col(|ui| { row.col(|ui| {
let address = 0x0003d800 + self.index * 32; let address = 0x0003d800 + self.state.index * 32;
let mut address_str = format!("{address:08x}"); let mut address_str = format!("{address:08x}");
ui.add_enabled( ui.add_enabled(
false, false,
@ -92,7 +109,7 @@ impl WorldWindow {
}); });
let mut data = { let mut data = {
let worlds = self.worlds.borrow(); let worlds = self.worlds.borrow();
worlds.read(self.index) worlds.read(self.state.index)
}; };
let mut world = World::parse(&data); let mut world = World::parse(&data);
ui.section("Properties", |ui| { ui.section("Properties", |ui| {
@ -272,7 +289,7 @@ impl WorldWindow {
}); });
}); });
if world.update(&mut data) { if world.update(&mut data) {
let address = 0x0003d800 + self.index * 32; let address = 0x0003d800 + self.state.index * 32;
self.memory.write(self.sim_id, address as u32, &data); self.memory.write(self.sim_id, address as u32, &data);
} }
if world.header.mode == WorldMode::HBias { if world.header.mode == WorldMode::HBias {
@ -287,10 +304,12 @@ impl WorldWindow {
}); });
row.col(|ui| { row.col(|ui| {
let max = world.height.max(8) as usize; let max = world.height.max(8) as usize;
ui.add(NumberEdit::new(&mut self.param_index).range(0..max)); ui.add(
NumberEdit::new(&mut self.state.param_index).range(0..max),
);
}); });
}); });
let base = (world.param_base + self.param_index * 2) & 0x1ffff; let base = (world.param_base + self.state.param_index * 2) & 0x1ffff;
let mut param = HBiasParam::load(&self.bgmaps.borrow(), base); let mut param = HBiasParam::load(&self.bgmaps.borrow(), base);
body.row(row_height, |mut row| { body.row(row_height, |mut row| {
row.col(|ui| { row.col(|ui| {
@ -337,10 +356,12 @@ impl WorldWindow {
}); });
row.col(|ui| { row.col(|ui| {
let max = world.height.max(1) as usize; let max = world.height.max(1) as usize;
ui.add(NumberEdit::new(&mut self.param_index).range(0..max)); ui.add(
NumberEdit::new(&mut self.state.param_index).range(0..max),
);
}); });
}); });
let base = (world.param_base + self.param_index * 8) & 0x1ffff; let base = (world.param_base + self.state.param_index * 8) & 0x1ffff;
let mut param = AffineParam::load(&self.bgmaps.borrow(), base); let mut param = AffineParam::load(&self.bgmaps.borrow(), base);
body.row(row_height, |mut row| { body.row(row_height, |mut row| {
row.col(|ui| { row.col(|ui| {
@ -400,49 +421,49 @@ impl WorldWindow {
}); });
}); });
} else { } else {
self.param_index = 0; self.state.param_index = 0;
} }
ui.section("Display", |ui| { ui.section("Display", |ui| {
ui.horizontal(|ui| { ui.horizontal(|ui| {
ui.label("Scale"); ui.label("Scale");
ui.spacing_mut().slider_width = ui.available_width(); ui.spacing_mut().slider_width = ui.available_width();
let slider = Slider::new(&mut self.scale, 1.0..=10.0) let slider = Slider::new(&mut self.state.scale, 1.0..=10.0)
.step_by(1.0) .step_by(1.0)
.show_value(false); .show_value(false);
ui.add(slider); ui.add(slider);
}); });
ui.checkbox(&mut self.generic_palette, "Generic palette"); ui.checkbox(&mut self.state.generic_palette, "Generic palette");
ui.checkbox(&mut self.show_extents, "Show extents"); ui.checkbox(&mut self.state.show_extents, "Show extents");
}); });
}); });
self.params.write(WorldParams { self.params.write(WorldParams {
index: self.index, index: self.state.index,
generic_palette: self.generic_palette, generic_palette: self.state.generic_palette,
..*self.params ..*self.params
}); });
} }
fn show_world(&mut self, ui: &mut Ui) { fn show_world(&mut self, ui: &mut Ui) {
let image = Image::new(self.image_url("world")) let image = Image::new(self.image_url("world"))
.fit_to_original_size(self.scale) .fit_to_original_size(self.state.scale)
.texture_options(TextureOptions::NEAREST); .texture_options(TextureOptions::NEAREST);
let res = ui.add(image); let res = ui.add(image);
if self.show_extents { if self.state.show_extents {
let world = { let world = {
let worlds = self.worlds.borrow(); let worlds = self.worlds.borrow();
let data = worlds.read(self.index); let data = worlds.read(self.state.index);
World::parse(&data) World::parse(&data)
}; };
if world.header.mode == WorldMode::Object { if world.header.mode == WorldMode::Object {
return; return;
} }
let lx1 = (world.dst_x - world.dst_parallax) as f32 * self.scale; let lx1 = (world.dst_x - world.dst_parallax) as f32 * self.state.scale;
let lx2 = lx1 + world.width as f32 * self.scale; let lx2 = lx1 + world.width as f32 * self.state.scale;
let rx1 = (world.dst_x + world.dst_parallax) as f32 * self.scale; let rx1 = (world.dst_x + world.dst_parallax) as f32 * self.state.scale;
let rx2 = rx1 + world.width as f32 * self.scale; let rx2 = rx1 + world.width as f32 * self.state.scale;
let y1 = world.dst_y as f32 * self.scale; let y1 = world.dst_y as f32 * self.state.scale;
let y2 = y1 + world.height as f32 * self.scale; let y2 = y1 + world.height as f32 * self.state.scale;
let left_color = self.params.left_color; let left_color = self.params.left_color;
let right_color = self.params.right_color; let right_color = self.params.right_color;
@ -517,6 +538,7 @@ impl AppWindow for WorldWindow {
} }
fn show(&mut self, ui: &mut Ui) { fn show(&mut self, ui: &mut Ui) {
self.state.load(ui);
CentralPanel::default().show_inside(ui, |ui| { CentralPanel::default().show_inside(ui, |ui| {
ui.horizontal_top(|ui| { ui.horizontal_top(|ui| {
StripBuilder::new(ui) StripBuilder::new(ui)
@ -532,6 +554,7 @@ impl AppWindow for WorldWindow {
}); });
}); });
}); });
self.state.save(ui);
} }
} }