Persist window state across runs
This commit is contained in:
parent
f7b37caccb
commit
e75bef634b
|
|
@ -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"
|
||||||
|
|
|
||||||
|
|
@ -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"
|
||||||
|
|
|
||||||
12
src/app.rs
12
src/app.rs
|
|
@ -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))
|
||||||
|
|
|
||||||
|
|
@ -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(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -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
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -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,
|
||||||
|
|
|
||||||
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue