lemur/src/window/vip/object.rs

343 lines
12 KiB
Rust
Raw Normal View History

2025-02-13 04:59:48 +00:00
use std::sync::Arc;
use egui::{
2025-02-15 17:56:58 +00:00
Align, CentralPanel, Checkbox, Color32, ComboBox, Context, Image, ScrollArea, Slider, TextEdit,
2025-02-13 04:59:48 +00:00
TextureOptions, Ui, ViewportBuilder, ViewportId,
};
use egui_extras::{Column, Size, StripBuilder, TableBuilder};
use crate::{
emulator::SimId,
2025-02-24 03:44:11 +00:00
images::{ImageBuffer, ImageParams, ImageProcessor, ImageRenderer, ImageTextureLoader},
2025-02-15 17:56:58 +00:00
memory::{MemoryClient, MemoryView},
2025-02-13 04:59:48 +00:00
window::{
utils::{NumberEdit, UiExt as _},
AppWindow,
},
};
2025-02-15 05:28:37 +00:00
use super::utils::{self, Object};
2025-02-13 04:59:48 +00:00
pub struct ObjectWindow {
sim_id: SimId,
2025-02-24 03:44:11 +00:00
loader: Arc<ImageTextureLoader>,
2025-02-15 17:56:58 +00:00
memory: Arc<MemoryClient>,
2025-02-13 04:59:48 +00:00
objects: MemoryView,
index: usize,
generic_palette: bool,
2025-02-24 03:44:11 +00:00
params: ImageParams<ObjectParams>,
2025-02-13 04:59:48 +00:00
scale: f32,
}
impl ObjectWindow {
2025-02-24 03:44:11 +00:00
pub fn new(sim_id: SimId, memory: &Arc<MemoryClient>, images: &mut ImageProcessor) -> Self {
let initial_params = ObjectParams {
index: 0,
generic_palette: false,
left_color: Color32::from_rgb(0xff, 0x00, 0x00),
right_color: Color32::from_rgb(0x00, 0xc6, 0xf0),
};
2025-02-13 04:59:48 +00:00
let renderer = ObjectRenderer::new(sim_id, memory);
2025-02-24 03:44:11 +00:00
let ([zoom, full], params) = images.add(renderer, initial_params);
2025-02-15 05:28:37 +00:00
let loader =
2025-02-24 03:44:11 +00:00
ImageTextureLoader::new([("vip://zoom".into(), zoom), ("vip://full".into(), full)]);
2025-02-13 04:59:48 +00:00
Self {
sim_id,
loader: Arc::new(loader),
2025-02-15 17:56:58 +00:00
memory: memory.clone(),
objects: memory.watch(sim_id, 0x0003e000, 0x2000),
index: params.index,
generic_palette: params.generic_palette,
2025-02-13 04:59:48 +00:00
params,
scale: 1.0,
}
}
fn show_form(&mut self, ui: &mut Ui) {
let row_height = ui.spacing().interact_size.y;
ui.vertical(|ui| {
TableBuilder::new(ui)
.column(Column::auto())
.column(Column::remainder())
.body(|mut body| {
body.row(row_height, |mut row| {
row.col(|ui| {
ui.label("Index");
});
row.col(|ui| {
ui.add(NumberEdit::new(&mut self.index).range(0..1024));
});
});
2025-02-15 05:28:37 +00:00
body.row(row_height, |mut row| {
row.col(|ui| {
ui.label("Address");
});
row.col(|ui| {
let address = 0x3e000 + self.index * 8;
let mut address_str = format!("{address:08x}");
ui.add_enabled(
false,
TextEdit::singleline(&mut address_str).horizontal_align(Align::Max),
);
});
});
2025-02-13 04:59:48 +00:00
});
2025-02-24 03:44:11 +00:00
let image = Image::new("vip://zoom")
2025-02-15 05:28:37 +00:00
.maintain_aspect_ratio(true)
.texture_options(TextureOptions::NEAREST);
ui.add(image);
2025-02-13 04:59:48 +00:00
ui.section("Properties", |ui| {
2025-02-15 17:56:58 +00:00
let mut object = self.objects.borrow().read::<[u16; 4]>(self.index);
2025-02-15 05:28:37 +00:00
let mut obj = Object::parse(object);
2025-02-13 04:59:48 +00:00
TableBuilder::new(ui)
.column(Column::remainder())
.column(Column::remainder())
.body(|mut body| {
body.row(row_height, |mut row| {
row.col(|ui| {
ui.label("Character");
});
row.col(|ui| {
2025-02-15 17:56:58 +00:00
ui.add(NumberEdit::new(&mut obj.data.char_index).range(0..2048));
2025-02-13 04:59:48 +00:00
});
});
body.row(row_height, |mut row| {
row.col(|ui| {
ui.label("Palette");
});
row.col(|ui| {
2025-02-15 17:56:58 +00:00
ComboBox::from_id_salt("palette")
.selected_text(format!("OBJ {}", obj.data.palette_index))
.width(ui.available_width())
.show_ui(ui, |ui| {
for palette in 0..4 {
ui.selectable_value(
&mut obj.data.palette_index,
palette,
format!("OBJ {palette}"),
);
}
});
2025-02-13 04:59:48 +00:00
});
});
body.row(row_height, |mut row| {
row.col(|ui| {
ui.label("X");
});
row.col(|ui| {
2025-02-15 17:56:58 +00:00
ui.add(NumberEdit::new(&mut obj.x).range(-512..512));
});
});
body.row(row_height, |mut row| {
row.col(|ui| {
ui.label("Y");
});
row.col(|ui| {
2025-02-15 17:56:58 +00:00
ui.add(NumberEdit::new(&mut obj.y).range(-8..=224));
});
});
body.row(row_height, |mut row| {
row.col(|ui| {
ui.label("Parallax");
});
row.col(|ui| {
2025-02-15 17:56:58 +00:00
ui.add(NumberEdit::new(&mut obj.parallax).range(-512..512));
});
});
2025-02-13 04:59:48 +00:00
body.row(row_height, |mut row| {
row.col(|ui| {
2025-02-15 17:56:58 +00:00
ui.add(Checkbox::new(&mut obj.data.hflip, "H-flip"));
2025-02-13 04:59:48 +00:00
});
row.col(|ui| {
2025-02-15 17:56:58 +00:00
ui.add(Checkbox::new(&mut obj.data.vflip, "V-flip"));
2025-02-15 05:28:37 +00:00
});
});
body.row(row_height, |mut row| {
row.col(|ui| {
2025-02-15 17:56:58 +00:00
ui.add(Checkbox::new(&mut obj.lon, "Left"));
2025-02-15 05:28:37 +00:00
});
row.col(|ui| {
2025-02-15 17:56:58 +00:00
ui.add(Checkbox::new(&mut obj.ron, "Right"));
2025-02-13 04:59:48 +00:00
});
});
});
2025-02-15 17:56:58 +00:00
if obj.update(&mut object) {
let address = 0x3e000 + self.index * 8;
self.memory.write(self.sim_id, address as u32, &object);
}
2025-02-13 04:59:48 +00:00
});
ui.section("Display", |ui| {
ui.horizontal(|ui| {
ui.label("Scale");
ui.spacing_mut().slider_width = ui.available_width();
let slider = Slider::new(&mut self.scale, 1.0..=10.0)
.step_by(1.0)
.show_value(false);
ui.add(slider);
});
ui.checkbox(&mut self.generic_palette, "Generic palette");
});
});
self.params.write(ObjectParams {
index: self.index,
generic_palette: self.generic_palette,
..*self.params
2025-02-13 04:59:48 +00:00
});
}
fn show_object(&mut self, ui: &mut Ui) {
2025-02-24 03:44:11 +00:00
let image = Image::new("vip://full")
2025-02-13 04:59:48 +00:00
.fit_to_original_size(self.scale)
.texture_options(TextureOptions::NEAREST);
ui.add(image);
}
}
impl AppWindow for ObjectWindow {
fn viewport_id(&self) -> egui::ViewportId {
ViewportId::from_hash_of(format!("object-{}", self.sim_id))
}
fn sim_id(&self) -> SimId {
self.sim_id
}
fn initial_viewport(&self) -> ViewportBuilder {
ViewportBuilder::default()
.with_title(format!("Object Data ({})", self.sim_id))
2025-02-15 05:28:37 +00:00
.with_inner_size((640.0, 500.0))
2025-02-13 04:59:48 +00:00
}
fn on_init(&mut self, ctx: &Context, _render_state: &egui_wgpu::RenderState) {
ctx.add_texture_loader(self.loader.clone());
}
fn show(&mut self, ctx: &Context) {
CentralPanel::default().show(ctx, |ui| {
ui.horizontal_top(|ui| {
StripBuilder::new(ui)
2025-02-24 02:18:21 +00:00
.size(Size::relative(0.3).at_most(200.0))
2025-02-13 04:59:48 +00:00
.size(Size::remainder())
.horizontal(|mut strip| {
strip.cell(|ui| {
ScrollArea::vertical().show(ui, |ui| self.show_form(ui));
});
strip.cell(|ui| {
ScrollArea::both().show(ui, |ui| self.show_object(ui));
});
});
});
});
}
}
#[derive(Clone, PartialEq, Eq)]
struct ObjectParams {
index: usize,
generic_palette: bool,
left_color: Color32,
right_color: Color32,
}
enum Eye {
Left,
Right,
}
struct ObjectRenderer {
chardata: MemoryView,
objects: MemoryView,
brightness: MemoryView,
palettes: MemoryView,
}
impl ObjectRenderer {
2025-02-15 17:56:58 +00:00
pub fn new(sim_id: SimId, memory: &MemoryClient) -> Self {
2025-02-13 04:59:48 +00:00
Self {
2025-02-15 17:56:58 +00:00
chardata: memory.watch(sim_id, 0x00078000, 0x8000),
objects: memory.watch(sim_id, 0x0003e000, 0x2000),
brightness: memory.watch(sim_id, 0x0005f824, 8),
palettes: memory.watch(sim_id, 0x0005f860, 16),
2025-02-13 04:59:48 +00:00
}
}
2025-02-24 03:44:11 +00:00
fn render_object(
&self,
image: &mut ImageBuffer,
params: &ObjectParams,
use_pos: bool,
eye: Eye,
) {
2025-02-13 04:59:48 +00:00
let chardata = self.chardata.borrow();
let objects = self.objects.borrow();
let brightness = self.brightness.borrow();
let palettes = self.palettes.borrow();
let object: [u16; 4] = objects.read(params.index);
2025-02-15 05:28:37 +00:00
let obj = Object::parse(object);
2025-02-13 04:59:48 +00:00
if match eye {
2025-02-15 05:28:37 +00:00
Eye::Left => !obj.lon,
Eye::Right => !obj.ron,
2025-02-13 04:59:48 +00:00
} {
return;
}
2025-02-15 21:14:03 +00:00
let brts = brightness.read::<[u8; 8]>(0);
2025-02-15 05:28:37 +00:00
let (x, y) = if use_pos {
let x = match eye {
Eye::Left => obj.x - obj.parallax,
Eye::Right => obj.x + obj.parallax,
};
(x, obj.y)
} else {
(0, 0)
};
2025-02-13 04:59:48 +00:00
2025-02-15 05:28:37 +00:00
let color = match eye {
Eye::Left => params.left_color,
Eye::Right => params.right_color,
2025-02-13 04:59:48 +00:00
};
2025-02-15 21:14:03 +00:00
let char = chardata.read::<[u16; 8]>(obj.data.char_index);
2025-02-13 04:59:48 +00:00
let palette = if params.generic_palette {
utils::generic_palette(color)
} else {
utils::palette_colors(palettes.read(8 + obj.data.palette_index * 2), &brts, color)
2025-02-13 04:59:48 +00:00
};
for row in 0..8 {
let real_y = y + row as i16;
2025-02-15 05:28:37 +00:00
if !(0..224).contains(&real_y) {
2025-02-13 04:59:48 +00:00
continue;
}
2025-02-15 05:28:37 +00:00
for (col, pixel) in
2025-02-15 21:14:03 +00:00
utils::read_char_row(&char, obj.data.hflip, obj.data.vflip, row).enumerate()
2025-02-15 05:28:37 +00:00
{
2025-02-13 04:59:48 +00:00
let real_x = x + col as i16;
2025-02-15 05:28:37 +00:00
if !(0..384).contains(&real_x) {
2025-02-13 04:59:48 +00:00
continue;
}
image.add((real_x as usize, real_y as usize), palette[pixel as usize]);
}
}
}
}
2025-02-24 03:44:11 +00:00
impl ImageRenderer<2> for ObjectRenderer {
2025-02-13 04:59:48 +00:00
type Params = ObjectParams;
2025-02-15 05:28:37 +00:00
fn sizes(&self) -> [[usize; 2]; 2] {
[[8, 8], [384, 224]]
2025-02-13 04:59:48 +00:00
}
2025-02-24 03:44:11 +00:00
fn render(&mut self, params: &Self::Params, images: &mut [ImageBuffer; 2]) {
2025-02-15 05:28:37 +00:00
images[0].clear();
self.render_object(&mut images[0], params, false, Eye::Left);
self.render_object(&mut images[0], params, false, Eye::Right);
images[1].clear();
self.render_object(&mut images[1], params, true, Eye::Left);
self.render_object(&mut images[1], params, true, Eye::Right);
2025-02-13 04:59:48 +00:00
}
}