#include "Space.hpp" #include "ViewManager.hpp" #include "ResourceBrowser.hpp" #include "ParticleEditor.hpp" #include "ModelViewer.hpp" #include "InformationCenter.hpp" #include "GameMode.hpp" #include "icons/icons.hpp" #include "specter/Menu.hpp" namespace urde { static logvisor::Module Log("URDE::Space"); Space::Space(ViewManager& vm, Class cls, Space* parent) : m_spaceMenuNode(*this), m_spaceSelectBind(*this), m_vm(vm), m_class(cls), m_parent(parent) {} specter::View* Space::buildSpaceView(specter::ViewResources& res) { if (usesToolbar()) { m_spaceView.reset( new specter::Space(res, *m_parent->basisView(), *this, specter::Toolbar::Position::Bottom, toolbarUnits())); specter::View* sview = buildContentView(res); m_spaceView->setContentView(sview); specter::Toolbar& tb = *m_spaceView->toolbar(); specter::Icon* classIcon = SpaceMenuNode::LookupClassIcon(m_class); const zeus::CColor* classColor = SpaceMenuNode::LookupClassColor(m_class); m_spaceSelectButton.reset(new specter::Button(res, tb, &m_spaceSelectBind, "", classIcon, specter::Button::Style::Block, classColor ? *classColor : zeus::skWhite)); tb.push_back(m_spaceSelectButton.get(), 0); buildToolbarView(res, tb); return m_spaceView.get(); } else { m_spaceView.reset(new specter::Space(res, *m_parent->basisView(), *this, specter::Toolbar::Position::None, 0)); specter::View* sview = buildContentView(res); m_spaceView->setContentView(sview); return m_spaceView.get(); } } std::vector<Space::SpaceMenuNode::SubNodeData> Space::SpaceMenuNode::s_subNodeDats = { {Class::ResourceBrowser, "Resource Browser", GetIcon(SpaceIcon::ResourceBrowser), {0.0f, 1.0f, 0.0f, 1.0f}}, {Class::EffectEditor, "Effect Editor", GetIcon(SpaceIcon::ParticleEditor), {1.0f, 0.5f, 0.0f, 1.0f}}, {Class::ModelViewer, "Model Viewer", GetIcon(SpaceIcon::ModelViewer), {0.95f, 0.95f, 0.95f, 1.0f}}, {Class::InformationCenter, "Information Center", GetIcon(SpaceIcon::InformationCenter), {0.0f, 1.0f, 1.0f, 1.0f}}, {Class::GameMode, "Game Mode", GetIcon(SpaceIcon::GameMode), {}}}; std::string Space::SpaceMenuNode::s_text = "Space Types"; template <typename Key> static void RecurseTranslations(ViewManager& vm, std::vector<Space::SpaceMenuNode::SubNodeData>::iterator it) { it->m_text = vm.translate<Key>(); } template <typename Key, typename Key2, typename... RemKeys> static void RecurseTranslations(ViewManager& vm, std::vector<Space::SpaceMenuNode::SubNodeData>::iterator it) { RecurseTranslations<Key>(vm, it); RecurseTranslations<Key2, RemKeys...>(vm, ++it); } void Space::SpaceMenuNode::InitializeStrings(ViewManager& vm) { s_text = vm.translate<locale::space_types>(); RecurseTranslations<locale::resource_browser, locale::effect_editor, locale::model_viewer, locale::information_center, locale::game_mode>(vm, s_subNodeDats.begin()); } std::unique_ptr<specter::View> Space::SpaceSelectBind::buildMenu(const specter::Button* button) { return std::unique_ptr<specter::View>( new specter::Menu(m_space.m_vm.rootView().viewRes(), *m_space.m_spaceView, &m_space.m_spaceMenuNode)); } specter::View* RootSpace::buildSpaceView(specter::ViewResources& res) { specter::View* newRoot = buildContentView(res); m_vm.RootSpaceViewBuilt(newRoot); return newRoot; } specter::View* RootSpace::basisView() { return &m_vm.rootView(); } specter::View* SplitSpace::buildContentView(specter::ViewResources& res) { int clearance = res.pixelFactor() * SPECTER_TOOLBAR_GAUGE; m_splitView.reset( new specter::SplitView(res, *m_parent->basisView(), this, m_state.axis, m_state.split, clearance, clearance)); if (m_slots[0]) m_splitView->setContentView(0, m_slots[0]->buildSpaceView(res)); if (m_slots[1]) m_splitView->setContentView(1, m_slots[1]->buildSpaceView(res)); return m_splitView.get(); } void SplitSpace::setChildSlot(unsigned slot, std::unique_ptr<Space>&& space) { if (slot > 1) Log.report(logvisor::Fatal, fmt("invalid slot {} for SplitView"), slot); m_slots[slot] = std::move(space); m_slots[slot]->m_parent = this; } void SplitSpace::joinViews(specter::SplitView* thisSplit, int thisSlot, specter::SplitView* otherSplit, int otherSlot) { if (thisSplit == otherSplit) { SplitSpace* thisSS = m_slots[thisSlot]->castToSplitSpace(); if (thisSS) { int ax = thisSS->m_state.axis == specter::SplitView::Axis::Horizontal ? 1 : 0; const boo::SWindowRect& thisRect = m_splitView->subRect(); const boo::SWindowRect& subRect = thisSS->m_splitView->subRect(); int splitPx = subRect.location[ax] + subRect.size[ax] * thisSS->m_state.split - thisRect.location[ax]; thisSS->m_state.split = splitPx / float(thisRect.size[ax]); } m_parent->exchangeSpaceSplitJoin(this, std::move(m_slots[thisSlot])); m_vm.BuildSpaceViews(); } else { for (int i = 0; i < 2; ++i) { SplitSpace* otherSS = m_slots[i]->castToSplitSpace(); if (otherSS && otherSS->m_splitView.get() == otherSplit) { int ax = m_state.axis == specter::SplitView::Axis::Horizontal ? 1 : 0; const boo::SWindowRect& thisRect = m_splitView->subRect(); const boo::SWindowRect& subRect = otherSS->m_splitView->subRect(); int splitPx = subRect.location[ax] + subRect.size[ax] * otherSS->m_state.split - thisRect.location[ax]; m_state.split = splitPx / float(thisRect.size[ax]); exchangeSpaceSplitJoin(otherSS, std::move(otherSS->m_slots[otherSlot ^ 1])); m_vm.BuildSpaceViews(); break; } } } } specter::ISplitSpaceController* Space::spaceSplit(specter::SplitView::Axis axis, int thisSlot) { if (m_parent) { /* Reject split operations with insufficient clearance */ int clearance = m_vm.m_viewResources.pixelFactor() * SPECTER_TOOLBAR_GAUGE; if (axis == specter::SplitView::Axis::Horizontal) { if (m_spaceView->subRect().size[1] <= clearance) return nullptr; } else { if (m_spaceView->subRect().size[0] <= clearance) return nullptr; } SplitSpace* ss = new SplitSpace(m_vm, m_parent, axis); ss->setChildSlot(thisSlot, m_parent->exchangeSpaceSplitJoin(this, std::unique_ptr<Space>(ss))); ss->setChildSlot(thisSlot ^ 1, std::unique_ptr<Space>(copy(ss))); m_vm.BuildSpaceViews(); return ss; } return nullptr; } std::unique_ptr<Space> RootSpace::exchangeSpaceSplitJoin(Space* removeSpace, std::unique_ptr<Space>&& keepSpace) { std::unique_ptr<Space> ret = std::move(keepSpace); if (removeSpace == m_spaceTree.get()) { m_spaceTree.swap(ret); m_spaceTree->m_parent = this; } else Log.report(logvisor::Fatal, fmt("RootSpace::exchangeSpaceSplitJoin() failure")); return ret; } std::unique_ptr<Space> SplitSpace::exchangeSpaceSplitJoin(Space* removeSpace, std::unique_ptr<Space>&& keepSpace) { std::unique_ptr<Space> ret = std::move(keepSpace); if (removeSpace == m_slots[0].get()) { m_slots[0].swap(ret); m_slots[0]->m_parent = this; } else if (removeSpace == m_slots[1].get()) { m_slots[1].swap(ret); m_slots[1]->m_parent = this; } else Log.report(logvisor::Fatal, fmt("SplitSpace::exchangeSpaceSplitJoin() failure")); return ret; } template <class Reader> static Space* BuildNewSpace(ViewManager& vm, Space::Class cls, Space* parent, Reader& r) { using Class = Space::Class; switch (cls) { case Class::SplitSpace: return new SplitSpace(vm, parent, r); case Class::ResourceBrowser: return new ResourceBrowser(vm, parent, r); case Class::EffectEditor: return new EffectEditor(vm, parent, r); case Class::ModelViewer: return new ModelViewer(vm, parent, r); case Class::InformationCenter: return new InformationCenter(vm, parent, r); case Class::GameMode: return new GameMode(vm, parent, r); default: break; } return nullptr; } void Space::saveState(athena::io::IStreamWriter& w) const { w.writeUint32Big(atUint32(m_class)); spaceState().write(w); } void Space::saveState(athena::io::YAMLDocWriter& w) const { w.writeUint32("class", atUint32(m_class)); spaceState().write(w); } Space* Space::NewSpaceFromConfigStream(ViewManager& vm, Space* parent, ConfigReader& r) { Class cls = Class(r.readUint32("class")); return BuildNewSpace(vm, cls, parent, r); } RootSpace* Space::NewRootSpaceFromConfigStream(ViewManager& vm, ConfigReader& r) { Class cls = Class(r.readUint32("class")); if (cls != Class::RootSpace) return nullptr; return new RootSpace(vm, r); } void Space::SpaceMenuNode::SubNode::activated(const boo::SWindowCoord& coord) { std::unique_ptr<Space> newSpace; switch (m_data.m_cls) { case Class::InformationCenter: if (m_space.cls() == Class::InformationCenter) newSpace.reset(new InformationCenter(m_space.m_parent->m_vm, m_space.m_parent)); break; case Class::EffectEditor: if (m_space.cls() == Class::EffectEditor) newSpace.reset(new EffectEditor(m_space.m_parent->m_vm, m_space.m_parent)); break; case Class::ResourceBrowser: if (m_space.cls() == Class::ResourceBrowser) newSpace.reset(new ResourceBrowser(m_space.m_parent->m_vm, m_space.m_parent)); break; case Class::ModelViewer: if (m_space.cls() == Class::ModelViewer) newSpace.reset(new ModelViewer(m_space.m_parent->m_vm, m_space.m_parent)); break; case Class::GameMode: if (m_space.cls() == Class::GameMode) newSpace.reset(new GameMode(m_space.m_parent->m_vm, m_space.m_parent)); break; default: break; } if (newSpace) { Space* parent = m_space.m_parent; m_space.m_parent->exchangeSpaceSplitJoin(&m_space, std::move(newSpace)); parent->m_vm.BuildSpaceViews(); } } } // namespace urde