Squashed 'deps/nostalgia/' changes from 1af4da43..1cc1d561
1cc1d561 [studio] Add a file explorer to NewMenu to choose where new files go d15a0df7 [studio] Make reusable FileTreeModel e1282b6b [studio] Fix build 5fe7c14c [nostalgia/sample_project] Rename TileSheet files using new file ext 42165ba2 [nostalgia/gfx] Change default file extension for TileSheets to nts git-subtree-dir: deps/nostalgia git-subtree-split: 1cc1d561e2edfd454335b0cb4d85332e8588237d
This commit is contained in:
Binary file not shown.
352
sample_project/TileSheets/Chester.nts
Normal file
352
sample_project/TileSheets/Chester.nts
Normal file
@@ -0,0 +1,352 @@
|
|||||||
|
K1;5667c759-7ba1-470a-8860-72f0720dc58c;O1;net.drinkingtea.nostalgia.gfx.TileSheet;5;{
|
||||||
|
"bpp" : 4,
|
||||||
|
"defaultPalette" : "uuid://14fc3dd8-42ff-4bf9-81f1-a010cc5ac251",
|
||||||
|
"idIt" : 7,
|
||||||
|
"subsheet" :
|
||||||
|
{
|
||||||
|
"columns" : -1,
|
||||||
|
"name" : "Root",
|
||||||
|
"rows" : -1,
|
||||||
|
"subsheets" :
|
||||||
|
[
|
||||||
|
{
|
||||||
|
"columns" : 1,
|
||||||
|
"id" : 5,
|
||||||
|
"name" : "Blank",
|
||||||
|
"pixels" :
|
||||||
|
[
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
],
|
||||||
|
"rows" : 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"columns" : 2,
|
||||||
|
"id" : 6,
|
||||||
|
"name" : "Dirt",
|
||||||
|
"pixels" :
|
||||||
|
[
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
2,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
2,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
2,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
2,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
2,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
2,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
2,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
2,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
2,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
2,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1,
|
||||||
|
1
|
||||||
|
],
|
||||||
|
"rows" : 2
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
BIN
sample_project/TileSheets/SC9K_Logo.nts
Normal file
BIN
sample_project/TileSheets/SC9K_Logo.nts
Normal file
Binary file not shown.
@@ -13,6 +13,7 @@ constexpr auto TileHeight = 8;
|
|||||||
constexpr auto PixelsPerTile = TileWidth * TileHeight;
|
constexpr auto PixelsPerTile = TileWidth * TileHeight;
|
||||||
|
|
||||||
constexpr ox::StringLiteral FileExt_ng("ng");
|
constexpr ox::StringLiteral FileExt_ng("ng");
|
||||||
|
constexpr ox::StringLiteral FileExt_nts("nts");
|
||||||
constexpr ox::StringLiteral FileExt_npal("npal");
|
constexpr ox::StringLiteral FileExt_npal("npal");
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@@ -14,14 +14,14 @@ namespace nostalgia::gfx {
|
|||||||
static class: public studio::Module {
|
static class: public studio::Module {
|
||||||
ox::Vector<studio::EditorMaker> editors(studio::StudioContext &ctx) const noexcept final {
|
ox::Vector<studio::EditorMaker> editors(studio::StudioContext &ctx) const noexcept final {
|
||||||
return {
|
return {
|
||||||
studio::editorMaker<TileSheetEditorImGui>(ctx, FileExt_ng),
|
studio::editorMaker<TileSheetEditorImGui>(ctx, {FileExt_ng, FileExt_nts}),
|
||||||
studio::editorMaker<PaletteEditorImGui>(ctx, FileExt_npal),
|
studio::editorMaker<PaletteEditorImGui>(ctx, FileExt_npal),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
ox::Vector<ox::UPtr<studio::ItemMaker>> itemMakers(studio::StudioContext&) const noexcept final {
|
ox::Vector<ox::UPtr<studio::ItemMaker>> itemMakers(studio::StudioContext&) const noexcept final {
|
||||||
ox::Vector<ox::UniquePtr<studio::ItemMaker>> out;
|
ox::Vector<ox::UniquePtr<studio::ItemMaker>> out;
|
||||||
out.emplace_back(ox::make<studio::ItemMakerT<TileSheet>>("Tile Sheet", "TileSheets", FileExt_ng));
|
out.emplace_back(ox::make<studio::ItemMakerT<TileSheet>>("Tile Sheet", "TileSheets", FileExt_nts));
|
||||||
out.emplace_back(ox::make<studio::ItemMakerT<Palette>>("Palette", "Palettes", FileExt_npal, Palette{
|
out.emplace_back(ox::make<studio::ItemMakerT<Palette>>("Palette", "Palettes", FileExt_npal, Palette{
|
||||||
.colorNames = {},
|
.colorNames = {},
|
||||||
.pages = {{"Page 1", ox::Vector<PaletteColor>{}}},
|
.pages = {{"Page 1", ox::Vector<PaletteColor>{}}},
|
||||||
|
@@ -9,7 +9,6 @@ add_library(
|
|||||||
newmenu.cpp
|
newmenu.cpp
|
||||||
newproject.cpp
|
newproject.cpp
|
||||||
projectexplorer.cpp
|
projectexplorer.cpp
|
||||||
projecttreemodel.cpp
|
|
||||||
studioapp.cpp
|
studioapp.cpp
|
||||||
)
|
)
|
||||||
target_compile_definitions(
|
target_compile_definitions(
|
||||||
|
@@ -12,9 +12,8 @@
|
|||||||
|
|
||||||
namespace studio {
|
namespace studio {
|
||||||
|
|
||||||
NewMenu::NewMenu() noexcept {
|
NewMenu::NewMenu(keel::Context &kctx) noexcept: m_kctx{kctx} {
|
||||||
setTitle("New Item");
|
setTitle("New Item");
|
||||||
setSize({280, 180});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void NewMenu::open() noexcept {
|
void NewMenu::open() noexcept {
|
||||||
@@ -23,6 +22,11 @@ void NewMenu::open() noexcept {
|
|||||||
m_itemName = "";
|
m_itemName = "";
|
||||||
m_typeName = "";
|
m_typeName = "";
|
||||||
m_path = "";
|
m_path = "";
|
||||||
|
m_explorer.setModel(buildFileTreeModel(
|
||||||
|
m_explorer,
|
||||||
|
[](ox::StringViewCR, ox::FileStat const&s) {
|
||||||
|
return s.fileType == ox::FileType::Directory;
|
||||||
|
}).or_value(ox::UPtr<FileTreeModel>{}));
|
||||||
}
|
}
|
||||||
|
|
||||||
void NewMenu::openPath(ox::StringParam path) noexcept {
|
void NewMenu::openPath(ox::StringParam path) noexcept {
|
||||||
@@ -49,8 +53,8 @@ void NewMenu::draw(StudioContext &sctx) noexcept {
|
|||||||
case Stage::NewItemType:
|
case Stage::NewItemType:
|
||||||
drawNewItemType(sctx);
|
drawNewItemType(sctx);
|
||||||
break;
|
break;
|
||||||
case Stage::NewItemName:
|
case Stage::NewItemPath:
|
||||||
drawNewItemName(sctx);
|
drawNewItemPath(sctx);
|
||||||
break;
|
break;
|
||||||
case Stage::NewItemTemplate:
|
case Stage::NewItemTemplate:
|
||||||
drawNewItemTemplate(sctx);
|
drawNewItemTemplate(sctx);
|
||||||
@@ -79,32 +83,45 @@ void NewMenu::installItemTemplate(ox::UPtr<ItemTemplate> &tmplt) noexcept {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void NewMenu::drawNewItemType(StudioContext const&sctx) noexcept {
|
void NewMenu::drawNewItemType(StudioContext const&sctx) noexcept {
|
||||||
|
setSize({280, 180});
|
||||||
drawWindow(sctx.tctx, m_open, [this] {
|
drawWindow(sctx.tctx, m_open, [this] {
|
||||||
ig::ListBox("Item Type", [&](size_t const i) -> ox::CStringView {
|
ig::ListBox("Item Type", [&](size_t const i) -> ox::CStringView {
|
||||||
return m_types[i]->typeDisplayName();
|
return m_types[i]->typeDisplayName();
|
||||||
}, m_types.size(), m_selectedType, {200, 100});
|
}, m_types.size(), m_selectedType, {200, 100});
|
||||||
auto const&im = *m_types[m_selectedType];
|
auto const&im = *m_types[m_selectedType];
|
||||||
drawFirstPageButtons(im.itemTemplates().size() == 1 ?
|
drawFirstPageButtons(im.itemTemplates().size() == 1 ?
|
||||||
Stage::NewItemName : Stage::NewItemTemplate);
|
Stage::NewItemPath : Stage::NewItemTemplate);
|
||||||
|
if (m_stage == Stage::NewItemPath) {
|
||||||
|
if (m_path.len() == 0) {
|
||||||
|
m_path = im.defaultPath();
|
||||||
|
}
|
||||||
|
std::ignore = m_explorer.setSelectedPath(m_path);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
void NewMenu::drawNewItemTemplate(StudioContext &sctx) noexcept {
|
void NewMenu::drawNewItemTemplate(StudioContext const&sctx) noexcept {
|
||||||
|
setSize({280, 180});
|
||||||
drawWindow(sctx.tctx, m_open, [this] {
|
drawWindow(sctx.tctx, m_open, [this] {
|
||||||
auto const&templates =
|
auto const&templates =
|
||||||
m_types[m_selectedType]->itemTemplates();
|
m_types[m_selectedType]->itemTemplates();
|
||||||
ig::ListBox("Template", [&](size_t const i) -> ox::CStringView {
|
ig::ListBox("Template", [&](size_t const i) -> ox::CStringView {
|
||||||
return templates[i]->name();
|
return templates[i]->name();
|
||||||
}, templates.size(), m_selectedTemplate, {200, 100});
|
}, templates.size(), m_selectedTemplate, {200, 100});
|
||||||
drawButtons(Stage::NewItemType, Stage::NewItemName);
|
drawButtons(Stage::NewItemType, Stage::NewItemPath);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
void NewMenu::drawNewItemName(StudioContext &sctx) noexcept {
|
void NewMenu::drawNewItemPath(StudioContext &sctx) noexcept {
|
||||||
|
setSize({380, 340});
|
||||||
drawWindow(sctx.tctx, m_open, [this, &sctx] {
|
drawWindow(sctx.tctx, m_open, [this, &sctx] {
|
||||||
if (m_selectedType < m_types.size()) {
|
if (m_selectedType < m_types.size()) {
|
||||||
ig::InputText("Name", m_itemName);
|
ig::InputText("Name", m_itemName);
|
||||||
}
|
}
|
||||||
|
ImGui::NewLine();
|
||||||
|
ImGui::Text("Path");
|
||||||
|
auto const vp = ImGui::GetContentRegionAvail();
|
||||||
|
m_explorer.draw(sctx, {vp.x, vp.y - 50});
|
||||||
drawLastPageButtons(sctx);
|
drawLastPageButtons(sctx);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -165,8 +182,10 @@ void NewMenu::finish(StudioContext &sctx) noexcept {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
auto const&im = *m_types[m_selectedType];
|
auto const&im = *m_types[m_selectedType];
|
||||||
auto const path = m_path.len() ?
|
if (auto p = m_explorer.selectedPath()) {
|
||||||
im.itemPath(m_itemName, m_path) : im.itemPath(m_itemName);
|
m_path = std::move(*p);
|
||||||
|
}
|
||||||
|
auto const path = sfmt("{}/{}.{}", m_path, m_itemName, im.fileExt());
|
||||||
if (sctx.project->exists(path)) {
|
if (sctx.project->exists(path)) {
|
||||||
oxLogError(ox::Error{1, "New file error: file already exists"});
|
oxLogError(ox::Error{1, "New file error: file already exists"});
|
||||||
return;
|
return;
|
||||||
|
@@ -8,6 +8,7 @@
|
|||||||
#include <ox/event/signal.hpp>
|
#include <ox/event/signal.hpp>
|
||||||
#include <ox/std/string.hpp>
|
#include <ox/std/string.hpp>
|
||||||
|
|
||||||
|
#include <studio/filetreemodel.hpp>
|
||||||
#include <studio/itemmaker.hpp>
|
#include <studio/itemmaker.hpp>
|
||||||
#include <studio/popup.hpp>
|
#include <studio/popup.hpp>
|
||||||
|
|
||||||
@@ -19,7 +20,7 @@ class NewMenu final: public Popup {
|
|||||||
Closed,
|
Closed,
|
||||||
Opening,
|
Opening,
|
||||||
NewItemType,
|
NewItemType,
|
||||||
NewItemName,
|
NewItemPath,
|
||||||
NewItemTemplate,
|
NewItemTemplate,
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -28,16 +29,18 @@ class NewMenu final: public Popup {
|
|||||||
|
|
||||||
private:
|
private:
|
||||||
Stage m_stage = Stage::Closed;
|
Stage m_stage = Stage::Closed;
|
||||||
|
keel::Context &m_kctx;
|
||||||
ox::String m_typeName;
|
ox::String m_typeName;
|
||||||
ox::IString<255> m_itemName;
|
ox::IString<255> m_itemName;
|
||||||
ox::String m_path;
|
ox::String m_path;
|
||||||
ox::Vector<ox::UPtr<studio::ItemMaker>> m_types;
|
ox::Vector<ox::UPtr<ItemMaker>> m_types;
|
||||||
|
FileExplorer m_explorer{m_kctx};
|
||||||
size_t m_selectedType = 0;
|
size_t m_selectedType = 0;
|
||||||
size_t m_selectedTemplate = 0;
|
size_t m_selectedTemplate = 0;
|
||||||
bool m_open = false;
|
bool m_open = false;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
NewMenu() noexcept;
|
NewMenu(keel::Context &kctx) noexcept;
|
||||||
|
|
||||||
void openPath(ox::StringParam path) noexcept;
|
void openPath(ox::StringParam path) noexcept;
|
||||||
|
|
||||||
@@ -72,9 +75,9 @@ class NewMenu final: public Popup {
|
|||||||
private:
|
private:
|
||||||
void drawNewItemType(StudioContext const&sctx) noexcept;
|
void drawNewItemType(StudioContext const&sctx) noexcept;
|
||||||
|
|
||||||
void drawNewItemName(StudioContext &sctx) noexcept;
|
void drawNewItemPath(StudioContext &sctx) noexcept;
|
||||||
|
|
||||||
void drawNewItemTemplate(StudioContext &sctx) noexcept;
|
void drawNewItemTemplate(StudioContext const &sctx) noexcept;
|
||||||
|
|
||||||
void drawButtons(Stage prev, Stage next) noexcept;
|
void drawButtons(Stage prev, Stage next) noexcept;
|
||||||
|
|
||||||
|
@@ -10,51 +10,42 @@
|
|||||||
|
|
||||||
namespace studio {
|
namespace studio {
|
||||||
|
|
||||||
static ox::Result<ox::UniquePtr<ProjectTreeModel>> buildProjectTreeModel(
|
ProjectExplorer::ProjectExplorer(keel::Context &kctx) noexcept:
|
||||||
ProjectExplorer &explorer,
|
FileExplorer{kctx, true} {
|
||||||
ox::StringParam name,
|
|
||||||
ox::StringView path,
|
|
||||||
ProjectTreeModel *parent) noexcept {
|
|
||||||
auto const fs = explorer.romFs();
|
|
||||||
OX_REQUIRE(stat, fs->stat(path));
|
|
||||||
auto out = ox::make_unique<ProjectTreeModel>(explorer, std::move(name), parent);
|
|
||||||
if (stat.fileType == ox::FileType::Directory) {
|
|
||||||
OX_REQUIRE_M(children, fs->ls(path));
|
|
||||||
std::sort(children.begin(), children.end());
|
|
||||||
ox::Vector<ox::UniquePtr<ProjectTreeModel>> outChildren;
|
|
||||||
for (auto const&childName : children) {
|
|
||||||
if (childName[0] != '.') {
|
|
||||||
auto const childPath = ox::sfmt("{}/{}", path, childName);
|
|
||||||
OX_REQUIRE_M(child, buildProjectTreeModel(explorer, childName, childPath, out.get()));
|
|
||||||
outChildren.emplace_back(std::move(child));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
out->setChildren(std::move(outChildren));
|
|
||||||
}
|
|
||||||
return out;
|
|
||||||
}
|
|
||||||
|
|
||||||
ProjectExplorer::ProjectExplorer(turbine::Context &ctx) noexcept: m_ctx(ctx) {
|
|
||||||
}
|
|
||||||
|
|
||||||
void ProjectExplorer::draw(StudioContext &ctx) noexcept {
|
|
||||||
auto const viewport = ImGui::GetContentRegionAvail();
|
|
||||||
ImGui::BeginChild("ProjectExplorer", ImVec2(300, viewport.y), true);
|
|
||||||
ImGui::SetNextItemOpen(true);
|
|
||||||
if (m_treeModel) {
|
|
||||||
m_treeModel->draw(ctx.tctx);
|
|
||||||
}
|
|
||||||
ImGui::EndChild();
|
|
||||||
}
|
|
||||||
|
|
||||||
void ProjectExplorer::setModel(ox::UPtr<ProjectTreeModel> &&model) noexcept {
|
|
||||||
m_treeModel = std::move(model);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ox::Error ProjectExplorer::refreshProjectTreeModel(ox::StringViewCR) noexcept {
|
ox::Error ProjectExplorer::refreshProjectTreeModel(ox::StringViewCR) noexcept {
|
||||||
OX_REQUIRE_M(model, buildProjectTreeModel(*this, "Project", "/", nullptr));
|
OX_REQUIRE_M(model, buildFileTreeModel(*this, "Project", "/", nullptr));
|
||||||
setModel(std::move(model));
|
setModel(std::move(model));
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ProjectExplorer::fileOpened(ox::StringViewCR path) const noexcept {
|
||||||
|
fileChosen.emit(path);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProjectExplorer::fileContextMenu(ox::StringViewCR path) const noexcept {
|
||||||
|
if (ImGui::BeginPopupContextItem("FileMenu", ImGuiPopupFlags_MouseButtonRight)) {
|
||||||
|
if (ImGui::MenuItem("Delete")) {
|
||||||
|
deleteItem.emit(path);
|
||||||
|
}
|
||||||
|
ImGui::EndPopup();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ProjectExplorer::dirContextMenu(ox::StringViewCR path) const noexcept {
|
||||||
|
if (ImGui::BeginPopupContextItem("DirMenu", ImGuiPopupFlags_MouseButtonRight)) {
|
||||||
|
if (ImGui::MenuItem("Add Item")) {
|
||||||
|
addItem.emit(path);
|
||||||
|
}
|
||||||
|
if (ImGui::MenuItem("Add Directory")) {
|
||||||
|
addDir.emit(path);
|
||||||
|
}
|
||||||
|
if (ImGui::MenuItem("Delete")) {
|
||||||
|
deleteItem.emit(path);
|
||||||
|
}
|
||||||
|
ImGui::EndPopup();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@@ -7,15 +7,12 @@
|
|||||||
#include <ox/event/signal.hpp>
|
#include <ox/event/signal.hpp>
|
||||||
#include <ox/std/memory.hpp>
|
#include <ox/std/memory.hpp>
|
||||||
|
|
||||||
|
#include <studio/filetreemodel.hpp>
|
||||||
#include <studio/widget.hpp>
|
#include <studio/widget.hpp>
|
||||||
#include "projecttreemodel.hpp"
|
|
||||||
|
|
||||||
namespace studio {
|
namespace studio {
|
||||||
|
|
||||||
class ProjectExplorer: public Widget {
|
class ProjectExplorer final: public FileExplorer {
|
||||||
private:
|
|
||||||
ox::UPtr<ProjectTreeModel> m_treeModel;
|
|
||||||
turbine::Context &m_ctx;
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
// slots
|
// slots
|
||||||
@@ -24,18 +21,16 @@ class ProjectExplorer: public Widget {
|
|||||||
ox::Signal<ox::Error(ox::StringViewCR)> addDir;
|
ox::Signal<ox::Error(ox::StringViewCR)> addDir;
|
||||||
ox::Signal<ox::Error(ox::StringViewCR)> deleteItem;
|
ox::Signal<ox::Error(ox::StringViewCR)> deleteItem;
|
||||||
|
|
||||||
explicit ProjectExplorer(turbine::Context &ctx) noexcept;
|
explicit ProjectExplorer(keel::Context &kctx) noexcept;
|
||||||
|
|
||||||
void draw(StudioContext &ctx) noexcept override;
|
|
||||||
|
|
||||||
void setModel(ox::UPtr<ProjectTreeModel> &&model) noexcept;
|
|
||||||
|
|
||||||
ox::Error refreshProjectTreeModel(ox::StringViewCR = {}) noexcept;
|
ox::Error refreshProjectTreeModel(ox::StringViewCR = {}) noexcept;
|
||||||
|
|
||||||
[[nodiscard]]
|
protected:
|
||||||
ox::FileSystem *romFs() noexcept {
|
void fileOpened(ox::StringViewCR path) const noexcept override;
|
||||||
return rom(m_ctx);
|
|
||||||
}
|
void fileContextMenu(ox::StringViewCR path) const noexcept override;
|
||||||
|
|
||||||
|
void dirContextMenu(ox::StringViewCR path) const noexcept override;
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@@ -1,85 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 - 2025 Gary Talent (gary@drinkingtea.net). All rights reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <imgui.h>
|
|
||||||
|
|
||||||
#include <studio/dragdrop.hpp>
|
|
||||||
#include <studio/imguiutil.hpp>
|
|
||||||
|
|
||||||
#include "projectexplorer.hpp"
|
|
||||||
#include "projecttreemodel.hpp"
|
|
||||||
|
|
||||||
namespace studio {
|
|
||||||
|
|
||||||
ProjectTreeModel::ProjectTreeModel(
|
|
||||||
ProjectExplorer &explorer,
|
|
||||||
ox::StringParam name,
|
|
||||||
ProjectTreeModel *parent) noexcept:
|
|
||||||
m_explorer(explorer),
|
|
||||||
m_parent(parent),
|
|
||||||
m_name(std::move(name)) {
|
|
||||||
}
|
|
||||||
|
|
||||||
ProjectTreeModel::ProjectTreeModel(ProjectTreeModel &&other) noexcept:
|
|
||||||
m_explorer(other.m_explorer),
|
|
||||||
m_parent(other.m_parent),
|
|
||||||
m_name(std::move(other.m_name)),
|
|
||||||
m_children(std::move(other.m_children)) {
|
|
||||||
}
|
|
||||||
|
|
||||||
void ProjectTreeModel::draw(turbine::Context &tctx) const noexcept {
|
|
||||||
constexpr auto dirFlags = ImGuiTreeNodeFlags_OpenOnArrow | ImGuiTreeNodeFlags_OpenOnDoubleClick;
|
|
||||||
if (!m_children.empty()) {
|
|
||||||
if (ImGui::TreeNodeEx(m_name.c_str(), dirFlags)) {
|
|
||||||
drawDirContextMenu();
|
|
||||||
for (auto const&child : m_children) {
|
|
||||||
child->draw(tctx);
|
|
||||||
}
|
|
||||||
ImGui::TreePop();
|
|
||||||
} else {
|
|
||||||
ig::IDStackItem const idStackItem{m_name};
|
|
||||||
drawDirContextMenu();
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
auto const path = fullPath();
|
|
||||||
auto const name = ox::sfmt<ox::BasicString<255>>("{}##{}", m_name, path);
|
|
||||||
if (ImGui::TreeNodeEx(name.c_str(), ImGuiTreeNodeFlags_Leaf)) {
|
|
||||||
if (ImGui::IsItemHovered() && ImGui::IsMouseDoubleClicked(0)) {
|
|
||||||
m_explorer.fileChosen.emit(path);
|
|
||||||
}
|
|
||||||
if (ImGui::BeginPopupContextItem("FileMenu", ImGuiPopupFlags_MouseButtonRight)) {
|
|
||||||
if (ImGui::MenuItem("Delete")) {
|
|
||||||
m_explorer.deleteItem.emit(path);
|
|
||||||
}
|
|
||||||
ImGui::EndPopup();
|
|
||||||
}
|
|
||||||
ImGui::TreePop();
|
|
||||||
std::ignore = ig::dragDropSource([this] {
|
|
||||||
ImGui::Text("%s", m_name.c_str());
|
|
||||||
return ig::setDragDropPayload("FileRef", FileRef{fullPath<ox::String>()});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void ProjectTreeModel::setChildren(ox::Vector<ox::UPtr<ProjectTreeModel>> children) noexcept {
|
|
||||||
m_children = std::move(children);
|
|
||||||
}
|
|
||||||
|
|
||||||
void ProjectTreeModel::drawDirContextMenu() const noexcept {
|
|
||||||
if (ImGui::BeginPopupContextItem("DirMenu", ImGuiPopupFlags_MouseButtonRight)) {
|
|
||||||
if (ImGui::MenuItem("Add Item")) {
|
|
||||||
m_explorer.addItem.emit(fullPath());
|
|
||||||
}
|
|
||||||
if (ImGui::MenuItem("Add Directory")) {
|
|
||||||
m_explorer.addDir.emit(fullPath());
|
|
||||||
}
|
|
||||||
if (ImGui::MenuItem("Delete")) {
|
|
||||||
m_explorer.deleteItem.emit(fullPath());
|
|
||||||
}
|
|
||||||
ImGui::EndPopup();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@@ -1,46 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright 2016 - 2025 Gary Talent (gary@drinkingtea.net). All rights reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <ox/std/memory.hpp>
|
|
||||||
#include <ox/std/string.hpp>
|
|
||||||
|
|
||||||
#include <turbine/context.hpp>
|
|
||||||
|
|
||||||
namespace studio {
|
|
||||||
|
|
||||||
class ProjectTreeModel {
|
|
||||||
private:
|
|
||||||
class ProjectExplorer &m_explorer;
|
|
||||||
ProjectTreeModel *m_parent = nullptr;
|
|
||||||
ox::String m_name;
|
|
||||||
ox::Vector<ox::UPtr<ProjectTreeModel>> m_children;
|
|
||||||
|
|
||||||
public:
|
|
||||||
explicit ProjectTreeModel(
|
|
||||||
ProjectExplorer &explorer, ox::StringParam name,
|
|
||||||
ProjectTreeModel *parent = nullptr) noexcept;
|
|
||||||
|
|
||||||
ProjectTreeModel(ProjectTreeModel &&other) noexcept;
|
|
||||||
|
|
||||||
void draw(turbine::Context &tctx) const noexcept;
|
|
||||||
|
|
||||||
void setChildren(ox::Vector<ox::UPtr<ProjectTreeModel>> children) noexcept;
|
|
||||||
|
|
||||||
private:
|
|
||||||
void drawDirContextMenu() const noexcept;
|
|
||||||
|
|
||||||
template<typename Str = ox::BasicString<255>>
|
|
||||||
[[nodiscard]]
|
|
||||||
Str fullPath() const noexcept {
|
|
||||||
if (m_parent) {
|
|
||||||
return m_parent->fullPath<Str>() + "/" + m_name;
|
|
||||||
}
|
|
||||||
return {};
|
|
||||||
}
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
|
@@ -50,7 +50,7 @@ StudioUI::StudioUI(turbine::Context &ctx, ox::StringParam projectDataDir) noexce
|
|||||||
m_sctx(*this, ctx),
|
m_sctx(*this, ctx),
|
||||||
m_tctx(ctx),
|
m_tctx(ctx),
|
||||||
m_projectDataDir(std::move(projectDataDir)),
|
m_projectDataDir(std::move(projectDataDir)),
|
||||||
m_projectExplorer(m_tctx),
|
m_projectExplorer(keelCtx(m_tctx)),
|
||||||
m_newProject(m_projectDataDir),
|
m_newProject(m_projectDataDir),
|
||||||
m_aboutPopup(m_tctx) {
|
m_aboutPopup(m_tctx) {
|
||||||
turbine::setApplicationData(m_tctx, &m_sctx);
|
turbine::setApplicationData(m_tctx, &m_sctx);
|
||||||
@@ -120,7 +120,8 @@ void StudioUI::draw() noexcept {
|
|||||||
ig::s_mainWinHasFocus = ImGui::IsWindowFocused(
|
ig::s_mainWinHasFocus = ImGui::IsWindowFocused(
|
||||||
ImGuiFocusedFlags_RootAndChildWindows | ImGuiFocusedFlags_NoPopupHierarchy);
|
ImGuiFocusedFlags_RootAndChildWindows | ImGuiFocusedFlags_NoPopupHierarchy);
|
||||||
if (m_showProjectExplorer) {
|
if (m_showProjectExplorer) {
|
||||||
m_projectExplorer.draw(m_sctx);
|
auto const v = ImGui::GetContentRegionAvail();
|
||||||
|
m_projectExplorer.draw(m_sctx, {300, v.y});
|
||||||
ImGui::SameLine();
|
ImGui::SameLine();
|
||||||
}
|
}
|
||||||
drawTabBar();
|
drawTabBar();
|
||||||
|
@@ -19,7 +19,6 @@
|
|||||||
#include "newmenu.hpp"
|
#include "newmenu.hpp"
|
||||||
#include "newproject.hpp"
|
#include "newproject.hpp"
|
||||||
#include "projectexplorer.hpp"
|
#include "projectexplorer.hpp"
|
||||||
#include "projecttreemodel.hpp"
|
|
||||||
|
|
||||||
namespace studio {
|
namespace studio {
|
||||||
|
|
||||||
@@ -40,7 +39,7 @@ class StudioUI: public ox::SignalHandler {
|
|||||||
BaseEditor *m_activeEditorOnLastDraw = nullptr;
|
BaseEditor *m_activeEditorOnLastDraw = nullptr;
|
||||||
BaseEditor *m_activeEditor = nullptr;
|
BaseEditor *m_activeEditor = nullptr;
|
||||||
BaseEditor *m_activeEditorUpdatePending = nullptr;
|
BaseEditor *m_activeEditorUpdatePending = nullptr;
|
||||||
NewMenu m_newMenu;
|
NewMenu m_newMenu{keelCtx(m_tctx)};
|
||||||
DeleteConfirmation m_deleteConfirmation;
|
DeleteConfirmation m_deleteConfirmation;
|
||||||
NewDir m_newDirDialog;
|
NewDir m_newDirDialog;
|
||||||
NewProject m_newProject;
|
NewProject m_newProject;
|
||||||
|
138
src/olympic/studio/modlib/include/studio/filetreemodel.hpp
Normal file
138
src/olympic/studio/modlib/include/studio/filetreemodel.hpp
Normal file
@@ -0,0 +1,138 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2016 - 2025 Gary Talent (gary@drinkingtea.net). All rights reserved.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
|
||||||
|
#include <ox/std/memory.hpp>
|
||||||
|
#include <ox/std/string.hpp>
|
||||||
|
|
||||||
|
#include <turbine/context.hpp>
|
||||||
|
|
||||||
|
#include "widget.hpp"
|
||||||
|
|
||||||
|
namespace studio {
|
||||||
|
|
||||||
|
constexpr void safeDelete(class FileTreeModel *m) noexcept;
|
||||||
|
|
||||||
|
class FileExplorer: public ox::SignalHandler {
|
||||||
|
|
||||||
|
friend class FileTreeModel;
|
||||||
|
|
||||||
|
private:
|
||||||
|
keel::Context &m_kctx;
|
||||||
|
class FileTreeModel const *m_selected{};
|
||||||
|
bool const m_fileDraggable{};
|
||||||
|
ox::UPtr<FileTreeModel> m_treeModel;
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit FileExplorer(keel::Context &kctx, bool const fileDraggable = false) noexcept:
|
||||||
|
m_kctx{kctx},
|
||||||
|
m_fileDraggable{fileDraggable} {}
|
||||||
|
|
||||||
|
virtual ~FileExplorer() = default;
|
||||||
|
|
||||||
|
void draw(StudioContext &ctx, ImVec2 const &sz) const noexcept;
|
||||||
|
|
||||||
|
void setModel(ox::UPtr<FileTreeModel> &&model, bool selectRoot = false) noexcept;
|
||||||
|
|
||||||
|
ox::Error setSelectedPath(ox::StringViewCR path) noexcept;
|
||||||
|
|
||||||
|
[[nodiscard]]
|
||||||
|
ox::Optional<ox::String> selectedPath() const;
|
||||||
|
|
||||||
|
virtual void fileOpened(ox::StringViewCR path) const noexcept;
|
||||||
|
|
||||||
|
void drawFileContextMenu(ox::CStringViewCR path) const noexcept;
|
||||||
|
|
||||||
|
void drawDirContextMenu(ox::CStringViewCR path) const noexcept;
|
||||||
|
|
||||||
|
[[nodiscard]]
|
||||||
|
ox::FileSystem &romFs() const noexcept {
|
||||||
|
return *m_kctx.rom;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
virtual void fileContextMenu(ox::StringViewCR path) const noexcept;
|
||||||
|
|
||||||
|
virtual void dirContextMenu(ox::StringViewCR path) const noexcept;
|
||||||
|
|
||||||
|
void setSelectedNode(FileTreeModel const *const node) noexcept {
|
||||||
|
m_selected = node;
|
||||||
|
}
|
||||||
|
|
||||||
|
[[nodiscard]]
|
||||||
|
bool selected(FileTreeModel const *const node) const noexcept {
|
||||||
|
return m_selected == node;
|
||||||
|
}
|
||||||
|
|
||||||
|
[[nodiscard]]
|
||||||
|
bool fileDraggable() const noexcept {
|
||||||
|
return m_fileDraggable;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
ox::Result<bool> setSelectedPath(ox::StringViewCR path, FileTreeModel const&node) noexcept;
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
class FileTreeModel {
|
||||||
|
private:
|
||||||
|
FileExplorer &m_explorer;
|
||||||
|
ox::String m_name;
|
||||||
|
ox::String m_fullPath;
|
||||||
|
ox::Vector<ox::UPtr<FileTreeModel>> m_children;
|
||||||
|
ox::FileType const m_fileType{};
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit FileTreeModel(
|
||||||
|
FileExplorer &explorer,
|
||||||
|
ox::StringParam name,
|
||||||
|
ox::FileType fileType,
|
||||||
|
FileTreeModel const *parent = nullptr) noexcept;
|
||||||
|
|
||||||
|
virtual ~FileTreeModel() = default;
|
||||||
|
|
||||||
|
FileTreeModel(FileTreeModel &&other) noexcept = default;
|
||||||
|
|
||||||
|
void draw(turbine::Context &tctx) const noexcept;
|
||||||
|
|
||||||
|
void setChildren(ox::Vector<ox::UPtr<FileTreeModel>> children) noexcept;
|
||||||
|
|
||||||
|
[[nodiscard]]
|
||||||
|
ox::Vector<ox::UPtr<FileTreeModel>> const&children() const noexcept {
|
||||||
|
return m_children;
|
||||||
|
}
|
||||||
|
|
||||||
|
[[nodiscard]]
|
||||||
|
bool isEmptyDir() const noexcept;
|
||||||
|
|
||||||
|
[[nodiscard]]
|
||||||
|
ox::String const &path() const noexcept {
|
||||||
|
return m_fullPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr void safeDelete(FileTreeModel *m) noexcept {
|
||||||
|
delete m;
|
||||||
|
}
|
||||||
|
|
||||||
|
ox::Result<ox::UPtr<FileTreeModel>> buildFileTreeModel(
|
||||||
|
FileExplorer &explorer,
|
||||||
|
ox::StringParam name,
|
||||||
|
ox::StringViewCR path,
|
||||||
|
FileTreeModel *parent = nullptr,
|
||||||
|
std::function<bool(ox::StringViewCR, ox::FileStat const&)> const&filter =
|
||||||
|
[](ox::StringViewCR, ox::FileStat const&) {return true;},
|
||||||
|
bool showEmptyDirs = true) noexcept;
|
||||||
|
|
||||||
|
ox::Result<ox::UPtr<FileTreeModel>> buildFileTreeModel(
|
||||||
|
FileExplorer &explorer,
|
||||||
|
std::function<bool(ox::StringViewCR, ox::FileStat const&)> const&filter =
|
||||||
|
[](ox::StringViewCR, ox::FileStat const&) {return true;},
|
||||||
|
bool showEmptyDirs = true) noexcept;
|
||||||
|
|
||||||
|
}
|
@@ -95,9 +95,9 @@ class ItemMaker {
|
|||||||
public:
|
public:
|
||||||
constexpr ItemMaker(
|
constexpr ItemMaker(
|
||||||
ox::StringParam pName,
|
ox::StringParam pName,
|
||||||
ox::StringParam pParentDir,
|
ox::StringViewCR pParentDir,
|
||||||
ox::StringParam pFileExt) noexcept:
|
ox::StringParam pFileExt) noexcept:
|
||||||
m_parentDir{std::move(pParentDir)},
|
m_parentDir{sfmt("/{}", pParentDir)},
|
||||||
m_fileExt{std::move(pFileExt)},
|
m_fileExt{std::move(pFileExt)},
|
||||||
m_typeDisplayName{std::move(pName)} {
|
m_typeDisplayName{std::move(pName)} {
|
||||||
}
|
}
|
||||||
@@ -128,6 +128,11 @@ class ItemMaker {
|
|||||||
return m_templates;
|
return m_templates;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[[nodiscard]]
|
||||||
|
ox::String const&fileExt() const noexcept {
|
||||||
|
return m_fileExt;
|
||||||
|
}
|
||||||
|
|
||||||
[[nodiscard]]
|
[[nodiscard]]
|
||||||
ox::String const&defaultPath() const noexcept {
|
ox::String const&defaultPath() const noexcept {
|
||||||
return m_parentDir;
|
return m_parentDir;
|
||||||
@@ -140,7 +145,7 @@ class ItemMaker {
|
|||||||
|
|
||||||
[[nodiscard]]
|
[[nodiscard]]
|
||||||
ox::String itemPath(ox::StringViewCR pName) const noexcept {
|
ox::String itemPath(ox::StringViewCR pName) const noexcept {
|
||||||
return ox::sfmt("/{}/{}.{}", m_parentDir, pName, m_fileExt);
|
return ox::sfmt("{}/{}.{}", m_parentDir, pName, m_fileExt);
|
||||||
}
|
}
|
||||||
|
|
||||||
[[nodiscard]]
|
[[nodiscard]]
|
||||||
@@ -196,12 +201,12 @@ class ItemMakerT final: public ItemMaker {
|
|||||||
public:
|
public:
|
||||||
constexpr ItemMakerT(
|
constexpr ItemMakerT(
|
||||||
ox::StringParam pDisplayName,
|
ox::StringParam pDisplayName,
|
||||||
ox::StringParam pParentDir,
|
ox::StringViewCR pParentDir,
|
||||||
ox::StringParam fileExt,
|
ox::StringParam fileExt,
|
||||||
ox::ClawFormat const pFmt = ox::ClawFormat::Metal) noexcept:
|
ox::ClawFormat const pFmt = ox::ClawFormat::Metal) noexcept:
|
||||||
ItemMaker(
|
ItemMaker(
|
||||||
std::move(pDisplayName),
|
std::move(pDisplayName),
|
||||||
std::move(pParentDir),
|
pParentDir,
|
||||||
std::move(fileExt)),
|
std::move(fileExt)),
|
||||||
m_fmt{pFmt} {
|
m_fmt{pFmt} {
|
||||||
installTemplate(ox::make_unique<ItemTemplateT<T>>());
|
installTemplate(ox::make_unique<ItemTemplateT<T>>());
|
||||||
@@ -209,13 +214,13 @@ class ItemMakerT final: public ItemMaker {
|
|||||||
|
|
||||||
constexpr ItemMakerT(
|
constexpr ItemMakerT(
|
||||||
ox::StringParam pDisplayName,
|
ox::StringParam pDisplayName,
|
||||||
ox::StringParam pParentDir,
|
ox::StringViewCR pParentDir,
|
||||||
ox::StringParam fileExt,
|
ox::StringParam fileExt,
|
||||||
T const&pItem,
|
T const&pItem,
|
||||||
ox::ClawFormat const pFmt) noexcept:
|
ox::ClawFormat const pFmt) noexcept:
|
||||||
ItemMaker(
|
ItemMaker(
|
||||||
std::move(pDisplayName),
|
std::move(pDisplayName),
|
||||||
std::move(pParentDir),
|
pParentDir,
|
||||||
std::move(fileExt)),
|
std::move(fileExt)),
|
||||||
m_fmt{pFmt} {
|
m_fmt{pFmt} {
|
||||||
installTemplate(ox::make_unique<ItemTemplateT<T>>("Default", std::move(pItem)));
|
installTemplate(ox::make_unique<ItemTemplateT<T>>("Default", std::move(pItem)));
|
||||||
@@ -223,13 +228,13 @@ class ItemMakerT final: public ItemMaker {
|
|||||||
|
|
||||||
constexpr ItemMakerT(
|
constexpr ItemMakerT(
|
||||||
ox::StringParam pDisplayName,
|
ox::StringParam pDisplayName,
|
||||||
ox::StringParam pParentDir,
|
ox::StringViewCR pParentDir,
|
||||||
ox::StringParam fileExt,
|
ox::StringParam fileExt,
|
||||||
T &&pItem,
|
T &&pItem,
|
||||||
ox::ClawFormat const pFmt) noexcept:
|
ox::ClawFormat const pFmt) noexcept:
|
||||||
ItemMaker(
|
ItemMaker(
|
||||||
std::move(pDisplayName),
|
std::move(pDisplayName),
|
||||||
std::move(pParentDir),
|
pParentDir,
|
||||||
std::move(fileExt)),
|
std::move(fileExt)),
|
||||||
m_fmt{pFmt} {
|
m_fmt{pFmt} {
|
||||||
installTemplate(ox::make_unique<ItemTemplateT<T>>("Default", std::move(pItem)));
|
installTemplate(ox::make_unique<ItemTemplateT<T>>("Default", std::move(pItem)));
|
||||||
|
@@ -37,10 +37,28 @@ class Module {
|
|||||||
|
|
||||||
template<typename Editor>
|
template<typename Editor>
|
||||||
[[nodiscard]]
|
[[nodiscard]]
|
||||||
studio::EditorMaker editorMaker(studio::StudioContext &ctx, ox::StringParam ext) noexcept {
|
EditorMaker editorMaker(StudioContext &ctx, ox::StringParam ext) noexcept {
|
||||||
return {
|
return {
|
||||||
{std::move(ext)},
|
{std::move(ext)},
|
||||||
[&ctx](ox::StringViewCR path) -> ox::Result<studio::BaseEditor*> {
|
[&ctx](ox::StringViewCR path) -> ox::Result<BaseEditor*> {
|
||||||
|
return ox::makeCatch<Editor>(ctx, path);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename Editor>
|
||||||
|
[[nodiscard]]
|
||||||
|
EditorMaker editorMaker(StudioContext &ctx, std::initializer_list<ox::StringView> exts) noexcept {
|
||||||
|
return {
|
||||||
|
[&exts] {
|
||||||
|
ox::Vector<ox::String> fileTypes;
|
||||||
|
fileTypes.reserve(exts.size());
|
||||||
|
for (auto &s : exts) {
|
||||||
|
fileTypes.emplace_back(s);
|
||||||
|
}
|
||||||
|
return fileTypes;
|
||||||
|
}(),
|
||||||
|
[&ctx](ox::StringViewCR path) -> ox::Result<BaseEditor*> {
|
||||||
return ox::makeCatch<Editor>(ctx, path);
|
return ox::makeCatch<Editor>(ctx, path);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@@ -0,0 +1,16 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2016 - 2025 Gary Talent (gary@drinkingtea.net). All rights reserved.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <ox/std/memory.hpp>
|
||||||
|
#include <ox/std/string.hpp>
|
||||||
|
|
||||||
|
#include <turbine/context.hpp>
|
||||||
|
|
||||||
|
#include <studio/filetreemodel.hpp>
|
||||||
|
|
||||||
|
namespace studio {
|
||||||
|
|
||||||
|
}
|
@@ -9,6 +9,7 @@
|
|||||||
#include <studio/dragdrop.hpp>
|
#include <studio/dragdrop.hpp>
|
||||||
#include <studio/editor.hpp>
|
#include <studio/editor.hpp>
|
||||||
#include <studio/filedialog.hpp>
|
#include <studio/filedialog.hpp>
|
||||||
|
#include <studio/filetreemodel.hpp>
|
||||||
#include <studio/imguiutil.hpp>
|
#include <studio/imguiutil.hpp>
|
||||||
#include <studio/module.hpp>
|
#include <studio/module.hpp>
|
||||||
#include <studio/itemmaker.hpp>
|
#include <studio/itemmaker.hpp>
|
||||||
|
@@ -2,6 +2,8 @@ add_library(
|
|||||||
Studio
|
Studio
|
||||||
configio.cpp
|
configio.cpp
|
||||||
editor.cpp
|
editor.cpp
|
||||||
|
projectfilepicker.cpp
|
||||||
|
filetreemodel.cpp
|
||||||
imguiutil.cpp
|
imguiutil.cpp
|
||||||
module.cpp
|
module.cpp
|
||||||
popup.cpp
|
popup.cpp
|
||||||
|
172
src/olympic/studio/modlib/src/filetreemodel.cpp
Normal file
172
src/olympic/studio/modlib/src/filetreemodel.cpp
Normal file
@@ -0,0 +1,172 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2016 - 2025 Gary Talent (gary@drinkingtea.net). All rights reserved.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <imgui.h>
|
||||||
|
|
||||||
|
#include <studio/dragdrop.hpp>
|
||||||
|
#include <studio/imguiutil.hpp>
|
||||||
|
|
||||||
|
#include <studio/filetreemodel.hpp>
|
||||||
|
|
||||||
|
namespace studio {
|
||||||
|
|
||||||
|
void FileExplorer::draw(StudioContext &ctx, ImVec2 const &sz) const noexcept {
|
||||||
|
ImGui::BeginChild("ProjectExplorer", sz, true);
|
||||||
|
ImGui::SetNextItemOpen(true);
|
||||||
|
if (m_treeModel) {
|
||||||
|
m_treeModel->draw(ctx.tctx);
|
||||||
|
}
|
||||||
|
ImGui::EndChild();
|
||||||
|
}
|
||||||
|
|
||||||
|
void FileExplorer::setModel(ox::UPtr<FileTreeModel> &&model, bool const selectRoot) noexcept {
|
||||||
|
m_treeModel = std::move(model);
|
||||||
|
setSelectedNode(selectRoot ? m_treeModel.get() : nullptr);
|
||||||
|
}
|
||||||
|
|
||||||
|
ox::Error FileExplorer::setSelectedPath(ox::StringViewCR path) noexcept {
|
||||||
|
return setSelectedPath(path, *m_treeModel).error;
|
||||||
|
}
|
||||||
|
|
||||||
|
[[nodiscard]]
|
||||||
|
ox::Optional<ox::String> FileExplorer::selectedPath() const {
|
||||||
|
if (m_selected) {
|
||||||
|
return ox::Optional<ox::String>{ox::in_place, m_selected->path()};
|
||||||
|
}
|
||||||
|
return ox::Optional<ox::String>{};
|
||||||
|
}
|
||||||
|
|
||||||
|
void FileExplorer::fileOpened(ox::StringViewCR) const noexcept {}
|
||||||
|
|
||||||
|
void FileExplorer::drawFileContextMenu(ox::CStringViewCR path) const noexcept {
|
||||||
|
ig::IDStackItem const idStackItem{path};
|
||||||
|
fileContextMenu(path);
|
||||||
|
}
|
||||||
|
|
||||||
|
void FileExplorer::drawDirContextMenu(ox::CStringViewCR path) const noexcept {
|
||||||
|
ig::IDStackItem const idStackItem{path};
|
||||||
|
dirContextMenu(path);
|
||||||
|
}
|
||||||
|
|
||||||
|
void FileExplorer::fileContextMenu(ox::StringViewCR) const noexcept {}
|
||||||
|
|
||||||
|
void FileExplorer::dirContextMenu(ox::StringViewCR) const noexcept {}
|
||||||
|
|
||||||
|
ox::Result<bool> FileExplorer::setSelectedPath(
|
||||||
|
ox::StringViewCR path, FileTreeModel const&node) noexcept {
|
||||||
|
if (path == node.path()) {
|
||||||
|
m_selected = &node;
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
for (auto &c : node.children()) {
|
||||||
|
OX_REQUIRE(done, setSelectedPath(path, *c));
|
||||||
|
if (done) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
FileTreeModel::FileTreeModel(
|
||||||
|
FileExplorer &explorer,
|
||||||
|
ox::StringParam name,
|
||||||
|
ox::FileType const fileType,
|
||||||
|
FileTreeModel const *const parent) noexcept:
|
||||||
|
m_explorer{explorer},
|
||||||
|
m_name{std::move(name)},
|
||||||
|
m_fullPath{parent ? sfmt("{}/{}", parent->m_fullPath, m_name) : ox::String{}},
|
||||||
|
m_fileType{fileType} {}
|
||||||
|
|
||||||
|
void FileTreeModel::draw(turbine::Context &tctx) const noexcept {
|
||||||
|
constexpr auto dirFlags = ImGuiTreeNodeFlags_OpenOnArrow | ImGuiTreeNodeFlags_OpenOnDoubleClick;
|
||||||
|
auto const selected = m_explorer.selected(this) ? ImGuiTreeNodeFlags_Selected : 0;
|
||||||
|
if (!m_children.empty()) {
|
||||||
|
auto const nodeOpen = ImGui::TreeNodeEx(m_name.c_str(), dirFlags | selected);
|
||||||
|
if (ImGui::IsItemClicked()) {
|
||||||
|
m_explorer.setSelectedNode(this);
|
||||||
|
}
|
||||||
|
ig::IDStackItem const idStackItem{m_name};
|
||||||
|
m_explorer.drawDirContextMenu(m_fullPath);
|
||||||
|
if (nodeOpen) {
|
||||||
|
for (auto const&child : m_children) {
|
||||||
|
child->draw(tctx);
|
||||||
|
}
|
||||||
|
ImGui::TreePop();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (ImGui::TreeNodeEx(m_name.c_str(), ImGuiTreeNodeFlags_Leaf | selected)) {
|
||||||
|
if (ImGui::IsItemHovered() && ImGui::IsMouseDoubleClicked(0)) {
|
||||||
|
m_explorer.fileOpened(m_fullPath);
|
||||||
|
}
|
||||||
|
if (ImGui::IsItemClicked()) {
|
||||||
|
m_explorer.setSelectedNode(this);
|
||||||
|
}
|
||||||
|
m_explorer.drawFileContextMenu(m_fullPath);
|
||||||
|
ImGui::TreePop();
|
||||||
|
if (m_explorer.fileDraggable()) {
|
||||||
|
std::ignore = ig::dragDropSource([this] {
|
||||||
|
ImGui::Text("%s", m_name.c_str());
|
||||||
|
return ig::setDragDropPayload("FileRef", FileRef{ox::String{m_fullPath}});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void FileTreeModel::setChildren(ox::Vector<ox::UPtr<FileTreeModel>> children) noexcept {
|
||||||
|
m_children = std::move(children);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool FileTreeModel::isEmptyDir() const noexcept {
|
||||||
|
return m_children.empty() && m_fileType == ox::FileType::Directory;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
ox::Result<ox::UPtr<FileTreeModel>> buildFileTreeModel(
|
||||||
|
FileExplorer &explorer,
|
||||||
|
ox::StringParam name,
|
||||||
|
ox::StringViewCR path,
|
||||||
|
FileTreeModel *parent,
|
||||||
|
std::function<bool(ox::StringViewCR, ox::FileStat const&)> const &filter,
|
||||||
|
bool const showEmptyDirs) noexcept {
|
||||||
|
auto const&fs = explorer.romFs();
|
||||||
|
OX_REQUIRE(stat, fs.stat(path));
|
||||||
|
auto out = ox::make_unique<FileTreeModel>(explorer, std::move(name), stat.fileType, parent);
|
||||||
|
if (stat.fileType == ox::FileType::Directory) {
|
||||||
|
OX_REQUIRE_M(children, fs.ls(path));
|
||||||
|
std::sort(children.begin(), children.end());
|
||||||
|
ox::Vector<ox::UPtr<FileTreeModel>> outChildren;
|
||||||
|
for (auto const&childName : children) {
|
||||||
|
if (childName[0] != '.') {
|
||||||
|
auto const childPath = ox::sfmt("{}/{}", path, childName);
|
||||||
|
OX_REQUIRE(childStat, fs.stat(childPath));
|
||||||
|
if (filter(childPath, childStat)) {
|
||||||
|
OX_REQUIRE_M(child, buildFileTreeModel(
|
||||||
|
explorer,
|
||||||
|
childName,
|
||||||
|
childPath,
|
||||||
|
out.get(),
|
||||||
|
filter,
|
||||||
|
showEmptyDirs));
|
||||||
|
auto const emptyDir = child->isEmptyDir();
|
||||||
|
if (!emptyDir || showEmptyDirs) {
|
||||||
|
outChildren.emplace_back(std::move(child));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
out->setChildren(std::move(outChildren));
|
||||||
|
}
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
|
ox::Result<ox::UPtr<FileTreeModel>> buildFileTreeModel(
|
||||||
|
FileExplorer &explorer,
|
||||||
|
std::function<bool(ox::StringViewCR, ox::FileStat const&)> const&filter,
|
||||||
|
bool const showEmptyDirs) noexcept {
|
||||||
|
return buildFileTreeModel(explorer, "Project", "/", nullptr, filter, showEmptyDirs);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
18
src/olympic/studio/modlib/src/projectfilepicker.cpp
Normal file
18
src/olympic/studio/modlib/src/projectfilepicker.cpp
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2016 - 2025 Gary Talent (gary@drinkingtea.net). All rights reserved.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <studio/filetreemodel.hpp>
|
||||||
|
#include <studio/projectfilepicker.hpp>
|
||||||
|
|
||||||
|
namespace studio {
|
||||||
|
|
||||||
|
class ProjectFilePicker: public FileExplorer {
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit ProjectFilePicker(keel::Context &kctx) noexcept: FileExplorer{kctx} {}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
};
|
||||||
|
|
||||||
|
}
|
Reference in New Issue
Block a user