[olympic/studio] Add new project menu, make file creation open file
	
		
			
	
		
	
	
		
	
		
			Some checks failed
		
		
	
	
		
			
				
	
				Build / build (push) Failing after 2m7s
				
			
		
		
	
	
				
					
				
			
		
			Some checks failed
		
		
	
	Build / build (push) Failing after 2m7s
				
			This commit is contained in:
		| @@ -5,6 +5,7 @@ add_library( | |||||||
| 		filedialogmanager.cpp | 		filedialogmanager.cpp | ||||||
| 		main.cpp | 		main.cpp | ||||||
| 		newmenu.cpp | 		newmenu.cpp | ||||||
|  | 		newproject.cpp | ||||||
| 		projectexplorer.cpp | 		projectexplorer.cpp | ||||||
| 		projecttreemodel.cpp | 		projecttreemodel.cpp | ||||||
| 		studioapp.cpp | 		studioapp.cpp | ||||||
|   | |||||||
| @@ -115,12 +115,12 @@ void NewMenu::drawLastPageButtons(turbine::Context &ctx) noexcept { | |||||||
| } | } | ||||||
|  |  | ||||||
| void NewMenu::finish(turbine::Context &ctx) noexcept { | void NewMenu::finish(turbine::Context &ctx) noexcept { | ||||||
| 	auto const err = m_types[static_cast<std::size_t>(m_selectedType)]->write(ctx, m_itemName); | 	auto const [path, err] = m_types[static_cast<std::size_t>(m_selectedType)]->write(ctx, m_itemName); | ||||||
| 	if (err) { | 	if (err) { | ||||||
| 		oxLogError(err); | 		oxLogError(err); | ||||||
| 		return; | 		return; | ||||||
| 	} | 	} | ||||||
| 	finished.emit(""); | 	finished.emit(path); | ||||||
| 	m_stage = Stage::Closed; | 	m_stage = Stage::Closed; | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										95
									
								
								src/olympic/studio/applib/src/newproject.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								src/olympic/studio/applib/src/newproject.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,95 @@ | |||||||
|  | /* | ||||||
|  |  * Copyright 2016 - 2023 Gary Talent (gary@drinkingtea.net). All rights reserved. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #include <imgui.h> | ||||||
|  |  | ||||||
|  | #include <studio/imguiuitl.hpp> | ||||||
|  | #include <utility> | ||||||
|  |  | ||||||
|  | #include "filedialogmanager.hpp" | ||||||
|  | #include "newproject.hpp" | ||||||
|  |  | ||||||
|  | namespace studio { | ||||||
|  |  | ||||||
|  | NewProject::NewProject(ox::String projectDatadir) noexcept: m_projectDataDir(std::move(projectDatadir)) { | ||||||
|  | 	setTitle(ox::String("New Project")); | ||||||
|  | 	setSize({230, 140}); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void NewProject::open() noexcept { | ||||||
|  | 	m_stage = Stage::Opening; | ||||||
|  | 	m_projectPath = ""; | ||||||
|  | 	m_projectName = ""; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void NewProject::close() noexcept { | ||||||
|  | 	m_stage = Stage::Closed; | ||||||
|  | 	m_open = false; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | bool NewProject::isOpen() const noexcept { | ||||||
|  | 	return m_open; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void NewProject::draw(turbine::Context &ctx) noexcept { | ||||||
|  | 	switch (m_stage) { | ||||||
|  | 		case Stage::Opening: | ||||||
|  | 			ImGui::OpenPopup(title().c_str()); | ||||||
|  | 			m_stage = Stage::NewItemName; | ||||||
|  | 			m_open = true; | ||||||
|  | 			[[fallthrough]]; | ||||||
|  | 		case Stage::NewItemName: | ||||||
|  | 			drawNewProjectName(ctx); | ||||||
|  | 			break; | ||||||
|  | 		case Stage::Closed: | ||||||
|  | 			m_open = false; | ||||||
|  | 			break; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void NewProject::drawNewProjectName(turbine::Context &ctx) noexcept { | ||||||
|  | 	drawWindow(ctx, &m_open, [this, &ctx] { | ||||||
|  | 		ImGui::InputText("Name", m_projectName.data(), m_projectName.cap()); | ||||||
|  | 		ImGui::Text("Path: %s", m_projectPath.c_str()); | ||||||
|  | 		if (ImGui::Button("Browse")) { | ||||||
|  | 			oxLogError(studio::chooseDirectory().moveTo(m_projectPath)); | ||||||
|  | 		} | ||||||
|  | 		drawLastPageButtons(ctx); | ||||||
|  | 	}); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void NewProject::drawFirstPageButtons() noexcept { | ||||||
|  | 	ImGui::SetCursorPosX(ImGui::GetCursorPosX() + ImGui::GetContentRegionAvail().x - 130); | ||||||
|  | 	ImGui::SetCursorPosY(ImGui::GetCursorPosY() + ImGui::GetContentRegionAvail().y - 20); | ||||||
|  | 	auto const btnSz = ImVec2(60, 20); | ||||||
|  | 	if (ImGui::Button("Next", btnSz)) { | ||||||
|  | 		m_stage = Stage::NewItemName; | ||||||
|  | 	} | ||||||
|  | 	ImGui::SameLine(); | ||||||
|  | 	if (ImGui::Button("Cancel", btnSz)) { | ||||||
|  | 		ImGui::CloseCurrentPopup(); | ||||||
|  | 		m_stage = Stage::Closed; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void NewProject::drawLastPageButtons(turbine::Context&) noexcept { | ||||||
|  | 	ImGui::SetCursorPosX(ImGui::GetCursorPosX() + ImGui::GetContentRegionAvail().x - 95); | ||||||
|  | 	ImGui::SetCursorPosY(ImGui::GetCursorPosY() + ImGui::GetContentRegionAvail().y - 20); | ||||||
|  | 	if (ImGui::Button("Finish")) { | ||||||
|  | 		finish(); | ||||||
|  | 	} | ||||||
|  | 	ImGui::SameLine(); | ||||||
|  | 	if (ImGui::Button("Quit")) { | ||||||
|  | 		ImGui::CloseCurrentPopup(); | ||||||
|  | 		m_stage = Stage::Closed; | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void NewProject::finish() noexcept { | ||||||
|  | 	auto projectPath = ox::sfmt("{}/{}", m_projectPath, m_projectName); | ||||||
|  | 	finished.emit(projectPath); | ||||||
|  | 	m_stage = Stage::Closed; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										58
									
								
								src/olympic/studio/applib/src/newproject.hpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										58
									
								
								src/olympic/studio/applib/src/newproject.hpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,58 @@ | |||||||
|  | /* | ||||||
|  |  * Copyright 2016 - 2023 Gary Talent (gary@drinkingtea.net). All rights reserved. | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | #pragma once | ||||||
|  |  | ||||||
|  | #include <ox/claw/claw.hpp> | ||||||
|  | #include <ox/event/signal.hpp> | ||||||
|  | #include <ox/std/string.hpp> | ||||||
|  |  | ||||||
|  | #include <studio/itemmaker.hpp> | ||||||
|  | #include <studio/popup.hpp> | ||||||
|  |  | ||||||
|  | namespace studio { | ||||||
|  |  | ||||||
|  | class NewProject: public studio::Popup { | ||||||
|  | 	public: | ||||||
|  | 		enum class Stage { | ||||||
|  | 			Closed, | ||||||
|  | 			Opening, | ||||||
|  | 			NewItemName, | ||||||
|  | 		}; | ||||||
|  |  | ||||||
|  | 		// emits path parameter | ||||||
|  | 		ox::Signal<ox::Error(ox::StringView)> finished; | ||||||
|  |  | ||||||
|  | 	private: | ||||||
|  | 		Stage m_stage = Stage::Closed; | ||||||
|  | 		ox::String const m_projectDataDir; | ||||||
|  | 		ox::String m_projectPath; | ||||||
|  | 		ox::BString<255> m_projectName; | ||||||
|  | 		ox::Vector<ox::UniquePtr<studio::ItemMaker>> m_types; | ||||||
|  | 		bool m_open = false; | ||||||
|  |  | ||||||
|  | 	public: | ||||||
|  | 		NewProject(ox::String projectDatadir) noexcept; | ||||||
|  |  | ||||||
|  | 		void open() noexcept override; | ||||||
|  |  | ||||||
|  | 		void close() noexcept override; | ||||||
|  |  | ||||||
|  | 		[[nodiscard]] | ||||||
|  | 		bool isOpen() const noexcept override; | ||||||
|  |  | ||||||
|  | 		void draw(turbine::Context &ctx) noexcept override; | ||||||
|  |  | ||||||
|  | 	private: | ||||||
|  | 		void drawNewProjectName(turbine::Context &ctx) noexcept; | ||||||
|  |  | ||||||
|  | 		void drawFirstPageButtons() noexcept; | ||||||
|  |  | ||||||
|  | 		void drawLastPageButtons(turbine::Context &ctx) noexcept; | ||||||
|  |  | ||||||
|  | 		void finish() noexcept; | ||||||
|  |  | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | } | ||||||
| @@ -2,6 +2,8 @@ | |||||||
|  * Copyright 2016 - 2023 Gary Talent (gary@drinkingtea.net). All rights reserved. |  * Copyright 2016 - 2023 Gary Talent (gary@drinkingtea.net). All rights reserved. | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
|  | #include <filesystem> | ||||||
|  |  | ||||||
| #include <imgui.h> | #include <imgui.h> | ||||||
|  |  | ||||||
| #include <keel/media.hpp> | #include <keel/media.hpp> | ||||||
| @@ -42,15 +44,18 @@ StudioUI::StudioUI(turbine::Context &ctx, ox::StringView projectDataDir) noexcep | |||||||
| 		m_ctx(ctx), | 		m_ctx(ctx), | ||||||
| 		m_projectDataDir(projectDataDir), | 		m_projectDataDir(projectDataDir), | ||||||
| 		m_projectExplorer(m_ctx), | 		m_projectExplorer(m_ctx), | ||||||
|  | 		m_newProject(ox::String(projectDataDir)), | ||||||
| 		m_aboutPopup(m_ctx) { | 		m_aboutPopup(m_ctx) { | ||||||
| 	m_projectExplorer.fileChosen.connect(this, &StudioUI::openFile); | 	m_projectExplorer.fileChosen.connect(this, &StudioUI::openFile); | ||||||
|  | 	m_newProject.finished.connect(this, &StudioUI::createOpenProject); | ||||||
|  | 	m_newMenu.finished.connect(this, &StudioUI::openFile); | ||||||
| 	ImGui::GetIO().IniFilename = nullptr; | 	ImGui::GetIO().IniFilename = nullptr; | ||||||
| 	loadModules(); | 	loadModules(); | ||||||
| 	// open project and files | 	// open project and files | ||||||
| 	auto const [config, err] = studio::readConfig<StudioConfig>(keelCtx(m_ctx)); | 	auto const [config, err] = studio::readConfig<StudioConfig>(keelCtx(m_ctx)); | ||||||
| 	m_showProjectExplorer = config.showProjectExplorer; | 	m_showProjectExplorer = config.showProjectExplorer; | ||||||
| 	if (!err) { | 	if (!err) { | ||||||
| 		auto const openProjErr = openProject(config.projectPath); | 		auto const openProjErr = openProjectPath(config.projectPath); | ||||||
| 		if (!openProjErr) { | 		if (!openProjErr) { | ||||||
| 			for (auto const&f: config.openFiles) { | 			for (auto const&f: config.openFiles) { | ||||||
| 				auto openFileErr = openFileActiveTab(f, config.activeTabItemName == f); | 				auto openFileErr = openFileActiveTab(f, config.activeTabItemName == f); | ||||||
| @@ -95,10 +100,14 @@ void StudioUI::handleKeyEvent(turbine::Key key, bool down) noexcept { | |||||||
| 				} | 				} | ||||||
| 				break; | 				break; | ||||||
| 			case turbine::Key::Alpha_N: | 			case turbine::Key::Alpha_N: | ||||||
|  | 				if (turbine::buttonDown(m_ctx, turbine::Key::Mod_Shift)) { | ||||||
|  | 					m_newProject.open(); | ||||||
|  | 				} else { | ||||||
| 					m_newMenu.open(); | 					m_newMenu.open(); | ||||||
|  | 				} | ||||||
| 				break; | 				break; | ||||||
| 			case turbine::Key::Alpha_O: | 			case turbine::Key::Alpha_O: | ||||||
| 				m_taskRunner.add(*ox::make<FileDialogManager>(this, &StudioUI::openProject)); | 				m_taskRunner.add(*ox::make<FileDialogManager>(this, &StudioUI::openProjectPath)); | ||||||
| 				break; | 				break; | ||||||
| 			case turbine::Key::Alpha_Q: | 			case turbine::Key::Alpha_Q: | ||||||
| 				turbine::requestShutdown(m_ctx); | 				turbine::requestShutdown(m_ctx); | ||||||
| @@ -168,8 +177,11 @@ void StudioUI::drawMenu() noexcept { | |||||||
| 			if (ImGui::MenuItem("New...", "Ctrl+N")) { | 			if (ImGui::MenuItem("New...", "Ctrl+N")) { | ||||||
| 				m_newMenu.open(); | 				m_newMenu.open(); | ||||||
| 			} | 			} | ||||||
|  | 			if (ImGui::MenuItem("New Project...", "Ctrl+Shift+N")) { | ||||||
|  | 				m_newProject.open(); | ||||||
|  | 			} | ||||||
| 			if (ImGui::MenuItem("Open Project...", "Ctrl+O")) { | 			if (ImGui::MenuItem("Open Project...", "Ctrl+O")) { | ||||||
| 				m_taskRunner.add(*ox::make<FileDialogManager>(this, &StudioUI::openProject)); | 				m_taskRunner.add(*ox::make<FileDialogManager>(this, &StudioUI::openProjectPath)); | ||||||
| 			} | 			} | ||||||
| 			if (ImGui::MenuItem("Save", "Ctrl+S", false, m_activeEditor && m_activeEditor->unsavedChanges())) { | 			if (ImGui::MenuItem("Save", "Ctrl+S", false, m_activeEditor && m_activeEditor->unsavedChanges())) { | ||||||
| 				m_activeEditor->save(); | 				m_activeEditor->save(); | ||||||
| @@ -317,19 +329,27 @@ void StudioUI::save() noexcept { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
| ox::Error StudioUI::openProject(ox::CRStringView path) noexcept { | ox::Error StudioUI::createOpenProject(ox::CRStringView path) noexcept { | ||||||
|  | 	std::error_code ec; | ||||||
|  | 	std::filesystem::create_directories(toStdStringView(path), ec); | ||||||
|  | 	oxReturnError(OxError(ec.value() != 0, "Could not create project directory")); | ||||||
|  | 	oxReturnError(openProjectPath(path)); | ||||||
|  | 	return m_project->writeAllTypeDescriptors(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | ox::Error StudioUI::openProjectPath(ox::CRStringView path) noexcept { | ||||||
| 	oxRequireM(fs, keel::loadRomFs(path)); | 	oxRequireM(fs, keel::loadRomFs(path)); | ||||||
| 	oxReturnError(keel::setRomFs(keelCtx(m_ctx), std::move(fs))); | 	oxReturnError(keel::setRomFs(keelCtx(m_ctx), std::move(fs))); | ||||||
| 	turbine::setWindowTitle(m_ctx, ox::sfmt("{} - {}", keelCtx(m_ctx).appName, path)); |  | ||||||
| 	m_project = ox::make_unique<studio::Project>(keelCtx(m_ctx), ox::String(path), m_projectDataDir); | 	m_project = ox::make_unique<studio::Project>(keelCtx(m_ctx), ox::String(path), m_projectDataDir); | ||||||
| 	auto sctx = applicationData<studio::StudioContext>(m_ctx); | 	auto const sctx = applicationData<studio::StudioContext>(m_ctx); | ||||||
| 	sctx->project = m_project.get(); | 	sctx->project = m_project.get(); | ||||||
|  | 	turbine::setWindowTitle(m_ctx, ox::sfmt("{} - {}", keelCtx(m_ctx).appName, m_project->projectPath())); | ||||||
| 	m_project->fileAdded.connect(&m_projectExplorer, &ProjectExplorer::refreshProjectTreeModel); | 	m_project->fileAdded.connect(&m_projectExplorer, &ProjectExplorer::refreshProjectTreeModel); | ||||||
| 	m_project->fileDeleted.connect(&m_projectExplorer, &ProjectExplorer::refreshProjectTreeModel); | 	m_project->fileDeleted.connect(&m_projectExplorer, &ProjectExplorer::refreshProjectTreeModel); | ||||||
| 	m_openFiles.clear(); | 	m_openFiles.clear(); | ||||||
| 	m_editors.clear(); | 	m_editors.clear(); | ||||||
| 	studio::editConfig<StudioConfig>(keelCtx(m_ctx), [&](StudioConfig *config) { | 	studio::editConfig<StudioConfig>(keelCtx(m_ctx), [&](StudioConfig *config) { | ||||||
| 		config->projectPath = ox::String(path); | 		config->projectPath = ox::String(m_project->projectPath()); | ||||||
| 		config->openFiles.clear(); | 		config->openFiles.clear(); | ||||||
| 	}); | 	}); | ||||||
| 	return m_projectExplorer.refreshProjectTreeModel(); | 	return m_projectExplorer.refreshProjectTreeModel(); | ||||||
|   | |||||||
| @@ -14,6 +14,7 @@ | |||||||
| #include <studio/task.hpp> | #include <studio/task.hpp> | ||||||
| #include "aboutpopup.hpp" | #include "aboutpopup.hpp" | ||||||
| #include "newmenu.hpp" | #include "newmenu.hpp" | ||||||
|  | #include "newproject.hpp" | ||||||
| #include "projectexplorer.hpp" | #include "projectexplorer.hpp" | ||||||
| #include "projecttreemodel.hpp" | #include "projecttreemodel.hpp" | ||||||
|  |  | ||||||
| @@ -36,9 +37,11 @@ class StudioUI: public ox::SignalHandler { | |||||||
| 		studio::BaseEditor *m_activeEditor = nullptr; | 		studio::BaseEditor *m_activeEditor = nullptr; | ||||||
| 		studio::BaseEditor *m_activeEditorUpdatePending = nullptr; | 		studio::BaseEditor *m_activeEditorUpdatePending = nullptr; | ||||||
| 		NewMenu m_newMenu; | 		NewMenu m_newMenu; | ||||||
|  | 		NewProject m_newProject; | ||||||
| 		AboutPopup m_aboutPopup; | 		AboutPopup m_aboutPopup; | ||||||
| 		ox::Array<studio::Popup*, 2> const m_popups = { | 		ox::Array<studio::Popup*, 2> const m_popups = { | ||||||
| 			&m_newMenu, | 			&m_newMenu, | ||||||
|  | 			&m_newProject, | ||||||
| 			&m_aboutPopup | 			&m_aboutPopup | ||||||
| 		}; | 		}; | ||||||
| 		bool m_showProjectExplorer = true; | 		bool m_showProjectExplorer = true; | ||||||
| @@ -79,7 +82,9 @@ class StudioUI: public ox::SignalHandler { | |||||||
|  |  | ||||||
| 		void save() noexcept; | 		void save() noexcept; | ||||||
|  |  | ||||||
| 		ox::Error openProject(ox::CRStringView path) noexcept; | 		ox::Error createOpenProject(ox::CRStringView path) noexcept; | ||||||
|  |  | ||||||
|  | 		ox::Error openProjectPath(ox::CRStringView path) noexcept; | ||||||
|  |  | ||||||
| 		ox::Error openFile(ox::CRStringView path) noexcept; | 		ox::Error openFile(ox::CRStringView path) noexcept; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -24,7 +24,14 @@ class ItemMaker { | |||||||
| 			fileExt(pFileExt) { | 			fileExt(pFileExt) { | ||||||
| 		} | 		} | ||||||
| 		virtual ~ItemMaker() noexcept = default; | 		virtual ~ItemMaker() noexcept = default; | ||||||
| 		virtual ox::Error write(turbine::Context &ctx, ox::CRStringView pName) const noexcept = 0; |  | ||||||
|  | 		/** | ||||||
|  | 		 * Returns path of the file created. | ||||||
|  | 		 * @param ctx | ||||||
|  | 		 * @param pName | ||||||
|  | 		 * @return path of file or error in Result | ||||||
|  | 		 */ | ||||||
|  | 		virtual ox::Result<ox::String> write(turbine::Context &ctx, ox::CRStringView pName) const noexcept = 0; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| template<typename T> | template<typename T> | ||||||
| @@ -61,11 +68,12 @@ class ItemMakerT: public ItemMaker { | |||||||
| 			 m_item(std::move(pItem)), | 			 m_item(std::move(pItem)), | ||||||
| 			 m_fmt(pFmt) { | 			 m_fmt(pFmt) { | ||||||
| 		} | 		} | ||||||
| 		ox::Error write(turbine::Context &ctx, ox::CRStringView pName) const noexcept override { | 		ox::Result<ox::String> write(turbine::Context &ctx, ox::CRStringView pName) const noexcept override { | ||||||
| 			auto const path = ox::sfmt("/{}/{}.{}", parentDir, pName, fileExt); | 			auto const path = ox::sfmt("/{}/{}.{}", parentDir, pName, fileExt); | ||||||
| 			auto sctx = turbine::applicationData<studio::StudioContext>(ctx); | 			auto sctx = turbine::applicationData<studio::StudioContext>(ctx); | ||||||
| 			keel::createUuidMapping(keelCtx(ctx), path, ox::UUID::generate().unwrap()); | 			keel::createUuidMapping(keelCtx(ctx), path, ox::UUID::generate().unwrap()); | ||||||
| 			return sctx->project->writeObj(path, m_item, m_fmt); | 			oxReturnError(sctx->project->writeObj(path, m_item, m_fmt)); | ||||||
|  | 			return path; | ||||||
| 		} | 		} | ||||||
| }; | }; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -50,6 +50,7 @@ class Project { | |||||||
| 		keel::Context &m_ctx; | 		keel::Context &m_ctx; | ||||||
| 		ox::String m_path; | 		ox::String m_path; | ||||||
| 		ox::String m_projectDataDir; | 		ox::String m_projectDataDir; | ||||||
|  | 		ox::String m_typeDescPath; | ||||||
| 		mutable keel::TypeStore m_typeStore; | 		mutable keel::TypeStore m_typeStore; | ||||||
| 		ox::FileSystem &m_fs; | 		ox::FileSystem &m_fs; | ||||||
| 		ox::HashMap<ox::String, ox::Vector<ox::String>> m_fileExtFileMap; | 		ox::HashMap<ox::String, ox::Vector<ox::String>> m_fileExtFileMap; | ||||||
| @@ -59,6 +60,9 @@ class Project { | |||||||
|  |  | ||||||
| 		ox::Error create() noexcept; | 		ox::Error create() noexcept; | ||||||
|  |  | ||||||
|  | 		[[nodiscard]] | ||||||
|  | 		ox::String const&projectPath() const noexcept; | ||||||
|  |  | ||||||
| 		[[nodiscard]] | 		[[nodiscard]] | ||||||
| 		ox::FileSystem *romFs() noexcept; | 		ox::FileSystem *romFs() noexcept; | ||||||
|  |  | ||||||
| @@ -87,6 +91,8 @@ class Project { | |||||||
| 		[[nodiscard]] | 		[[nodiscard]] | ||||||
| 		ox::Vector<ox::String> const&fileList(ox::CRStringView ext) noexcept; | 		ox::Vector<ox::String> const&fileList(ox::CRStringView ext) noexcept; | ||||||
|  |  | ||||||
|  | 		ox::Error writeAllTypeDescriptors() noexcept; | ||||||
|  |  | ||||||
| 	private: | 	private: | ||||||
| 		void buildFileIndex() noexcept; | 		void buildFileIndex() noexcept; | ||||||
|  |  | ||||||
| @@ -123,21 +129,10 @@ ox::Error Project::writeObj(ox::CRStringView path, T const&obj, ox::ClawFormat f | |||||||
| 	if (m_typeStore.get<T>().error) { | 	if (m_typeStore.get<T>().error) { | ||||||
| 		oxReturnError(ox::buildTypeDef(&m_typeStore, &obj)); | 		oxReturnError(ox::buildTypeDef(&m_typeStore, &obj)); | ||||||
| 	} | 	} | ||||||
| 	// write out type store |  | ||||||
| 	auto const descPath = ox::sfmt("/{}/type_descriptors", m_projectDataDir); |  | ||||||
| 	oxRequire(desc, m_typeStore.get<T>()); | 	oxRequire(desc, m_typeStore.get<T>()); | ||||||
| 	auto const descExists = m_fs.stat(ox::sfmt("{}/{}", descPath, buildTypeId(*desc))).error != 0; | 	auto const descExists = m_fs.stat(ox::sfmt("{}/{}", m_typeDescPath, buildTypeId(*desc))).error != 0; | ||||||
| 	if (!descExists || ox::defines::Debug) { | 	if (!descExists || ox::defines::Debug) { | ||||||
| 		// write all descriptors because we don't know which types T depends on | 		oxReturnError(writeAllTypeDescriptors()); | ||||||
| 		oxReturnError(mkdir(descPath)); |  | ||||||
| 		for (auto const &t: m_typeStore.typeList()) { |  | ||||||
| 			oxRequireM(typeOut, ox::writeClaw(*t, ox::ClawFormat::Organic)); |  | ||||||
| 			// replace garbage last character with new line |  | ||||||
| 			*typeOut.back().value = '\n'; |  | ||||||
| 			// write to FS |  | ||||||
| 			auto const typePath = ox::sfmt("/{}/{}", descPath, buildTypeId(*t)); |  | ||||||
| 			oxReturnError(writeBuff(typePath, typeOut)); |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
| 	oxReturnError(keel::setAsset(m_ctx, path, obj)); | 	oxReturnError(keel::setAsset(m_ctx, path, obj)); | ||||||
| 	fileUpdated.emit(path); | 	fileUpdated.emit(path); | ||||||
|   | |||||||
| @@ -30,6 +30,7 @@ Project::Project(keel::Context &ctx, ox::String path, ox::CRStringView projectDa | |||||||
| 	m_ctx(ctx), | 	m_ctx(ctx), | ||||||
| 	m_path(std::move(path)), | 	m_path(std::move(path)), | ||||||
| 	m_projectDataDir(projectDataDir), | 	m_projectDataDir(projectDataDir), | ||||||
|  | 	m_typeDescPath(ox::sfmt("/{}/type_descriptors", m_projectDataDir)), | ||||||
| 	m_typeStore(*m_ctx.rom, ox::sfmt("/{}/type_descriptors", projectDataDir)), | 	m_typeStore(*m_ctx.rom, ox::sfmt("/{}/type_descriptors", projectDataDir)), | ||||||
| 	m_fs(*m_ctx.rom) { | 	m_fs(*m_ctx.rom) { | ||||||
| 	oxTracef("studio", "Project: {}", m_path); | 	oxTracef("studio", "Project: {}", m_path); | ||||||
| @@ -43,6 +44,10 @@ ox::Error Project::create() noexcept { | |||||||
| 	return OxError(static_cast<ox::ErrorCode>(ec.value()), "PassThroughFS: mkdir failed"); | 	return OxError(static_cast<ox::ErrorCode>(ec.value()), "PassThroughFS: mkdir failed"); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | ox::String const&Project::projectPath() const noexcept { | ||||||
|  | 	return m_path; | ||||||
|  | } | ||||||
|  |  | ||||||
| ox::FileSystem *Project::romFs() noexcept { | ox::FileSystem *Project::romFs() noexcept { | ||||||
| 	return &m_fs; | 	return &m_fs; | ||||||
| } | } | ||||||
| @@ -65,6 +70,20 @@ ox::Vector<ox::String> const&Project::fileList(ox::CRStringView ext) noexcept { | |||||||
| 	return m_fileExtFileMap[ext]; | 	return m_fileExtFileMap[ext]; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | ox::Error Project::writeAllTypeDescriptors() noexcept { | ||||||
|  | 	// write all descriptors because we don't know which types T depends on | ||||||
|  | 	oxReturnError(mkdir(m_typeDescPath)); | ||||||
|  | 	for (auto const &t: m_typeStore.typeList()) { | ||||||
|  | 		oxRequireM(typeOut, ox::writeClaw(*t, ox::ClawFormat::Organic)); | ||||||
|  | 		// replace garbage last character with new line | ||||||
|  | 		*typeOut.back().value = '\n'; | ||||||
|  | 		// write to FS | ||||||
|  | 		auto const typePath = ox::sfmt("{}/{}", m_typeDescPath, buildTypeId(*t)); | ||||||
|  | 		oxReturnError(writeBuff(typePath, typeOut)); | ||||||
|  | 	} | ||||||
|  | 	return {}; | ||||||
|  | } | ||||||
|  |  | ||||||
| void Project::buildFileIndex() noexcept { | void Project::buildFileIndex() noexcept { | ||||||
| 	auto [files, err] = listFiles(); | 	auto [files, err] = listFiles(); | ||||||
| 	if (err) { | 	if (err) { | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user