#include "../tomoko.hpp" #include "about.cpp" unique_pointer aboutWindow; unique_pointer presentation; Presentation::Presentation() { presentation = this; libraryMenu.setText("Library"); string_vector manufacturers; for(auto& emulator : program->emulators) { if(!manufacturers.find(emulator->information.manufacturer)) { manufacturers.append(emulator->information.manufacturer); } } for(auto& manufacturer : manufacturers) { Menu manufacturerMenu{&libraryMenu}; manufacturerMenu.setText(manufacturer); for(auto& emulator : program->emulators) { if(emulator->information.manufacturer != manufacturer) continue; for(auto& medium : emulator->media) { auto item = new MenuItem{&manufacturerMenu}; item->setText({medium.name, " ..."}).onActivate([=] { program->loadMedium(*emulator, medium); }); } } } //add icarus menu options -- but only if icarus binary is present if(execute("icarus", "--name").output.strip() == "icarus") { libraryMenu.append(MenuSeparator()); libraryMenu.append(MenuItem().setText("Load ROM File ...").onActivate([&] { audio->clear(); if(auto location = execute("icarus", "--import")) { program->mediumQueue.append(location.output.strip()); program->loadMedium(); } })); libraryMenu.append(MenuItem().setText("Import ROM Files ...").onActivate([&] { invoke("icarus"); })); } systemMenu.setText("System").setVisible(false); reloadSystem.setText("Power Cycle").onActivate([&] { program->powerCycle(); }); unloadSystem.setText("Unload").onActivate([&] { program->unloadMedium(); }); settingsMenu.setText("Settings"); videoScaleMenu.setText("Video Scale"); if(settings["Video/Scale"].text() == "Small") videoScaleSmall.setChecked(); if(settings["Video/Scale"].text() == "Medium") videoScaleMedium.setChecked(); if(settings["Video/Scale"].text() == "Large") videoScaleLarge.setChecked(); videoScaleSmall.setText("Small").onActivate([&] { settings["Video/Scale"].setValue("Small"); resizeViewport(); }); videoScaleMedium.setText("Medium").onActivate([&] { settings["Video/Scale"].setValue("Medium"); resizeViewport(); }); videoScaleLarge.setText("Large").onActivate([&] { settings["Video/Scale"].setValue("Large"); resizeViewport(); }); aspectCorrection.setText("Aspect Correction").setChecked(settings["Video/AspectCorrection"].boolean()).onToggle([&] { settings["Video/AspectCorrection"].setValue(aspectCorrection.checked()); resizeViewport(); }); videoEmulationMenu.setText("Video Emulation"); blurEmulation.setText("Blurring").setChecked(settings["Video/BlurEmulation"].boolean()).onToggle([&] { settings["Video/BlurEmulation"].setValue(blurEmulation.checked()); if(emulator) emulator->set("Blur Emulation", blurEmulation.checked()); }); colorEmulation.setText("Colors").setChecked(settings["Video/ColorEmulation"].boolean()).onToggle([&] { settings["Video/ColorEmulation"].setValue(colorEmulation.checked()); if(emulator) emulator->set("Color Emulation", colorEmulation.checked()); }); scanlineEmulation.setText("Scanlines").setChecked(settings["Video/ScanlineEmulation"].boolean()).setVisible(false).onToggle([&] { settings["Video/ScanlineEmulation"].setValue(scanlineEmulation.checked()); if(emulator) emulator->set("Scanline Emulation", scanlineEmulation.checked()); }); maskOverscan.setText("Mask Overscan").setChecked(settings["Video/Overscan/Mask"].boolean()).onToggle([&] { settings["Video/Overscan/Mask"].setValue(maskOverscan.checked()); }); videoShaderMenu.setText("Video Shader"); videoShaderNone.setText("None").onActivate([&] { settings["Video/Shader"].setValue("None"); program->updateVideoShader(); }); videoShaderBlur.setText("Blur").onActivate([&] { settings["Video/Shader"].setValue("Blur"); program->updateVideoShader(); }); loadShaders(); synchronizeVideo.setText("Synchronize Video").setChecked(settings["Video/Synchronize"].boolean()).setVisible(false).onToggle([&] { settings["Video/Synchronize"].setValue(synchronizeVideo.checked()); video->set(Video::Synchronize, synchronizeVideo.checked()); }); synchronizeAudio.setText("Synchronize Audio").setChecked(settings["Audio/Synchronize"].boolean()).onToggle([&] { settings["Audio/Synchronize"].setValue(synchronizeAudio.checked()); audio->set(Audio::Synchronize, synchronizeAudio.checked()); }); muteAudio.setText("Mute Audio").setChecked(settings["Audio/Mute"].boolean()).onToggle([&] { settings["Audio/Mute"].setValue(muteAudio.checked()); program->updateAudioEffects(); }); showStatusBar.setText("Show Status Bar").setChecked(settings["UserInterface/ShowStatusBar"].boolean()).onToggle([&] { settings["UserInterface/ShowStatusBar"].setValue(showStatusBar.checked()); statusBar.setVisible(showStatusBar.checked()); if(visible()) resizeViewport(); }); showConfiguration.setText("Configuration ...").onActivate([&] { settingsManager->show(2); }); toolsMenu.setText("Tools").setVisible(false); saveStateMenu.setText("Save State"); saveSlot1.setText("Slot 1").onActivate([&] { program->saveState(1); }); saveSlot2.setText("Slot 2").onActivate([&] { program->saveState(2); }); saveSlot3.setText("Slot 3").onActivate([&] { program->saveState(3); }); saveSlot4.setText("Slot 4").onActivate([&] { program->saveState(4); }); saveSlot5.setText("Slot 5").onActivate([&] { program->saveState(5); }); loadStateMenu.setText("Load State"); loadSlot1.setText("Slot 1").onActivate([&] { program->loadState(1); }); loadSlot2.setText("Slot 2").onActivate([&] { program->loadState(2); }); loadSlot3.setText("Slot 3").onActivate([&] { program->loadState(3); }); loadSlot4.setText("Slot 4").onActivate([&] { program->loadState(4); }); loadSlot5.setText("Slot 5").onActivate([&] { program->loadState(5); }); cheatEditor.setText("Cheat Editor").onActivate([&] { toolsManager->show(0); }); stateManager.setText("State Manager").onActivate([&] { toolsManager->show(1); }); manifestViewer.setText("Manifest Viewer").onActivate([&] { toolsManager->show(2); }); helpMenu.setText("Help"); documentation.setText("Documentation ...").onActivate([&] { invoke("http://doc.byuu.org/higan/"); }); about.setText("About ...").onActivate([&] { aboutWindow->setVisible().setFocused(); }); statusBar.setFont(Font().setBold()); statusBar.setVisible(settings["UserInterface/ShowStatusBar"].boolean()); viewport.setDroppable().onDrop([&](auto locations) { if(!directory::exists(locations(0))) return; program->mediumQueue.append(locations(0)); program->loadMedium(); }); onClose([&] { program->quit(); }); setTitle({"higan v", Emulator::Version}); setResizable(false); setBackgroundColor({0, 0, 0}); resizeViewport(); setCentered(); #if defined(PLATFORM_WINDOWS) Application::Windows::onModalChange([](bool modal) { if(modal && audio) audio->clear(); }); #endif #if defined(PLATFORM_MACOSX) showConfigurationSeparator.setVisible(false); showConfiguration.setVisible(false); about.setVisible(false); Application::Cocoa::onAbout([&] { about.doActivate(); }); Application::Cocoa::onActivate([&] { setFocused(); }); Application::Cocoa::onPreferences([&] { showConfiguration.doActivate(); }); Application::Cocoa::onQuit([&] { doClose(); }); #endif } auto Presentation::updateEmulator() -> void { if(!emulator) return; inputPort1.setVisible(false).reset(); inputPort2.setVisible(false).reset(); inputPort3.setVisible(false).reset(); for(auto n : range(emulator->ports)) { if(n >= 3) break; auto& port = emulator->ports[n]; auto& menu = (n == 0 ? inputPort1 : n == 1 ? inputPort2 : inputPort3); menu.setText(port.name); Group devices; for(auto& device : port.devices) { MenuRadioItem item{&menu}; item.setText(device.name).onActivate([=] { auto path = string{emulator->information.name, "/", port.name}.replace(" ", ""); settings[path].setValue(device.name); emulator->connect(port.id, device.id); }); devices.append(item); } if(devices.objectCount() > 1) { auto path = string{emulator->information.name, "/", port.name}.replace(" ", ""); auto device = settings(path).text(); for(auto item : devices.objects()) { if(item.text() == device) item.setChecked(); } menu.setVisible(); } } systemMenuSeparatorPorts.setVisible(inputPort1.visible() || inputPort2.visible() || inputPort3.visible()); emulator->set("Blur Emulation", blurEmulation.checked()); emulator->set("Color Emulation", colorEmulation.checked()); emulator->set("Scanline Emulation", scanlineEmulation.checked()); } auto Presentation::clearViewport() -> void { if(!video) return; uint32_t* output; uint length = 0; uint width = viewport.geometry().width(); uint height = viewport.geometry().height(); if(video->lock(output, length, width, height)) { for(uint y : range(height)) { auto dp = output + y * (length >> 2); for(uint x : range(width)) *dp++ = 0xff000000; } video->unlock(); video->refresh(); } } auto Presentation::resizeViewport() -> void { //clear video area before resizing to avoid seeing distorted video momentarily clearViewport(); uint scale = 2; if(settings["Video/Scale"].text() == "Small" ) scale = 2; if(settings["Video/Scale"].text() == "Medium") scale = 3; if(settings["Video/Scale"].text() == "Large" ) scale = 4; uint windowWidth = 0, windowHeight = 0; bool aspectCorrection = true; if(!fullScreen()) { windowWidth = 326 * scale; windowHeight = 242 * scale; aspectCorrection = settings["Video/AspectCorrection"].boolean(); } else { windowWidth = geometry().width(); windowHeight = geometry().height(); } if(!fullScreen()) setSize({windowWidth, windowHeight}); if(!emulator) { viewport.setGeometry({0, 0, windowWidth, windowHeight}); } else { auto videoSize = emulator->videoSize(windowWidth, windowHeight, aspectCorrection); viewport.setGeometry({ (windowWidth - videoSize.width) / 2, (windowHeight - videoSize.height) / 2, videoSize.width, videoSize.height }); } //clear video area again to ensure entire viewport area has been painted in clearViewport(); } auto Presentation::toggleFullScreen() -> void { if(!fullScreen()) { menuBar.setVisible(false); statusBar.setVisible(false); setResizable(true); setFullScreen(true); if(!input->acquired()) input->acquire(); } else { if(input->acquired()) input->release(); setFullScreen(false); setResizable(false); menuBar.setVisible(true); statusBar.setVisible(settings["UserInterface/ShowStatusBar"].boolean()); } resizeViewport(); } auto Presentation::loadShaders() -> void { auto pathname = locate("Video Shaders/"); if(settings["Video/Driver"].text() == "OpenGL") { for(auto shader : directory::folders(pathname, "*.shader")) { if(videoShaders.objectCount() == 2) videoShaderMenu.append(MenuSeparator()); MenuRadioItem item{&videoShaderMenu}; item.setText(string{shader}.trimRight(".shader/", 1L)).onActivate([=] { settings["Video/Shader"].setValue({pathname, shader}); program->updateVideoShader(); }); videoShaders.append(item); } } if(settings["Video/Shader"].text() == "None") videoShaderNone.setChecked(); if(settings["Video/Shader"].text() == "Blur") videoShaderBlur.setChecked(); for(auto radioItem : videoShaders.objects()) { if(settings["Video/Shader"].text() == string{pathname, radioItem.text(), ".shader/"}) { radioItem.setChecked(); } } }