VIP inspection tooling #4
			
				
			
		
		
		
	
							
								
								
									
										22
									
								
								src/app.rs
								
								
								
								
							
							
						
						
									
										22
									
								
								src/app.rs
								
								
								
								
							| 
						 | 
					@ -19,7 +19,10 @@ use crate::{
 | 
				
			||||||
    emulator::{EmulatorClient, EmulatorCommand, SimId},
 | 
					    emulator::{EmulatorClient, EmulatorCommand, SimId},
 | 
				
			||||||
    input::MappingProvider,
 | 
					    input::MappingProvider,
 | 
				
			||||||
    persistence::Persistence,
 | 
					    persistence::Persistence,
 | 
				
			||||||
    window::{AboutWindow, AppWindow, GameWindow, GdbServerWindow, InputWindow},
 | 
					    vram::VramLoader,
 | 
				
			||||||
 | 
					    window::{
 | 
				
			||||||
 | 
					        AboutWindow, AppWindow, CharacterDataWindow, GameWindow, GdbServerWindow, InputWindow,
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
fn load_icon() -> anyhow::Result<IconData> {
 | 
					fn load_icon() -> anyhow::Result<IconData> {
 | 
				
			||||||
| 
						 | 
					@ -36,6 +39,7 @@ fn load_icon() -> anyhow::Result<IconData> {
 | 
				
			||||||
pub struct Application {
 | 
					pub struct Application {
 | 
				
			||||||
    icon: Option<Arc<IconData>>,
 | 
					    icon: Option<Arc<IconData>>,
 | 
				
			||||||
    wgpu: WgpuState,
 | 
					    wgpu: WgpuState,
 | 
				
			||||||
 | 
					    vram: Arc<VramLoader>,
 | 
				
			||||||
    client: EmulatorClient,
 | 
					    client: EmulatorClient,
 | 
				
			||||||
    proxy: EventLoopProxy<UserEvent>,
 | 
					    proxy: EventLoopProxy<UserEvent>,
 | 
				
			||||||
    mappings: MappingProvider,
 | 
					    mappings: MappingProvider,
 | 
				
			||||||
| 
						 | 
					@ -65,6 +69,7 @@ impl Application {
 | 
				
			||||||
        Self {
 | 
					        Self {
 | 
				
			||||||
            icon,
 | 
					            icon,
 | 
				
			||||||
            wgpu,
 | 
					            wgpu,
 | 
				
			||||||
 | 
					            vram: Arc::new(VramLoader::new(client.clone())),
 | 
				
			||||||
            client,
 | 
					            client,
 | 
				
			||||||
            proxy,
 | 
					            proxy,
 | 
				
			||||||
            mappings,
 | 
					            mappings,
 | 
				
			||||||
| 
						 | 
					@ -83,7 +88,13 @@ impl Application {
 | 
				
			||||||
        }
 | 
					        }
 | 
				
			||||||
        self.viewports.insert(
 | 
					        self.viewports.insert(
 | 
				
			||||||
            viewport_id,
 | 
					            viewport_id,
 | 
				
			||||||
            Viewport::new(event_loop, &self.wgpu, self.icon.clone(), window),
 | 
					            Viewport::new(
 | 
				
			||||||
 | 
					                event_loop,
 | 
				
			||||||
 | 
					                &self.wgpu,
 | 
				
			||||||
 | 
					                self.icon.clone(),
 | 
				
			||||||
 | 
					                self.vram.clone(),
 | 
				
			||||||
 | 
					                window,
 | 
				
			||||||
 | 
					            ),
 | 
				
			||||||
        );
 | 
					        );
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -197,6 +208,10 @@ impl ApplicationHandler<UserEvent> for Application {
 | 
				
			||||||
                let about = AboutWindow;
 | 
					                let about = AboutWindow;
 | 
				
			||||||
                self.open(event_loop, Box::new(about));
 | 
					                self.open(event_loop, Box::new(about));
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
 | 
					            UserEvent::OpenCharacterData(sim_id) => {
 | 
				
			||||||
 | 
					                let vram = CharacterDataWindow::new(sim_id);
 | 
				
			||||||
 | 
					                self.open(event_loop, Box::new(vram));
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
            UserEvent::OpenDebugger(sim_id) => {
 | 
					            UserEvent::OpenDebugger(sim_id) => {
 | 
				
			||||||
                let debugger =
 | 
					                let debugger =
 | 
				
			||||||
                    GdbServerWindow::new(sim_id, self.client.clone(), self.proxy.clone());
 | 
					                    GdbServerWindow::new(sim_id, self.client.clone(), self.proxy.clone());
 | 
				
			||||||
| 
						 | 
					@ -308,6 +323,7 @@ impl Viewport {
 | 
				
			||||||
        event_loop: &ActiveEventLoop,
 | 
					        event_loop: &ActiveEventLoop,
 | 
				
			||||||
        wgpu: &WgpuState,
 | 
					        wgpu: &WgpuState,
 | 
				
			||||||
        icon: Option<Arc<IconData>>,
 | 
					        icon: Option<Arc<IconData>>,
 | 
				
			||||||
 | 
					        vram: Arc<VramLoader>,
 | 
				
			||||||
        mut app: Box<dyn AppWindow>,
 | 
					        mut app: Box<dyn AppWindow>,
 | 
				
			||||||
    ) -> Self {
 | 
					    ) -> Self {
 | 
				
			||||||
        let ctx = Context::default();
 | 
					        let ctx = Context::default();
 | 
				
			||||||
| 
						 | 
					@ -329,6 +345,7 @@ impl Viewport {
 | 
				
			||||||
            s.visuals.menu_rounding = Default::default();
 | 
					            s.visuals.menu_rounding = Default::default();
 | 
				
			||||||
        });
 | 
					        });
 | 
				
			||||||
        egui_extras::install_image_loaders(&ctx);
 | 
					        egui_extras::install_image_loaders(&ctx);
 | 
				
			||||||
 | 
					        ctx.add_image_loader(vram);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        let wgpu_config = egui_wgpu::WgpuConfiguration {
 | 
					        let wgpu_config = egui_wgpu::WgpuConfiguration {
 | 
				
			||||||
            present_mode: wgpu::PresentMode::AutoNoVsync,
 | 
					            present_mode: wgpu::PresentMode::AutoNoVsync,
 | 
				
			||||||
| 
						 | 
					@ -455,6 +472,7 @@ impl Drop for Viewport {
 | 
				
			||||||
pub enum UserEvent {
 | 
					pub enum UserEvent {
 | 
				
			||||||
    GamepadEvent(gilrs::Event),
 | 
					    GamepadEvent(gilrs::Event),
 | 
				
			||||||
    OpenAbout,
 | 
					    OpenAbout,
 | 
				
			||||||
 | 
					    OpenCharacterData(SimId),
 | 
				
			||||||
    OpenDebugger(SimId),
 | 
					    OpenDebugger(SimId),
 | 
				
			||||||
    OpenInput,
 | 
					    OpenInput,
 | 
				
			||||||
    OpenPlayer2,
 | 
					    OpenPlayer2,
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -20,6 +20,7 @@ mod gdbserver;
 | 
				
			||||||
mod graphics;
 | 
					mod graphics;
 | 
				
			||||||
mod input;
 | 
					mod input;
 | 
				
			||||||
mod persistence;
 | 
					mod persistence;
 | 
				
			||||||
 | 
					mod vram;
 | 
				
			||||||
mod window;
 | 
					mod window;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
#[derive(Parser)]
 | 
					#[derive(Parser)]
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,372 @@
 | 
				
			||||||
 | 
					use serde::{Deserialize, Serialize};
 | 
				
			||||||
 | 
					use std::{
 | 
				
			||||||
 | 
					    collections::{hash_map::Entry, HashMap},
 | 
				
			||||||
 | 
					    fmt::Display,
 | 
				
			||||||
 | 
					    sync::{Arc, Mutex},
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					use tokio::sync::mpsc;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					use egui::{
 | 
				
			||||||
 | 
					    load::{ImageLoader, ImagePoll, LoadError},
 | 
				
			||||||
 | 
					    ColorImage, Context, Vec2,
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					use crate::emulator::{EmulatorClient, EmulatorCommand, SimId};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					enum VramRequest {
 | 
				
			||||||
 | 
					    Load(String, VramResource),
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					enum VramResponse {
 | 
				
			||||||
 | 
					    Loaded(String, ColorImage),
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					pub struct VramResource {
 | 
				
			||||||
 | 
					    sim: SimId,
 | 
				
			||||||
 | 
					    kind: VramResourceKind,
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					impl VramResource {
 | 
				
			||||||
 | 
					    pub fn character_data(sim: SimId, palette: VramPalette) -> Self {
 | 
				
			||||||
 | 
					        Self {
 | 
				
			||||||
 | 
					            sim,
 | 
				
			||||||
 | 
					            kind: VramResourceKind::CharacterData { palette },
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    pub fn character(sim: SimId, palette: VramPalette, index: usize) -> Self {
 | 
				
			||||||
 | 
					        Self {
 | 
				
			||||||
 | 
					            sim,
 | 
				
			||||||
 | 
					            kind: VramResourceKind::Character { palette, index },
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    pub fn palette_color(sim: SimId, palette: VramPalette, index: usize) -> Self {
 | 
				
			||||||
 | 
					        Self {
 | 
				
			||||||
 | 
					            sim,
 | 
				
			||||||
 | 
					            kind: VramResourceKind::PaletteColor { palette, index },
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    pub fn to_uri(&self) -> String {
 | 
				
			||||||
 | 
					        format!(
 | 
				
			||||||
 | 
					            "vram://{}:{}",
 | 
				
			||||||
 | 
					            self.sim.to_index(),
 | 
				
			||||||
 | 
					            serde_json::to_string(&self.kind).unwrap(),
 | 
				
			||||||
 | 
					        )
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    fn from_uri(uri: &str) -> Option<VramResource> {
 | 
				
			||||||
 | 
					        let uri = uri.strip_prefix("vram://")?;
 | 
				
			||||||
 | 
					        let (sim, uri) = match uri.split_at_checked(2)? {
 | 
				
			||||||
 | 
					            ("0:", rest) => (SimId::Player1, rest),
 | 
				
			||||||
 | 
					            ("1:", rest) => (SimId::Player2, rest),
 | 
				
			||||||
 | 
					            _ => return None,
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					        let kind = serde_json::from_str(uri).ok()?;
 | 
				
			||||||
 | 
					        Some(Self { sim, kind })
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#[derive(Serialize, Deserialize)]
 | 
				
			||||||
 | 
					enum VramResourceKind {
 | 
				
			||||||
 | 
					    Character { palette: VramPalette, index: usize },
 | 
				
			||||||
 | 
					    CharacterData { palette: VramPalette },
 | 
				
			||||||
 | 
					    PaletteColor { palette: VramPalette, index: usize },
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					impl VramResourceKind {
 | 
				
			||||||
 | 
					    fn size(&self) -> Option<Vec2> {
 | 
				
			||||||
 | 
					        match self {
 | 
				
			||||||
 | 
					            Self::Character { .. } => Some((8.0, 8.0).into()),
 | 
				
			||||||
 | 
					            Self::CharacterData { .. } => Some((8.0 * 16.0, 8.0 * 128.0).into()),
 | 
				
			||||||
 | 
					            Self::PaletteColor { .. } => Some((1.0, 1.0).into()),
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					#[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize)]
 | 
				
			||||||
 | 
					pub enum VramPalette {
 | 
				
			||||||
 | 
					    Generic,
 | 
				
			||||||
 | 
					    Bg0,
 | 
				
			||||||
 | 
					    Bg1,
 | 
				
			||||||
 | 
					    Bg2,
 | 
				
			||||||
 | 
					    Bg3,
 | 
				
			||||||
 | 
					    Obj0,
 | 
				
			||||||
 | 
					    Obj1,
 | 
				
			||||||
 | 
					    Obj2,
 | 
				
			||||||
 | 
					    Obj3,
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					impl VramPalette {
 | 
				
			||||||
 | 
					    pub const fn values() -> [VramPalette; 9] {
 | 
				
			||||||
 | 
					        [
 | 
				
			||||||
 | 
					            Self::Generic,
 | 
				
			||||||
 | 
					            Self::Bg0,
 | 
				
			||||||
 | 
					            Self::Bg1,
 | 
				
			||||||
 | 
					            Self::Bg2,
 | 
				
			||||||
 | 
					            Self::Bg3,
 | 
				
			||||||
 | 
					            Self::Obj0,
 | 
				
			||||||
 | 
					            Self::Obj1,
 | 
				
			||||||
 | 
					            Self::Obj2,
 | 
				
			||||||
 | 
					            Self::Obj3,
 | 
				
			||||||
 | 
					        ]
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					impl Display for VramPalette {
 | 
				
			||||||
 | 
					    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
 | 
				
			||||||
 | 
					        match self {
 | 
				
			||||||
 | 
					            Self::Generic => f.write_str("Generic"),
 | 
				
			||||||
 | 
					            Self::Bg0 => f.write_str("BG 0"),
 | 
				
			||||||
 | 
					            Self::Bg1 => f.write_str("BG 1"),
 | 
				
			||||||
 | 
					            Self::Bg2 => f.write_str("BG 2"),
 | 
				
			||||||
 | 
					            Self::Bg3 => f.write_str("BG 3"),
 | 
				
			||||||
 | 
					            Self::Obj0 => f.write_str("OBJ 0"),
 | 
				
			||||||
 | 
					            Self::Obj1 => f.write_str("OBJ 1"),
 | 
				
			||||||
 | 
					            Self::Obj2 => f.write_str("OBJ 2"),
 | 
				
			||||||
 | 
					            Self::Obj3 => f.write_str("OBJ 3"),
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					pub struct VramLoader {
 | 
				
			||||||
 | 
					    cache: Mutex<HashMap<String, ImagePoll>>,
 | 
				
			||||||
 | 
					    source: Mutex<mpsc::UnboundedReceiver<VramResponse>>,
 | 
				
			||||||
 | 
					    sink: mpsc::UnboundedSender<VramRequest>,
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					impl VramLoader {
 | 
				
			||||||
 | 
					    pub fn new(client: EmulatorClient) -> Self {
 | 
				
			||||||
 | 
					        let (tx1, rx1) = mpsc::unbounded_channel();
 | 
				
			||||||
 | 
					        let (tx2, rx2) = mpsc::unbounded_channel();
 | 
				
			||||||
 | 
					        std::thread::spawn(move || {
 | 
				
			||||||
 | 
					            tokio::runtime::Builder::new_current_thread()
 | 
				
			||||||
 | 
					                .enable_all()
 | 
				
			||||||
 | 
					                .build()
 | 
				
			||||||
 | 
					                .unwrap()
 | 
				
			||||||
 | 
					                .block_on(async move {
 | 
				
			||||||
 | 
					                    let worker = VramLoadingWorker::new(rx2, tx1, client);
 | 
				
			||||||
 | 
					                    worker.run().await;
 | 
				
			||||||
 | 
					                })
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					        Self {
 | 
				
			||||||
 | 
					            cache: Mutex::new(HashMap::new()),
 | 
				
			||||||
 | 
					            source: Mutex::new(rx1),
 | 
				
			||||||
 | 
					            sink: tx2,
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					impl ImageLoader for VramLoader {
 | 
				
			||||||
 | 
					    fn id(&self) -> &str {
 | 
				
			||||||
 | 
					        concat!(module_path!(), "::VramLoader")
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    fn load(
 | 
				
			||||||
 | 
					        &self,
 | 
				
			||||||
 | 
					        _ctx: &Context,
 | 
				
			||||||
 | 
					        uri: &str,
 | 
				
			||||||
 | 
					        _size_hint: egui::SizeHint,
 | 
				
			||||||
 | 
					    ) -> Result<ImagePoll, LoadError> {
 | 
				
			||||||
 | 
					        let Some(resource) = VramResource::from_uri(uri) else {
 | 
				
			||||||
 | 
					            return Err(LoadError::NotSupported);
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					        let mut cache = self.cache.lock().unwrap();
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					            let mut source = self.source.lock().unwrap();
 | 
				
			||||||
 | 
					            while let Ok(response) = source.try_recv() {
 | 
				
			||||||
 | 
					                match response {
 | 
				
			||||||
 | 
					                    VramResponse::Loaded(uri, image) => {
 | 
				
			||||||
 | 
					                        cache.insert(
 | 
				
			||||||
 | 
					                            uri,
 | 
				
			||||||
 | 
					                            ImagePoll::Ready {
 | 
				
			||||||
 | 
					                                image: Arc::new(image),
 | 
				
			||||||
 | 
					                            },
 | 
				
			||||||
 | 
					                        );
 | 
				
			||||||
 | 
					                    }
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        let poll = match cache.entry(uri.to_string()) {
 | 
				
			||||||
 | 
					            Entry::Occupied(entry) => entry.into_mut(),
 | 
				
			||||||
 | 
					            Entry::Vacant(entry) => {
 | 
				
			||||||
 | 
					                let pending = ImagePoll::Pending {
 | 
				
			||||||
 | 
					                    size: resource.kind.size(),
 | 
				
			||||||
 | 
					                };
 | 
				
			||||||
 | 
					                let _ = self.sink.send(VramRequest::Load(uri.to_string(), resource));
 | 
				
			||||||
 | 
					                entry.insert(pending)
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					        Ok(poll.clone())
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    fn forget(&self, uri: &str) {
 | 
				
			||||||
 | 
					        self.cache.lock().unwrap().remove(uri);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    fn forget_all(&self) {
 | 
				
			||||||
 | 
					        self.cache.lock().unwrap().clear();
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    fn byte_size(&self) -> usize {
 | 
				
			||||||
 | 
					        self.cache
 | 
				
			||||||
 | 
					            .lock()
 | 
				
			||||||
 | 
					            .unwrap()
 | 
				
			||||||
 | 
					            .values()
 | 
				
			||||||
 | 
					            .map(|c| match c {
 | 
				
			||||||
 | 
					                ImagePoll::Pending { .. } => 0,
 | 
				
			||||||
 | 
					                ImagePoll::Ready { image } => image.pixels.len() * size_of::<ColorImage>(),
 | 
				
			||||||
 | 
					            })
 | 
				
			||||||
 | 
					            .sum()
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					struct VramLoadingWorker {
 | 
				
			||||||
 | 
					    source: mpsc::UnboundedReceiver<VramRequest>,
 | 
				
			||||||
 | 
					    sink: mpsc::UnboundedSender<VramResponse>,
 | 
				
			||||||
 | 
					    client: EmulatorClient,
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					impl VramLoadingWorker {
 | 
				
			||||||
 | 
					    fn new(
 | 
				
			||||||
 | 
					        source: mpsc::UnboundedReceiver<VramRequest>,
 | 
				
			||||||
 | 
					        sink: mpsc::UnboundedSender<VramResponse>,
 | 
				
			||||||
 | 
					        client: EmulatorClient,
 | 
				
			||||||
 | 
					    ) -> Self {
 | 
				
			||||||
 | 
					        Self {
 | 
				
			||||||
 | 
					            source,
 | 
				
			||||||
 | 
					            sink,
 | 
				
			||||||
 | 
					            client,
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    async fn run(mut self) {
 | 
				
			||||||
 | 
					        while let Some(request) = self.source.recv().await {
 | 
				
			||||||
 | 
					            #[allow(irrefutable_let_patterns)]
 | 
				
			||||||
 | 
					            if let VramRequest::Load(uri, resource) = request {
 | 
				
			||||||
 | 
					                let Some(image) = self.load(resource).await else {
 | 
				
			||||||
 | 
					                    continue;
 | 
				
			||||||
 | 
					                };
 | 
				
			||||||
 | 
					                if self.sink.send(VramResponse::Loaded(uri, image)).is_err() {
 | 
				
			||||||
 | 
					                    return;
 | 
				
			||||||
 | 
					                }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async fn load(&self, resource: VramResource) -> Option<ColorImage> {
 | 
				
			||||||
 | 
					        let sim = resource.sim;
 | 
				
			||||||
 | 
					        match resource.kind {
 | 
				
			||||||
 | 
					            VramResourceKind::Character { palette, index } => {
 | 
				
			||||||
 | 
					                self.load_character(sim, palette, index).await
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            VramResourceKind::CharacterData { palette } => {
 | 
				
			||||||
 | 
					                self.load_character_data(sim, palette).await
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            VramResourceKind::PaletteColor { palette, index } => {
 | 
				
			||||||
 | 
					                self.load_palette_color(sim, palette, index).await
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async fn load_character(
 | 
				
			||||||
 | 
					        &self,
 | 
				
			||||||
 | 
					        sim: SimId,
 | 
				
			||||||
 | 
					        palette: VramPalette,
 | 
				
			||||||
 | 
					        index: usize,
 | 
				
			||||||
 | 
					    ) -> Option<ColorImage> {
 | 
				
			||||||
 | 
					        if index >= 2048 {
 | 
				
			||||||
 | 
					            return None;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        let address = 0x00078000 + (index as u32 * 16);
 | 
				
			||||||
 | 
					        let (memory, palette) = tokio::join!(
 | 
				
			||||||
 | 
					            self.read_memory(sim, address, 16),
 | 
				
			||||||
 | 
					            self.load_palette_colors(sim, palette),
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					        let palette = palette?;
 | 
				
			||||||
 | 
					        let mut buffer = vec![];
 | 
				
			||||||
 | 
					        for byte in memory? {
 | 
				
			||||||
 | 
					            for offset in (0..8).step_by(2) {
 | 
				
			||||||
 | 
					                let char = (byte >> offset) & 0x3;
 | 
				
			||||||
 | 
					                buffer.push(palette[char as usize]);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        Some(ColorImage::from_gray([8, 8], &buffer))
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async fn load_character_data(&self, sim: SimId, palette: VramPalette) -> Option<ColorImage> {
 | 
				
			||||||
 | 
					        let (memory, palette) = tokio::join!(
 | 
				
			||||||
 | 
					            self.read_memory(sim, 0x00078000, 16 * 2048),
 | 
				
			||||||
 | 
					            self.load_palette_colors(sim, palette),
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					        let palette = palette?;
 | 
				
			||||||
 | 
					        let mut buffer = vec![0; 8 * 8 * 2048];
 | 
				
			||||||
 | 
					        for (i, byte) in memory?.into_iter().enumerate() {
 | 
				
			||||||
 | 
					            let bytes = [0, 2, 4, 6].map(|off| palette[(byte as usize >> off) & 0x3]);
 | 
				
			||||||
 | 
					            let char_index = i / 16;
 | 
				
			||||||
 | 
					            let in_char_pos = i % 16;
 | 
				
			||||||
 | 
					            let x = ((char_index % 16) * 8) + ((in_char_pos % 2) * 4);
 | 
				
			||||||
 | 
					            let y = ((char_index / 16) * 8) + (in_char_pos / 2);
 | 
				
			||||||
 | 
					            let write_index = (y * 16 * 8) + x;
 | 
				
			||||||
 | 
					            buffer[write_index..(write_index + 4)].copy_from_slice(&bytes);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        Some(ColorImage::from_gray([8 * 16, 8 * 128], &buffer))
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async fn load_palette_color(
 | 
				
			||||||
 | 
					        &self,
 | 
				
			||||||
 | 
					        sim: SimId,
 | 
				
			||||||
 | 
					        palette: VramPalette,
 | 
				
			||||||
 | 
					        index: usize,
 | 
				
			||||||
 | 
					    ) -> Option<ColorImage> {
 | 
				
			||||||
 | 
					        if index == 0 {
 | 
				
			||||||
 | 
					            return Some(ColorImage::from_gray([1, 1], &[0]));
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        if index > 3 {
 | 
				
			||||||
 | 
					            return None;
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        let shade = *self.load_palette_colors(sim, palette).await?.get(index)?;
 | 
				
			||||||
 | 
					        Some(ColorImage::from_gray([1, 1], &[shade]))
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async fn load_palette_colors(&self, sim: SimId, palette: VramPalette) -> Option<[u8; 4]> {
 | 
				
			||||||
 | 
					        let offset = match palette {
 | 
				
			||||||
 | 
					            VramPalette::Generic => {
 | 
				
			||||||
 | 
					                return Some([0, 64, 128, 255]);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            VramPalette::Bg0 => 0,
 | 
				
			||||||
 | 
					            VramPalette::Bg1 => 2,
 | 
				
			||||||
 | 
					            VramPalette::Bg2 => 4,
 | 
				
			||||||
 | 
					            VramPalette::Bg3 => 6,
 | 
				
			||||||
 | 
					            VramPalette::Obj0 => 8,
 | 
				
			||||||
 | 
					            VramPalette::Obj1 => 10,
 | 
				
			||||||
 | 
					            VramPalette::Obj2 => 12,
 | 
				
			||||||
 | 
					            VramPalette::Obj3 => 14,
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					        let (palettes, brightnesses) = tokio::join!(
 | 
				
			||||||
 | 
					            self.read_memory(sim, 0x0005f860, 16),
 | 
				
			||||||
 | 
					            self.read_memory(sim, 0x0005f824, 6),
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					        let palette = *palettes?.get(offset)?;
 | 
				
			||||||
 | 
					        let brts = brightnesses?;
 | 
				
			||||||
 | 
					        let shades = [
 | 
				
			||||||
 | 
					            0,
 | 
				
			||||||
 | 
					            brts[0],
 | 
				
			||||||
 | 
					            brts[2],
 | 
				
			||||||
 | 
					            brts[0].saturating_add(brts[2]).saturating_add(brts[4]),
 | 
				
			||||||
 | 
					        ];
 | 
				
			||||||
 | 
					        Some([
 | 
				
			||||||
 | 
					            0,
 | 
				
			||||||
 | 
					            shades[(palette >> 2) as usize & 0x03],
 | 
				
			||||||
 | 
					            shades[(palette >> 4) as usize & 0x03],
 | 
				
			||||||
 | 
					            shades[(palette >> 6) as usize & 0x03],
 | 
				
			||||||
 | 
					        ])
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    async fn read_memory(&self, sim: SimId, address: u32, size: usize) -> Option<Vec<u8>> {
 | 
				
			||||||
 | 
					        let (tx, rx) = oneshot::channel();
 | 
				
			||||||
 | 
					        self.client
 | 
				
			||||||
 | 
					            .send_command(EmulatorCommand::ReadMemory(sim, address, size, vec![], tx));
 | 
				
			||||||
 | 
					        rx.await.ok()
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -1,4 +1,5 @@
 | 
				
			||||||
pub use about::AboutWindow;
 | 
					pub use about::AboutWindow;
 | 
				
			||||||
 | 
					pub use character_data::CharacterDataWindow;
 | 
				
			||||||
use egui::{Context, ViewportBuilder, ViewportId};
 | 
					use egui::{Context, ViewportBuilder, ViewportId};
 | 
				
			||||||
pub use game::GameWindow;
 | 
					pub use game::GameWindow;
 | 
				
			||||||
pub use gdb::GdbServerWindow;
 | 
					pub use gdb::GdbServerWindow;
 | 
				
			||||||
| 
						 | 
					@ -8,6 +9,7 @@ use winit::event::KeyEvent;
 | 
				
			||||||
use crate::emulator::SimId;
 | 
					use crate::emulator::SimId;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
mod about;
 | 
					mod about;
 | 
				
			||||||
 | 
					mod character_data;
 | 
				
			||||||
mod game;
 | 
					mod game;
 | 
				
			||||||
mod game_screen;
 | 
					mod game_screen;
 | 
				
			||||||
mod gdb;
 | 
					mod gdb;
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -0,0 +1,253 @@
 | 
				
			||||||
 | 
					use egui::{
 | 
				
			||||||
 | 
					    Align, CentralPanel, Color32, ComboBox, Context, Frame, Image, RichText, ScrollArea, Sense,
 | 
				
			||||||
 | 
					    Slider, TextEdit, TextureOptions, Ui, UiBuilder, Vec2, ViewportBuilder, ViewportId,
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					use egui_extras::{Column, Size, StripBuilder, TableBuilder};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					use crate::{
 | 
				
			||||||
 | 
					    emulator::SimId,
 | 
				
			||||||
 | 
					    vram::{VramPalette, VramResource},
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					use super::AppWindow;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					pub struct CharacterDataWindow {
 | 
				
			||||||
 | 
					    sim_id: SimId,
 | 
				
			||||||
 | 
					    palette: VramPalette,
 | 
				
			||||||
 | 
					    index: usize,
 | 
				
			||||||
 | 
					    index_str: String,
 | 
				
			||||||
 | 
					    scale: f32,
 | 
				
			||||||
 | 
					    show_grid: bool,
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					impl CharacterDataWindow {
 | 
				
			||||||
 | 
					    pub fn new(sim_id: SimId) -> Self {
 | 
				
			||||||
 | 
					        Self {
 | 
				
			||||||
 | 
					            sim_id,
 | 
				
			||||||
 | 
					            palette: VramPalette::Generic,
 | 
				
			||||||
 | 
					            index: 0,
 | 
				
			||||||
 | 
					            index_str: "0".into(),
 | 
				
			||||||
 | 
					            scale: 4.0,
 | 
				
			||||||
 | 
					            show_grid: true,
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    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| {
 | 
				
			||||||
 | 
					                            let res = ui.add(
 | 
				
			||||||
 | 
					                                TextEdit::singleline(&mut self.index_str)
 | 
				
			||||||
 | 
					                                    .horizontal_align(Align::Max),
 | 
				
			||||||
 | 
					                            );
 | 
				
			||||||
 | 
					                            if res.changed() {
 | 
				
			||||||
 | 
					                                if let Some(index) =
 | 
				
			||||||
 | 
					                                    self.index_str.parse().ok().filter(|id| *id < 2048)
 | 
				
			||||||
 | 
					                                {
 | 
				
			||||||
 | 
					                                    self.index = index;
 | 
				
			||||||
 | 
					                                }
 | 
				
			||||||
 | 
					                            }
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					                    });
 | 
				
			||||||
 | 
					                    body.row(row_height, |mut row| {
 | 
				
			||||||
 | 
					                        row.col(|ui| {
 | 
				
			||||||
 | 
					                            ui.label("Address");
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					                        row.col(|ui| {
 | 
				
			||||||
 | 
					                            let address = match self.index {
 | 
				
			||||||
 | 
					                                0x000..0x200 => 0x00060000 + self.index,
 | 
				
			||||||
 | 
					                                0x200..0x400 => 0x000e0000 + (self.index - 0x200),
 | 
				
			||||||
 | 
					                                0x400..0x600 => 0x00160000 + (self.index - 0x400),
 | 
				
			||||||
 | 
					                                0x600..0x800 => 0x001e0000 + (self.index - 0x600),
 | 
				
			||||||
 | 
					                                _ => unreachable!("can't happen"),
 | 
				
			||||||
 | 
					                            };
 | 
				
			||||||
 | 
					                            let mut address_str = format!("{address:08x}");
 | 
				
			||||||
 | 
					                            ui.add_enabled(
 | 
				
			||||||
 | 
					                                false,
 | 
				
			||||||
 | 
					                                TextEdit::singleline(&mut address_str).horizontal_align(Align::Max),
 | 
				
			||||||
 | 
					                            );
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					                    });
 | 
				
			||||||
 | 
					                    body.row(row_height, |mut row| {
 | 
				
			||||||
 | 
					                        row.col(|ui| {
 | 
				
			||||||
 | 
					                            ui.label("Mirror");
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					                        row.col(|ui| {
 | 
				
			||||||
 | 
					                            let mirror = 0x00078000 + (self.index * 16);
 | 
				
			||||||
 | 
					                            let mut mirror_str = format!("{mirror:08x}");
 | 
				
			||||||
 | 
					                            ui.add_enabled(
 | 
				
			||||||
 | 
					                                false,
 | 
				
			||||||
 | 
					                                TextEdit::singleline(&mut mirror_str).horizontal_align(Align::Max),
 | 
				
			||||||
 | 
					                            );
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					                    });
 | 
				
			||||||
 | 
					                });
 | 
				
			||||||
 | 
					            let resource = VramResource::character(self.sim_id, self.palette, self.index);
 | 
				
			||||||
 | 
					            let image = Image::new(resource.to_uri())
 | 
				
			||||||
 | 
					                .maintain_aspect_ratio(true)
 | 
				
			||||||
 | 
					                .tint(Color32::RED)
 | 
				
			||||||
 | 
					                .texture_options(TextureOptions::NEAREST);
 | 
				
			||||||
 | 
					            ui.add(image);
 | 
				
			||||||
 | 
					            ui.section("Colors", |ui| {
 | 
				
			||||||
 | 
					                ui.horizontal(|ui| {
 | 
				
			||||||
 | 
					                    ui.label("Palette");
 | 
				
			||||||
 | 
					                    ComboBox::from_id_salt("palette")
 | 
				
			||||||
 | 
					                        .selected_text(self.palette.to_string())
 | 
				
			||||||
 | 
					                        .width(ui.available_width())
 | 
				
			||||||
 | 
					                        .show_ui(ui, |ui| {
 | 
				
			||||||
 | 
					                            for palette in VramPalette::values() {
 | 
				
			||||||
 | 
					                                ui.selectable_value(
 | 
				
			||||||
 | 
					                                    &mut self.palette,
 | 
				
			||||||
 | 
					                                    palette,
 | 
				
			||||||
 | 
					                                    palette.to_string(),
 | 
				
			||||||
 | 
					                                );
 | 
				
			||||||
 | 
					                            }
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					                });
 | 
				
			||||||
 | 
					                TableBuilder::new(ui)
 | 
				
			||||||
 | 
					                    .columns(Column::remainder(), 4)
 | 
				
			||||||
 | 
					                    .body(|mut body| {
 | 
				
			||||||
 | 
					                        body.row(30.0, |mut row| {
 | 
				
			||||||
 | 
					                            for index in 0..4 {
 | 
				
			||||||
 | 
					                                let resource =
 | 
				
			||||||
 | 
					                                    VramResource::palette_color(self.sim_id, self.palette, index);
 | 
				
			||||||
 | 
					                                row.col(|ui| {
 | 
				
			||||||
 | 
					                                    let rect = ui.available_rect_before_wrap();
 | 
				
			||||||
 | 
					                                    let scale = rect.height() / rect.width();
 | 
				
			||||||
 | 
					                                    let rect = rect.scale_from_center2(Vec2::new(scale, 1.0));
 | 
				
			||||||
 | 
					                                    let image = Image::new(resource.to_uri())
 | 
				
			||||||
 | 
					                                        .tint(Color32::RED)
 | 
				
			||||||
 | 
					                                        .fit_to_exact_size(rect.max - rect.min);
 | 
				
			||||||
 | 
					                                    ui.put(rect, image);
 | 
				
			||||||
 | 
					                                });
 | 
				
			||||||
 | 
					                            }
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					                    });
 | 
				
			||||||
 | 
					            });
 | 
				
			||||||
 | 
					            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.show_grid, "Show grid");
 | 
				
			||||||
 | 
					            });
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    fn show_chardata(&mut self, ui: &mut Ui) {
 | 
				
			||||||
 | 
					        let start_pos = ui.cursor().min;
 | 
				
			||||||
 | 
					        let resource = VramResource::character_data(self.sim_id, self.palette);
 | 
				
			||||||
 | 
					        let image = Image::new(resource.to_uri())
 | 
				
			||||||
 | 
					            .fit_to_original_size(self.scale)
 | 
				
			||||||
 | 
					            .tint(Color32::RED)
 | 
				
			||||||
 | 
					            .texture_options(TextureOptions::NEAREST)
 | 
				
			||||||
 | 
					            .sense(Sense::click());
 | 
				
			||||||
 | 
					        let res = ui.add(image);
 | 
				
			||||||
 | 
					        if res.clicked() {
 | 
				
			||||||
 | 
					            if let Some(click_pos) = res.interact_pointer_pos() {
 | 
				
			||||||
 | 
					                let fixed_pos = (click_pos - start_pos) / self.scale;
 | 
				
			||||||
 | 
					                let x = (fixed_pos.x / 8.0) as usize;
 | 
				
			||||||
 | 
					                let y = (fixed_pos.y / 8.0) as usize;
 | 
				
			||||||
 | 
					                self.index = (y * 16) + x;
 | 
				
			||||||
 | 
					                self.index_str = self.index.to_string();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        let painter = ui.painter_at(res.rect);
 | 
				
			||||||
 | 
					        if self.show_grid {
 | 
				
			||||||
 | 
					            let stroke = ui.style().visuals.widgets.noninteractive.fg_stroke;
 | 
				
			||||||
 | 
					            for x in (1..16).map(|i| (i as f32) * 8.0 * self.scale) {
 | 
				
			||||||
 | 
					                let p1 = res.rect.min + (x, 0.0).into();
 | 
				
			||||||
 | 
					                let p2 = res.rect.min + (x, 128.0 * 8.0 * self.scale).into();
 | 
				
			||||||
 | 
					                painter.line(vec![p1, p2], stroke);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					            for y in (1..128).map(|i| (i as f32) * 8.0 * self.scale) {
 | 
				
			||||||
 | 
					                let p1 = res.rect.min + (0.0, y).into();
 | 
				
			||||||
 | 
					                let p2 = res.rect.min + (16.0 * 8.0 * self.scale, y).into();
 | 
				
			||||||
 | 
					                painter.line(vec![p1, p2], stroke);
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					        // draw box around selected
 | 
				
			||||||
 | 
					        let x1 = (self.index % 16) as f32 * 8.0 * self.scale;
 | 
				
			||||||
 | 
					        let x2 = x1 + (8.0 * self.scale);
 | 
				
			||||||
 | 
					        let y1 = (self.index / 16) as f32 * 8.0 * self.scale;
 | 
				
			||||||
 | 
					        let y2 = y1 + (8.0 * self.scale);
 | 
				
			||||||
 | 
					        painter.line(
 | 
				
			||||||
 | 
					            vec![
 | 
				
			||||||
 | 
					                (res.rect.min + (x1, y1).into()),
 | 
				
			||||||
 | 
					                (res.rect.min + (x2, y1).into()),
 | 
				
			||||||
 | 
					                (res.rect.min + (x2, y2).into()),
 | 
				
			||||||
 | 
					                (res.rect.min + (x1, y2).into()),
 | 
				
			||||||
 | 
					                (res.rect.min + (x1, y1).into()),
 | 
				
			||||||
 | 
					            ],
 | 
				
			||||||
 | 
					            ui.style().visuals.widgets.active.fg_stroke,
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					impl AppWindow for CharacterDataWindow {
 | 
				
			||||||
 | 
					    fn viewport_id(&self) -> ViewportId {
 | 
				
			||||||
 | 
					        ViewportId::from_hash_of(format!("chardata-{}", self.sim_id))
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    fn sim_id(&self) -> SimId {
 | 
				
			||||||
 | 
					        self.sim_id
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    fn initial_viewport(&self) -> ViewportBuilder {
 | 
				
			||||||
 | 
					        ViewportBuilder::default()
 | 
				
			||||||
 | 
					            .with_title(format!("Character Data ({})", self.sim_id))
 | 
				
			||||||
 | 
					            .with_inner_size((640.0, 480.0))
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    fn show(&mut self, ctx: &Context) {
 | 
				
			||||||
 | 
					        CentralPanel::default().show(ctx, |ui| {
 | 
				
			||||||
 | 
					            ui.horizontal_top(|ui| {
 | 
				
			||||||
 | 
					                StripBuilder::new(ui)
 | 
				
			||||||
 | 
					                    .size(Size::relative(0.3))
 | 
				
			||||||
 | 
					                    .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_chardata(ui));
 | 
				
			||||||
 | 
					                        });
 | 
				
			||||||
 | 
					                    });
 | 
				
			||||||
 | 
					            });
 | 
				
			||||||
 | 
					        });
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					trait UiExt {
 | 
				
			||||||
 | 
					    fn section(&mut self, title: impl Into<String>, add_contents: impl FnOnce(&mut Ui));
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					impl UiExt for Ui {
 | 
				
			||||||
 | 
					    fn section(&mut self, title: impl Into<String>, add_contents: impl FnOnce(&mut Ui)) {
 | 
				
			||||||
 | 
					        let mut frame = Frame::group(self.style());
 | 
				
			||||||
 | 
					        frame.outer_margin.top += 10.0;
 | 
				
			||||||
 | 
					        frame.inner_margin.top += 2.0;
 | 
				
			||||||
 | 
					        let res = frame.show(self, add_contents);
 | 
				
			||||||
 | 
					        let text = RichText::new(title).background_color(self.style().visuals.panel_fill);
 | 
				
			||||||
 | 
					        let old_rect = res.response.rect;
 | 
				
			||||||
 | 
					        let mut text_rect = old_rect;
 | 
				
			||||||
 | 
					        text_rect.min.x += 6.0;
 | 
				
			||||||
 | 
					        let new_rect = self
 | 
				
			||||||
 | 
					            .allocate_new_ui(UiBuilder::new().max_rect(text_rect), |ui| ui.label(text))
 | 
				
			||||||
 | 
					            .response
 | 
				
			||||||
 | 
					            .rect;
 | 
				
			||||||
 | 
					        self.allocate_space((old_rect.max - new_rect.max) - (old_rect.min - new_rect.min));
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					@ -132,6 +132,12 @@ impl GameWindow {
 | 
				
			||||||
                    .unwrap();
 | 
					                    .unwrap();
 | 
				
			||||||
                ui.close_menu();
 | 
					                ui.close_menu();
 | 
				
			||||||
            }
 | 
					            }
 | 
				
			||||||
 | 
					            if ui.button("Character Data").clicked() {
 | 
				
			||||||
 | 
					                self.proxy
 | 
				
			||||||
 | 
					                    .send_event(UserEvent::OpenCharacterData(self.sim_id))
 | 
				
			||||||
 | 
					                    .unwrap();
 | 
				
			||||||
 | 
					                ui.close_menu();
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
        });
 | 
					        });
 | 
				
			||||||
        ui.menu_button("About", |ui| {
 | 
					        ui.menu_button("About", |ui| {
 | 
				
			||||||
            self.proxy.send_event(UserEvent::OpenAbout).unwrap();
 | 
					            self.proxy.send_event(UserEvent::OpenAbout).unwrap();
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in New Issue