[nostalgia] Add PaletteEditor

This commit is contained in:
Gary Talent 2022-04-02 16:38:38 -05:00
parent 99987ee423
commit 71e0f181ea
14 changed files with 251 additions and 334 deletions

View File

@ -129,12 +129,13 @@ constexpr float bluef(Color32 c) noexcept {
[[nodiscard]]
constexpr Color16 color16(uint8_t r, uint8_t g, uint8_t b) noexcept {
return r | (g << 5) | (b << 10);
constexpr Color16 color16(uint8_t r, uint8_t g, uint8_t b, uint8_t a = 0) noexcept {
return ox::min<uint8_t>(r, 31) | (ox::min<uint8_t>(g, 31) << 5) | (ox::min<uint8_t>(b, 31) << 10) | (a << 15);
}
static_assert(color16(0, 31, 0) == 992);
static_assert(color16(16, 31, 0) == 1008);
static_assert(color16(16, 31, 8) == 9200);
static_assert(color16(16, 32, 8) == 9200);
}

View File

@ -36,7 +36,6 @@ class ClipboardObject: public BaseClipboardObject {
}
};
class Context;
class Drawer;
// User Input Output

View File

@ -4,7 +4,7 @@ add_library(
module.cpp
#new_tilesheet_wizard.cpp
#newpalettewizard.cpp
#paletteeditor.cpp
paletteeditor.cpp
tilesheeteditor-imgui.cpp
tilesheeteditorview.cpp
tilesheeteditormodel.cpp

View File

@ -2,18 +2,29 @@
* Copyright 2016 - 2022 Gary Talent (gary@drinkingtea.net). All rights reserved.
*/
#include "paletteeditor.hpp"
#include "tilesheeteditor-imgui.hpp"
#include "module.hpp"
namespace nostalgia::core {
ox::Vector<studio::EditorMaker> Module::editors(core::Context *ctx) {
ox::Vector<studio::EditorMaker> Module::editors(core::Context *ctx) noexcept {
return {
{
{"ng"},
[ctx](const ox::String &path) {
[ctx](const ox::String &path) -> ox::Result<studio::Editor*> {
try {
return new TileSheetEditorImGui(ctx, path);
} catch (const ox::Exception &ex) {
return ex.toError();
}
}
},
{
{"npal"},
[ctx](const ox::String &path) -> ox::Result<studio::Editor*> {
return PaletteEditorImGui::make(ctx, path);
}
}
};

View File

@ -10,7 +10,7 @@ namespace nostalgia::core {
class Module: public studio::Module {
public:
ox::Vector<studio::EditorMaker> editors(core::Context *ctx) override;
ox::Vector<studio::EditorMaker> editors(core::Context *ctx) noexcept override;
};
}

View File

@ -2,16 +2,14 @@
* Copyright 2016 - 2022 Gary Talent (gary@drinkingtea.net). All rights reserved.
*/
#include <QHeaderView>
#include <QItemDelegate>
#include <QPainter>
#include <QPushButton>
#include <QTableWidget>
#include <QToolBar>
#include <algorithm>
#include <imgui.h>
#include <nostalgia/core/gfx.hpp>
#include <nostalgia/core/media.hpp>
#include <ox/std/memory.hpp>
#include "util.hpp"
#include "paletteeditor.hpp"
namespace nostalgia::core {
@ -24,359 +22,294 @@ enum class PaletteEditorCommandId {
};
class ColorChannelValidator: public QValidator {
public:
explicit ColorChannelValidator(QLineEdit *parent);
QValidator::State validate(QString &input, int&) const override;
class AddColorCommand: public studio::UndoCommand {
private:
[[nodiscard]] static QString convert(const QString &input);
};
ColorChannelValidator::ColorChannelValidator(QLineEdit *parent): QValidator(parent) {
connect(parent, &QLineEdit::editingFinished, [parent] {
parent->setText(convert(parent->text()));
});
}
QString ColorChannelValidator::convert(const QString &input) {
int num = 0;
if (input[0] == '_') {
num = input.mid(1).toInt() >> 3;
}
return QString::number(num);
}
QValidator::State ColorChannelValidator::validate(QString &input, int&) const {
if (input.size() == 0) {
return QValidator::State::Intermediate;
}
const auto convert = input[0] == '_';
const auto max = convert ? 255 : 31;
const auto numTxt = convert ? input.mid(1) : input;
bool isNumber = false;
const auto num = numTxt.toInt(&isNumber);
if (isNumber && num >= 0 && num <= max) {
return QValidator::State::Acceptable;
} else if (numTxt == "") {
return QValidator::State::Intermediate;
} else {
return QValidator::State::Invalid;
}
}
class AddColorCommand: public QUndoCommand {
private:
PaletteEditor *m_editor = nullptr;
PaletteEditorImGui *m_editor = nullptr;
Color16 m_color = 0;
int m_idx = -1;
public:
AddColorCommand(PaletteEditor *editor, Color16 color, int idx) {
AddColorCommand(PaletteEditorImGui *editor, Color16 color, int idx) noexcept {
m_editor = editor;
m_color = color;
m_idx = idx;
}
~AddColorCommand() override = default;
~AddColorCommand() noexcept override = default;
[[nodiscard]] int id() const override {
[[nodiscard]]
int commandId() const noexcept override {
return static_cast<int>(PaletteEditorCommandId::AddColor);
}
void redo() override {
void redo() noexcept override {
m_editor->addColor(m_idx, m_color);
}
void undo() override {
void undo() noexcept override {
m_editor->rmColor(m_idx);
}
};
class RemoveColorCommand: public QUndoCommand {
class RemoveColorCommand: public studio::UndoCommand {
private:
PaletteEditor *m_editor = nullptr;
PaletteEditorImGui *m_editor = nullptr;
Color16 m_color = 0;
int m_idx = -1;
public:
RemoveColorCommand(PaletteEditor *editor, Color16 color, int idx) {
RemoveColorCommand(PaletteEditorImGui *editor, Color16 color, int idx) noexcept {
m_editor = editor;
m_color = color;
m_idx = idx;
}
~RemoveColorCommand() override = default;
~RemoveColorCommand() noexcept override = default;
[[nodiscard]] int id() const override {
[[nodiscard]]
int commandId() const noexcept override {
return static_cast<int>(PaletteEditorCommandId::RemoveColor);
}
void redo() override {
void redo() noexcept override {
m_editor->rmColor(m_idx);
}
void undo() override {
void undo() noexcept override {
m_editor->addColor(m_idx, m_color);
}
};
class UpdateColorCommand: public QUndoCommand {
class UpdateColorCommand: public studio::UndoCommand {
private:
PaletteEditor *m_editor = nullptr;
PaletteEditorImGui *m_editor = nullptr;
Color16 m_oldColor = 0;
Color16 m_newColor = 0;
int m_idx = -1;
public:
UpdateColorCommand(PaletteEditor *editor, int idx, Color16 oldColor, Color16 newColor) {
UpdateColorCommand(PaletteEditorImGui *editor, int idx, Color16 oldColor, Color16 newColor) noexcept {
m_editor = editor;
m_idx = idx;
m_oldColor = oldColor;
m_newColor = newColor;
setObsolete(m_oldColor == m_newColor);
//setObsolete(m_oldColor == m_newColor);
}
~UpdateColorCommand() override = default;
~UpdateColorCommand() noexcept override = default;
[[nodiscard]] int id() const override {
bool mergeWith(const UndoCommand *cmd) noexcept final {
if (cmd->commandId() != static_cast<int>(PaletteEditorCommandId::UpdateColor)) {
return false;
}
auto ucCmd = static_cast<const UpdateColorCommand*>(cmd);
if (m_idx != ucCmd->m_idx) {
return false;
}
m_newColor = ucCmd->m_newColor;
return true;
}
int commandId() const noexcept final {
return static_cast<int>(PaletteEditorCommandId::UpdateColor);
}
void redo() override {
void redo() noexcept final {
m_editor->updateColor(m_idx, m_newColor);
}
void undo() override {
void undo() noexcept final {
m_editor->updateColor(m_idx, m_oldColor);
}
};
class MoveColorCommand: public QUndoCommand {
class MoveColorCommand: public studio::UndoCommand {
private:
PaletteEditor *m_editor = nullptr;
int m_idx = -1;
PaletteEditorImGui *m_editor = nullptr;
std::size_t m_idx = 0;
int m_offset = 0;
public:
MoveColorCommand(PaletteEditor *editor, int idx, int offset) {
MoveColorCommand(PaletteEditorImGui *editor, std::size_t idx, int offset) noexcept {
m_editor = editor;
m_idx = idx;
m_offset = offset;
}
~MoveColorCommand() override = default;
~MoveColorCommand() noexcept override = default;
[[nodiscard]] int id() const override {
[[nodiscard]]
int commandId() const noexcept override {
return static_cast<int>(PaletteEditorCommandId::MoveColor);
}
void redo() override {
void redo() noexcept override {
m_editor->moveColor(m_idx, m_offset);
}
void undo() override {
m_editor->moveColor(m_idx + m_offset, -m_offset);
void undo() noexcept override {
m_editor->moveColor(static_cast<int>(m_idx) + m_offset, -m_offset);
}
};
QWidget *PaletteEditorColorTableDelegate::createEditor(QWidget *parent, const QStyleOptionViewItem&, const QModelIndex &idx) const {
auto le = new QLineEdit(parent);
if (idx.column()) {
auto validator = new ColorChannelValidator(le);
le->setValidator(validator);
ox::Result<PaletteEditorImGui*> PaletteEditorImGui::make(Context *ctx, const ox::String &path) noexcept {
auto out = ox::UniquePtr<PaletteEditorImGui>(new PaletteEditorImGui);
out->m_ctx = ctx;
out->m_itemPath = path;
const auto lastSlash = std::find(out->m_itemPath.rbegin(), out->m_itemPath.rend(), '/').offset();
out->m_itemName = out->m_itemPath.substr(lastSlash + 1);
oxRequire(pal, core::readObj<Palette>(out->m_ctx, out->m_itemPath));
out->m_pal = *pal.get();
out->undoStack()->changeTriggered.connect(out.get(), &PaletteEditorImGui::markUnsavedChanges);
return out.release();
}
const ox::String &PaletteEditorImGui::itemName() const {
return m_itemName;
}
studio::UndoStack *PaletteEditorImGui::undoStack() noexcept {
return &m_undoStack;
}
void PaletteEditorImGui::draw(core::Context*) noexcept {
static constexpr auto flags = ImGuiTableFlags_RowBg;
const auto paneSize = ImGui::GetContentRegionAvail();
ImGui::BeginChild("PaletteEditor");
{
ImGui::BeginChild("Colors", ImVec2(paneSize.x - 200, paneSize.y), false);
{
const auto colorsSz = ImGui::GetContentRegionAvail();
static constexpr auto toolbarHeight = 40;
ImGui::BeginChild("Toolbar", ImVec2(colorsSz.x, toolbarHeight), true);
{
const auto sz = ImVec2(70, 24);
if (ImGui::Button("Add", sz)) {
m_undoStack.push(new AddColorCommand(this, 0, m_pal.colors.size()));
}
return le;
}
void PaletteEditorColorTableDelegate::paint(QPainter *painter, const QStyleOptionViewItem &opt, const QModelIndex &idx) const {
if (idx.column() != 3) {
QStyledItemDelegate::paint(painter, opt, idx);
} else {
auto color = idx.model()->data(idx, Qt::DisplayRole).toString();
painter->fillRect(opt.rect, QColor(color));
ImGui::SameLine();
ImGui::BeginDisabled(m_selectedRow >= m_pal.colors.size());
{
if (ImGui::Button("Remove", sz)) {
m_undoStack.push(new RemoveColorCommand(this, m_pal.colors[static_cast<std::size_t>(m_selectedRow)], m_selectedRow));
m_selectedRow = ox::min(m_pal.colors.size() - 1, m_selectedRow);
}
}
static QTableWidgetItem *mkCell(const QString& v, bool editable = true) {
auto c = new QTableWidgetItem;
c->setText(v);
c->setFont(QFont("monospace"));
c->setTextAlignment(Qt::AlignRight | Qt::AlignVCenter);
if (!editable) {
c->setFlags(c->flags() & ~Qt::ItemIsEditable);
ImGui::SameLine();
ImGui::BeginDisabled(m_selectedRow <= 0);
{
if (ImGui::Button("Move Up", sz)) {
m_undoStack.push(new MoveColorCommand(this, m_selectedRow, -1));
--m_selectedRow;
}
return c;
}
ImGui::EndDisabled();
ImGui::SameLine();
ImGui::BeginDisabled(m_selectedRow >= m_pal.colors.size() - 1);
{
if (ImGui::Button("Move Down", sz)) {
m_undoStack.push(new MoveColorCommand(this, m_selectedRow, 1));
++m_selectedRow;
}
}
ImGui::EndDisabled();
}
ImGui::EndDisabled();
}
ImGui::EndChild();
ImGui::BeginTable("Colors", 5, flags, ImVec2(colorsSz.x, colorsSz.y - (toolbarHeight + 5)));
{
ImGui::TableSetupColumn("Idx", ImGuiTableColumnFlags_WidthFixed, 25);
ImGui::TableSetupColumn("Red", ImGuiTableColumnFlags_WidthFixed, 50);
ImGui::TableSetupColumn("Green", ImGuiTableColumnFlags_WidthFixed, 50);
ImGui::TableSetupColumn("Blue", ImGuiTableColumnFlags_WidthFixed, 50);
ImGui::TableSetupColumn("Color Preview", ImGuiTableColumnFlags_NoHide);
ImGui::TableHeadersRow();
for (auto i = 0u; const auto c : m_pal.colors) {
ImGui::PushID(static_cast<int>(i));
ImGui::TableNextRow();
// Color No.
ImGui::TableNextColumn();
ImGui::Text("%d", i);
// Red
ImGui::TableNextColumn();
ImGui::Text("%d", red16(c));
// Green
ImGui::TableNextColumn();
ImGui::Text("%d", green16(c));
// Blue
ImGui::TableNextColumn();
ImGui::Text("%d", blue16(c));
// ColorPreview
ImGui::TableNextColumn();
const auto ic = ImGui::GetColorU32(ImVec4(redf(c), greenf(c), bluef(c), 1));
ImGui::TableSetBgColor(ImGuiTableBgTarget_CellBg, ic);
if (ImGui::Selectable("##ColorRow", i == m_selectedRow, ImGuiSelectableFlags_SpanAllColumns)) {
m_selectedRow = i;
}
ImGui::PopID();
++i;
}
}
ImGui::EndTable();
}
ImGui::EndChild();
if (m_selectedRow < m_pal.colors.size()) {
ImGui::SameLine();
ImGui::BeginChild("ColorEditor", ImVec2(200, paneSize.y), true);
{
const auto c = m_pal.colors[m_selectedRow];
int r = red16(c);
int g = green16(c);
int b = blue16(c);
int a = alpha16(c);
ImGui::InputInt("Red", &r, 1, 5);
ImGui::InputInt("Green", &g, 1, 5);
ImGui::InputInt("Blue", &b, 1, 5);
const auto newColor = color16(r, g, b, a);
if (c != newColor) {
m_undoStack.push(new UpdateColorCommand(this, m_selectedRow, c, newColor));
}
}
ImGui::EndChild();
}
}
ImGui::EndChild();
}
static QTableWidgetItem *mkCell(uint8_t v) {
return mkCell(QString("%1").arg(v), true);
}
PaletteEditor::PaletteEditor(QString path, const studio::Context *ctx, QWidget *parent): studio::Editor(parent) {
m_ctx = ctx;
m_itemPath = path;
auto canvasLyt = new QVBoxLayout(this);
auto tb = new QToolBar(tr("Tile Sheet Options"));
m_addBtn = new QPushButton(tr("Add"), tb);
m_rmBtn = new QPushButton(tr("Remove"), tb);
m_moveUpBtn = new QPushButton(tr("Move Up"), tb);
m_moveDownBtn = new QPushButton(tr("Move Down"), tb);
m_rmBtn->setEnabled(false);
m_moveUpBtn->setEnabled(false);
m_moveDownBtn->setEnabled(false);
tb->addWidget(m_addBtn);
tb->addWidget(m_rmBtn);
tb->addWidget(m_moveUpBtn);
tb->addWidget(m_moveDownBtn);
canvasLyt->setMenuBar(tb);
m_table = new QTableWidget(this);
m_table->setItemDelegate(&m_colorTableDelegate);
m_table->setColumnCount(4);
m_table->setSelectionBehavior(QAbstractItemView::SelectRows);
m_table->setSelectionMode(QAbstractItemView::SingleSelection);
m_table->setHorizontalHeaderLabels(QStringList() << tr("Red") << tr("Green") << tr("Blue") << tr("Color Preview"));
m_table->horizontalHeader()->setStretchLastSection(true);
m_table->verticalHeader()->hide();
canvasLyt->addWidget(m_table);
connect(m_table, &QTableWidget::itemSelectionChanged, this, &PaletteEditor::colorSelected);
connect(m_addBtn, &QPushButton::clicked, this, &PaletteEditor::addColorClicked);
connect(m_rmBtn, &QPushButton::clicked, this, &PaletteEditor::rmColorClicked);
connect(m_moveUpBtn, &QPushButton::clicked, this, &PaletteEditor::moveColorUpClicked);
connect(m_moveDownBtn, &QPushButton::clicked, this, &PaletteEditor::moveColorDownClicked);
connect(m_table, &QTableWidget::cellChanged, this, &PaletteEditor::cellChanged);
m_pal = m_ctx->project->loadObj<NostalgiaPalette>(m_itemPath);
load();
}
QString PaletteEditor::itemName() const {
return m_itemPath.mid(m_itemPath.lastIndexOf('/'));
}
void PaletteEditor::addTableRow(int i, Color16 c) {
disconnect(m_table, &QTableWidget::cellChanged, this, &PaletteEditor::cellChanged);
m_table->insertRow(i);
m_table->setItem(i, 0, mkCell(red16(c)));
m_table->setItem(i, 1, mkCell(green16(c)));
m_table->setItem(i, 2, mkCell(blue16(c)));
m_table->setItem(i, 3, mkCell(toQColor(m_pal->colors[static_cast<std::size_t>(i)]).name(QColor::HexRgb), false));
connect(m_table, &QTableWidget::cellChanged, this, &PaletteEditor::cellChanged);
}
void PaletteEditor::rmTableRow(int idx) {
disconnect(m_table, &QTableWidget::cellChanged, this, &PaletteEditor::cellChanged);
m_table->removeRow(idx);
connect(m_table, &QTableWidget::cellChanged, this, &PaletteEditor::cellChanged);
}
void PaletteEditor::setTableRow(int idx, Color16 c) {
disconnect(m_table, &QTableWidget::cellChanged, this, &PaletteEditor::cellChanged);
m_table->item(idx, 0)->setText(QString::number(red16(c)));
m_table->item(idx, 1)->setText(QString::number(green16(c)));
m_table->item(idx, 2)->setText(QString::number(blue16(c)));
m_table->item(idx, 3)->setText(toQColor(m_pal->colors[static_cast<std::size_t>(idx)]).name(QColor::HexRgb));
connect(m_table, &QTableWidget::cellChanged, this, &PaletteEditor::cellChanged);
}
void PaletteEditor::addColor(int idx, Color16 c) {
m_pal->colors.insert(static_cast<std::size_t>(idx), c);
addTableRow(idx, c);
void PaletteEditorImGui::addColor(int idx, Color16 c) noexcept {
m_pal.colors.insert(static_cast<std::size_t>(idx), c);
setUnsavedChanges(true);
}
void PaletteEditor::rmColor(int idx) {
rmTableRow(idx);
oxIgnoreError(m_pal->colors.erase(static_cast<std::size_t>(idx)));
void PaletteEditorImGui::rmColor(int idx) noexcept {
oxIgnoreError(m_pal.colors.erase(static_cast<std::size_t>(idx)));
setUnsavedChanges(true);
}
void PaletteEditor::updateColor(int idx, Color16 c) {
m_pal->colors[static_cast<std::size_t>(idx)] = c;
setTableRow(idx, c);
void PaletteEditorImGui::updateColor(int idx, Color16 c) noexcept {
m_pal.colors[static_cast<std::size_t>(idx)] = c;
setUnsavedChanges(true);
}
void PaletteEditor::moveColor(int idx, int offset) {
auto c = m_pal->colors[static_cast<std::size_t>(idx)];
oxIgnoreError(m_pal->colors.erase(static_cast<std::size_t>(idx)));
m_pal->colors.insert(static_cast<std::size_t>(idx + offset), c);
rmTableRow(idx);
addTableRow(idx + offset, c);
void PaletteEditorImGui::moveColor(int idx, int offset) noexcept {
const auto c = m_pal.colors[static_cast<std::size_t>(idx)];
oxIgnoreError(m_pal.colors.erase(static_cast<std::size_t>(idx)));
m_pal.colors.insert(static_cast<std::size_t>(idx + offset), c);
setUnsavedChanges(true);
}
void PaletteEditor::saveItem() {
m_ctx->project->writeObj(m_itemPath, m_pal.get());
void PaletteEditorImGui::saveItem() {
//m_ctx->project->writeObj(m_itemPath, &m_pal);
}
void PaletteEditor::load() {
disconnect(m_table, &QTableWidget::cellChanged, this, &PaletteEditor::cellChanged);
for (std::size_t i = 0; i < m_pal->colors.size(); ++i) {
auto c = m_pal->colors[i];
addTableRow(i, c);
}
connect(m_table, &QTableWidget::cellChanged, this, &PaletteEditor::cellChanged);
}
Color16 PaletteEditor::rowColor(int row) const {
auto r = static_cast<uint8_t>(m_table->item(row, 0)->text().toInt());
auto g = static_cast<uint8_t>(m_table->item(row, 1)->text().toInt());
auto b = static_cast<uint8_t>(m_table->item(row, 2)->text().toInt());
return color16(r, g, b);
}
void PaletteEditor::colorSelected() {
auto selIdxs = m_table->selectionModel()->selectedIndexes();
auto row = !selIdxs.empty() ? selIdxs[0].row() : -1;
if (row > -1) {
m_rmBtn->setEnabled(true);
m_moveUpBtn->setEnabled(row > 0);
m_moveDownBtn->setEnabled(row < m_table->rowCount() - 1);
} else {
m_rmBtn->setEnabled(false);
m_moveUpBtn->setEnabled(false);
m_moveDownBtn->setEnabled(false);
}
}
void PaletteEditor::cellChanged(int row, int) {
auto oldColor = m_pal->colors[static_cast<std::size_t>(row)];
auto newColor = rowColor(row);
undoStack()->push(new UpdateColorCommand(this, row, oldColor, newColor));
}
void PaletteEditor::addColorClicked() {
auto row = m_table->rowCount();
undoStack()->push(new AddColorCommand(this, 0, row));
}
void PaletteEditor::rmColorClicked() {
auto row = m_table->currentRow();
undoStack()->push(new RemoveColorCommand(this, m_pal->colors[static_cast<std::size_t>(row)], row));
}
void PaletteEditor::moveColorUpClicked() {
auto row = m_table->currentRow();
undoStack()->push(new MoveColorCommand(this, row, -1));
}
void PaletteEditor::moveColorDownClicked() {
auto row = m_table->currentRow();
undoStack()->push(new MoveColorCommand(this, row, 1));
ox::Error PaletteEditorImGui::markUnsavedChanges(int) noexcept {
setUnsavedChanges(true);
return OxError(0);
}
}

View File

@ -4,22 +4,14 @@
#pragma once
#include <QStyledItemDelegate>
#include "nostalgia/studio/lib/undostack.hpp"
#include <nostalgia/core/gfx.hpp>
#include <nostalgia/studio/studio.hpp>
#include <ox/std/memory.hpp>
namespace nostalgia::core {
struct PaletteEditorColorTableDelegate: public QStyledItemDelegate {
QWidget *createEditor(QWidget *parent, const QStyleOptionViewItem&, const QModelIndex &idx) const;
void paint(QPainter *painter, const QStyleOptionViewItem &opt, const QModelIndex &idx) const;
};
class PaletteEditor: public studio::Editor {
class PaletteEditorImGui: public studio::Editor {
friend class AddColorCommand;
friend class RemoveColorCommand;
@ -27,58 +19,40 @@ class PaletteEditor: public studio::Editor {
friend class MoveColorCommand;
private:
PaletteEditorColorTableDelegate m_colorTableDelegate;
const studio::Context *m_ctx = nullptr;
QString m_itemPath;
std::unique_ptr<NostalgiaPalette> m_pal;
class QTableWidget *m_table = nullptr;
class QPushButton *m_addBtn = nullptr;
class QPushButton *m_rmBtn = nullptr;
class QPushButton *m_moveUpBtn = nullptr;
class QPushButton *m_moveDownBtn = nullptr;
Context *m_ctx = nullptr;
ox::String m_itemName;
ox::String m_itemPath;
Palette m_pal;
std::size_t m_selectedRow = 0;
studio::UndoStack m_undoStack;
PaletteEditorImGui() noexcept = default;
public:
PaletteEditor(QString path, const studio::Context *ctx, QWidget *parent);
static ox::Result<PaletteEditorImGui*> make(Context *ctx, const ox::String &path) noexcept;
/**
* Returns the name of item being edited.
*/
QString itemName() const override;
const ox::String &itemName() const override;
void draw(core::Context*) noexcept override;
studio::UndoStack *undoStack() noexcept final;
protected:
void addColor(int idx, Color16 c);
void addColor(int idx, Color16 c) noexcept;
void rmColor(int idx);
void rmColor(int idx) noexcept;
void updateColor(int idx, Color16);
void updateColor(int idx, Color16) noexcept;
void moveColor(int idx, int offset);
void moveColor(int idx, int offset) noexcept;
void saveItem() override;
private:
void load();
[[nodiscard]] Color16 rowColor(int i) const;
void addTableRow(int i, Color16 c);
void rmTableRow(int i);
void setTableRow(int idx, Color16 c);
private slots:
void colorSelected();
void cellChanged(int row, int col);
void addColorClicked();
void rmColorClicked();
void moveColorUpClicked();
void moveColorDownClicked();
ox::Error markUnsavedChanges(int) noexcept;
};

View File

@ -17,7 +17,7 @@ TileSheetEditorImGui::TileSheetEditorImGui(Context *ctx, const ox::String &path)
const auto lastSlash = std::find(m_itemPath.rbegin(), m_itemPath.rend(), '/').offset();
m_itemName = m_itemPath.substr(lastSlash + 1);
// init palette idx
const auto palPath = model()->palPath();
const auto &palPath = model()->palPath();
auto sctx = applicationData<studio::StudioContext>(m_ctx);
const auto &palList = sctx->project->fileList(core::FileExt_npal + 1);
for (std::size_t i = 0; const auto &pal : palList) {
@ -32,11 +32,11 @@ TileSheetEditorImGui::TileSheetEditorImGui(Context *ctx, const ox::String &path)
m_subsheetEditor.inputSubmitted.connect(this, &TileSheetEditorImGui::updateActiveSubsheet);
}
ox::String TileSheetEditorImGui::itemName() const noexcept {
const ox::String &TileSheetEditorImGui::itemName() const noexcept {
return m_itemPath;
}
ox::String TileSheetEditorImGui::itemDisplayName() const noexcept {
const ox::String &TileSheetEditorImGui::itemDisplayName() const noexcept {
return m_itemName;
}
@ -303,7 +303,7 @@ ox::Error TileSheetEditorImGui::markUnsavedChanges(int) noexcept {
void TileSheetEditorImGui::SubSheetEditor::draw() noexcept {
constexpr auto modalFlags = ImGuiWindowFlags_NoCollapse | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoResize;
constexpr auto popupName = "Edit SubSheet";
constexpr auto popupName = "Edit Subsheet";
if (!m_show) {
return;
}

View File

@ -58,9 +58,9 @@ class TileSheetEditorImGui: public studio::Editor {
~TileSheetEditorImGui() override = default;
ox::String itemName() const noexcept override;
const ox::String &itemName() const noexcept override;
ox::String itemDisplayName() const noexcept override;
const ox::String &itemDisplayName() const noexcept override;
void exportFile() override;

View File

@ -8,7 +8,7 @@
namespace nostalgia::studio {
ox::String Editor::itemDisplayName() const {
const ox::String &Editor::itemDisplayName() const {
return itemName();
}

View File

@ -30,10 +30,10 @@ class NOSTALGIASTUDIO_EXPORT Editor: public Widget {
* Returns the name of item being edited.
*/
[[nodiscard]]
virtual ox::String itemName() const = 0;
virtual const ox::String &itemName() const = 0;
[[nodiscard]]
virtual ox::String itemDisplayName() const;
virtual const ox::String &itemDisplayName() const;
virtual void cut();

View File

@ -14,7 +14,7 @@
namespace nostalgia::studio {
struct EditorMaker {
using Func = std::function<class Editor*(ox::String)>;
using Func = std::function<ox::Result<class Editor*>(const ox::String&)>;
ox::Vector<ox::String> fileTypes;
Func make;
};

View File

@ -110,8 +110,8 @@ ox::Error Project::writeObj(const ox::String &path, auto *obj) const noexcept {
// replace garbage last character with new line
typeOut.back().value = '\n';
// write to FS
static constexpr auto descPath = "/.nostalgia/type_descriptors/";
const auto typePath = ox::sfmt("{}{}", descPath, type->typeName);
static constexpr auto descPath = "/.nostalgia/type_descriptors";
const auto typePath = ox::sfmt("{}/{}", descPath, type->typeName);
oxReturnError(mkdir(descPath));
oxReturnError(writeBuff(typePath, typeOut));
fileUpdated.emit(path);
@ -136,7 +136,7 @@ ox::Error Project::subscribe(ProjectEvent e, ox::SignalHandler *tgt, Functor &&s
case ProjectEvent::FileRecognized:
{
oxRequire(files, listFiles());
for (auto f : files) {
for (const auto &f : files) {
slot(f);
}
connect(this, &Project::fileRecognized, tgt, slot);

View File

@ -182,7 +182,7 @@ void StudioUI::drawTabBar() noexcept {
void StudioUI::drawTabs() noexcept {
for (auto it = m_editors.begin(); it != m_editors.end();) {
auto const &e = *it;
bool open = true;
auto open = true;
if (ImGui::BeginTabItem(e->itemDisplayName().c_str(), &open)) {
m_acitveEditor = e.get();
e->draw(m_ctx);
@ -289,14 +289,13 @@ ox::Error StudioUI::openFile(const ox::String &path) noexcept {
if (!m_editorMakers.contains(ext)) {
return OxError(1, "There is no editor for this file extension");
}
try {
auto editor = m_editorMakers[ext](path);
auto [editor, err] = m_editorMakers[ext](path);
if (err) {
oxErrorf("Could not open Editor: {} ({}:{})", err.msg, err.file, err.line);
return err;
}
editor->closed.connect(this, &StudioUI::closeFile);
m_editors.emplace_back(editor);
} catch (const ox::Exception &ex) {
oxErrorf("Could not open Editor: {} ({}:{})", ex.msg, ex.file, ex.line);
return ex.toError();
}
m_openFiles.emplace_back(path);
// save to config
studio::editConfig<StudioConfig>(m_ctx, [&](StudioConfig *config) {