Minor refactor in ExportSpriteSheet adding functions to update params from window widgets and update DocExporter from params

This commit is contained in:
David Capello 2019-10-03 10:08:46 -03:00
parent b533e77b3f
commit 7a10139a87
3 changed files with 190 additions and 156 deletions

View File

@ -66,12 +66,12 @@ namespace {
}; };
// Calculate best size for the given sprite // Calculate best size for the given sprite
Fit best_fit(Sprite* sprite, Fit best_fit(const Sprite* sprite,
int nframes, const int nframes,
int borderPadding, const int borderPadding,
int shapePadding, const int shapePadding,
int innerPadding, const int innerPadding,
SpriteSheetType type) { const SpriteSheetType type) {
int framew = sprite->width()+2*innerPadding + shapePadding; int framew = sprite->width()+2*innerPadding + shapePadding;
int frameh = sprite->height()+2*innerPadding + shapePadding; int frameh = sprite->height()+2*innerPadding + shapePadding;
@ -122,11 +122,13 @@ namespace {
return result; return result;
} }
Fit calculate_sheet_size(Sprite* sprite, int nframes, Fit calculate_sheet_size(const Sprite* sprite,
int columns, int rows, const int nframes,
int borderPadding, int columns,
int shapePadding, int rows,
int innerPadding) { const int borderPadding,
const int shapePadding,
const int innerPadding) {
if (columns == 0) { if (columns == 0) {
rows = base::clamp(rows, 1, nframes); rows = base::clamp(rows, 1, nframes);
columns = ((nframes/rows) + ((nframes%rows) > 0 ? 1: 0)); columns = ((nframes/rows) + ((nframes%rows) > 0 ? 1: 0));
@ -142,8 +144,8 @@ namespace {
} }
#ifdef ENABLE_UI #ifdef ENABLE_UI
bool ask_overwrite(bool askFilename, std::string filename, bool ask_overwrite(const bool askFilename, const std::string& filename,
bool askDataname, std::string dataname) { const bool askDataname, const std::string& dataname) {
if ((askFilename && if ((askFilename &&
!filename.empty() && !filename.empty() &&
base::is_file(filename)) || base::is_file(filename)) ||
@ -158,7 +160,8 @@ namespace {
if (base::is_file(dataname)) if (base::is_file(dataname))
text << "<<" << base::get_file_name(dataname).c_str(); text << "<<" << base::get_file_name(dataname).c_str();
int ret = OptionalAlert::show( const int ret =
OptionalAlert::show(
Preferences::instance().spriteSheet.showOverwriteFilesAlert, Preferences::instance().spriteSheet.showOverwriteFilesAlert,
1, // Yes is the default option when the alert dialog is disabled 1, // Yes is the default option when the alert dialog is disabled
fmt::format(Strings::alerts_overwrite_files_on_export_sprite_sheet(), fmt::format(Strings::alerts_overwrite_files_on_export_sprite_sheet(),
@ -170,8 +173,6 @@ namespace {
} }
#endif // ENABLE_UI #endif // ENABLE_UI
}
struct ExportSpriteSheetParams : public NewParams { struct ExportSpriteSheetParams : public NewParams {
Param<bool> ui { this, true, "ui" }; Param<bool> ui { this, true, "ui" };
Param<bool> askOverwrite { this, true, { "askOverwrite", "ask-overwrite" } }; Param<bool> askOverwrite { this, true, { "askOverwrite", "ask-overwrite" } };
@ -198,6 +199,115 @@ struct ExportSpriteSheetParams : public NewParams {
Param<bool> listSlices { this, true, "listSlices" }; Param<bool> listSlices { this, true, "listSlices" };
}; };
void update_doc_exporter_from_params(const Site& site,
const ExportSpriteSheetParams& params,
DocExporter& exporter)
{
const app::SpriteSheetType type = params.type();
int columns = params.columns();
int rows = params.rows();
int width = params.width();
int height = params.height();
const bool bestFit = params.bestFit();
const std::string filename = params.textureFilename();
const std::string dataFilename = params.dataFilename();
const DocExporter::DataFormat dataFormat = params.dataFormat();
const std::string layerName = params.layer();
const std::string tagName = params.tag();
const int borderPadding = base::clamp(params.borderPadding(), 0, 100);
const int shapePadding = base::clamp(params.shapePadding(), 0, 100);
const int innerPadding = base::clamp(params.innerPadding(), 0, 100);
const bool trimCels = params.trim();
const bool trimByGrid = params.trimByGrid();
const bool extrude = params.extrude();
const int extrudePadding = (extrude ? 1: 0);
const bool listLayers = params.listLayers();
const bool listTags = params.listTags();
const bool listSlices = params.listSlices();
SelectedFrames selFrames;
Tag* tag = calculate_selected_frames(site, tagName, selFrames);
frame_t nframes = selFrames.size();
ASSERT(nframes > 0);
const Sprite* sprite = site.sprite();
// If the user choose to render selected layers only, we can
// temporaly make them visible and hide the other ones.
RestoreVisibleLayers layersVisibility;
calculate_visible_layers(site, layerName, layersVisibility);
SelectedLayers selLayers;
if (layerName != kSelectedLayers) {
// TODO add a getLayerByName
for (const Layer* layer : sprite->allLayers()) {
if (layer->name() == layerName) {
selLayers.insert(const_cast<Layer*>(layer));
break;
}
}
}
if (bestFit) {
Fit fit = best_fit(sprite, nframes, borderPadding, shapePadding,
innerPadding + extrudePadding,
type);
columns = fit.columns;
rows = fit.rows;
width = fit.width;
height = fit.height;
}
int sheet_w = 0;
int sheet_h = 0;
switch (type) {
case app::SpriteSheetType::Horizontal:
columns = nframes;
rows = 1;
break;
case app::SpriteSheetType::Vertical:
columns = 1;
rows = nframes;
break;
case app::SpriteSheetType::Rows:
case app::SpriteSheetType::Columns:
if (width > 0) sheet_w = width;
if (height > 0) sheet_h = height;
break;
}
Fit fit = calculate_sheet_size(
sprite, nframes,
columns, rows,
borderPadding, shapePadding, innerPadding + extrudePadding);
if (sheet_w == 0) sheet_w = fit.width;
if (sheet_h == 0) sheet_h = fit.height;
if (!filename.empty())
exporter.setTextureFilename(filename);
if (!dataFilename.empty()) {
exporter.setDataFilename(dataFilename);
exporter.setDataFormat(dataFormat);
}
exporter.setTextureWidth(sheet_w);
exporter.setTextureHeight(sheet_h);
exporter.setSpriteSheetType(type);
exporter.setBorderPadding(borderPadding);
exporter.setShapePadding(shapePadding);
exporter.setInnerPadding(innerPadding);
exporter.setTrimCels(trimCels);
exporter.setTrimByGrid(trimByGrid);
exporter.setExtrude(extrude);
if (listLayers) exporter.setListLayers(true);
if (listTags) exporter.setListTags(true);
if (listSlices) exporter.setListSlices(true);
exporter.addDocument(const_cast<Doc*>(site.document()), tag,
(!selLayers.empty() ? &selLayers: nullptr),
(!selFrames.empty() ? &selFrames: nullptr));
}
#if ENABLE_UI #if ENABLE_UI
class ExportSpriteSheetWindow : public app::gen::ExportSpriteSheet { class ExportSpriteSheetWindow : public app::gen::ExportSpriteSheet {
@ -318,6 +428,32 @@ public:
return closer() == exportButton(); return closer() == exportButton();
} }
void updateParams(ExportSpriteSheetParams& params) {
params.type (spriteSheetTypeValue());
params.columns (columnsValue());
params.rows (rowsValue());
params.width (fitWidthValue());
params.height (fitHeightValue());
params.bestFit (bestFitValue());
params.textureFilename (filenameValue());
params.dataFilename (dataFilenameValue());
params.dataFormat (dataFormatValue());
params.borderPadding (borderPaddingValue());
params.shapePadding (shapePaddingValue());
params.innerPadding (innerPaddingValue());
params.trim (trimValue());
params.trimByGrid (trimByGridValue());
params.extrude (extrudeValue());
params.openGenerated (openGeneratedValue());
params.layer (layerValue());
params.tag (tagValue());
params.listLayers (listLayersValue());
params.listTags (listTagsValue());
params.listSlices (listSlicesValue());
}
private:
app::SpriteSheetType spriteSheetTypeValue() const { app::SpriteSheetType spriteSheetTypeValue() const {
return (app::SpriteSheetType)(sheetType()->getSelectedItemIndex()+1); return (app::SpriteSheetType)(sheetType()->getSelectedItemIndex()+1);
} }
@ -436,8 +572,6 @@ public:
return listSlices()->isSelected(); return listSlices()->isSelected();
} }
private:
void onExport() { void onExport() {
if (!ask_overwrite(m_filenameAskOverwrite, filenameValue(), if (!ask_overwrite(m_filenameAskOverwrite, filenameValue(),
m_dataFilenameAskOverwrite, dataFilenameValue())) m_dataFilenameAskOverwrite, dataFilenameValue()))
@ -684,10 +818,11 @@ private:
#endif // ENABLE_UI #endif // ENABLE_UI
} // anonymous namespace
class ExportSpriteSheetCommand : public CommandWithNewParams<ExportSpriteSheetParams> { class ExportSpriteSheetCommand : public CommandWithNewParams<ExportSpriteSheetParams> {
public: public:
ExportSpriteSheetCommand(); ExportSpriteSheetCommand();
protected: protected:
bool onEnabled(Context* context) override; bool onEnabled(Context* context) override;
void onExecute(Context* context) override; void onExecute(Context* context) override;
@ -707,7 +842,6 @@ void ExportSpriteSheetCommand::onExecute(Context* context)
{ {
Site site = context->activeSite(); Site site = context->activeSite();
Doc* document = site.document(); Doc* document = site.document();
Sprite* sprite = site.sprite();
auto& params = this->params(); auto& params = this->params();
#ifdef ENABLE_UI #ifdef ENABLE_UI
@ -758,28 +892,29 @@ void ExportSpriteSheetCommand::onExecute(Context* context)
if (!window.ok()) if (!window.ok())
return; return;
window.updateParams(params);
docPref.spriteSheet.defined(true); docPref.spriteSheet.defined(true);
docPref.spriteSheet.type (params.type (window.spriteSheetTypeValue())); docPref.spriteSheet.type (params.type());
docPref.spriteSheet.columns (params.columns (window.columnsValue())); docPref.spriteSheet.columns (params.columns());
docPref.spriteSheet.rows (params.rows (window.rowsValue())); docPref.spriteSheet.rows (params.rows());
docPref.spriteSheet.width (params.width (window.fitWidthValue())); docPref.spriteSheet.width (params.width());
docPref.spriteSheet.height (params.height (window.fitHeightValue())); docPref.spriteSheet.height (params.height());
docPref.spriteSheet.bestFit (params.bestFit (window.bestFitValue())); docPref.spriteSheet.bestFit (params.bestFit());
docPref.spriteSheet.textureFilename (params.textureFilename (window.filenameValue())); docPref.spriteSheet.textureFilename (params.textureFilename());
docPref.spriteSheet.dataFilename (params.dataFilename (window.dataFilenameValue())); docPref.spriteSheet.dataFilename (params.dataFilename());
docPref.spriteSheet.dataFormat (params.dataFormat (window.dataFormatValue())); docPref.spriteSheet.dataFormat (params.dataFormat());
docPref.spriteSheet.borderPadding (params.borderPadding (window.borderPaddingValue())); docPref.spriteSheet.borderPadding (params.borderPadding());
docPref.spriteSheet.shapePadding (params.shapePadding (window.shapePaddingValue())); docPref.spriteSheet.shapePadding (params.shapePadding());
docPref.spriteSheet.innerPadding (params.innerPadding (window.innerPaddingValue())); docPref.spriteSheet.innerPadding (params.innerPadding());
docPref.spriteSheet.trim (params.trim (window.trimValue())); docPref.spriteSheet.trim (params.trim());
docPref.spriteSheet.trimByGrid (params.trimByGrid (window.trimByGridValue())); docPref.spriteSheet.trimByGrid (params.trimByGrid());
docPref.spriteSheet.extrude (params.extrude (window.extrudeValue())); docPref.spriteSheet.extrude (params.extrude());
docPref.spriteSheet.openGenerated (params.openGenerated (window.openGeneratedValue())); docPref.spriteSheet.openGenerated (params.openGenerated());
docPref.spriteSheet.layer (params.layer (window.layerValue())); docPref.spriteSheet.layer (params.layer());
docPref.spriteSheet.frameTag (params.tag (window.tagValue())); docPref.spriteSheet.frameTag (params.tag());
docPref.spriteSheet.listLayers (params.listLayers (window.listLayersValue())); docPref.spriteSheet.listLayers (params.listLayers());
docPref.spriteSheet.listFrameTags (params.listTags (window.listTagsValue())); docPref.spriteSheet.listFrameTags (params.listTags());
docPref.spriteSheet.listSlices (params.listSlices (window.listSlicesValue())); docPref.spriteSheet.listSlices (params.listSlices());
// Default preferences for future sprites // Default preferences for future sprites
DocumentPreferences& defPref(Preferences::instance().document(nullptr)); DocumentPreferences& defPref(Preferences::instance().document(nullptr));
@ -793,118 +928,16 @@ void ExportSpriteSheetCommand::onExecute(Context* context)
askOverwrite = false; // Already asked in the ExportSpriteSheetWindow askOverwrite = false; // Already asked in the ExportSpriteSheetWindow
} }
#endif // ENABLE_UI
const app::SpriteSheetType type = params.type();
int columns = params.columns();
int rows = params.rows();
int width = params.width();
int height = params.height();
const bool bestFit = params.bestFit();
const std::string filename = params.textureFilename();
const std::string dataFilename = params.dataFilename();
const DocExporter::DataFormat dataFormat = params.dataFormat();
const std::string layerName = params.layer();
const std::string tagName = params.tag();
const int borderPadding = base::clamp(params.borderPadding(), 0, 100);
const int shapePadding = base::clamp(params.shapePadding(), 0, 100);
const int innerPadding = base::clamp(params.innerPadding(), 0, 100);
const bool trimCels = params.trim();
const bool trimByGrid = params.trimByGrid();
const bool extrude = params.extrude();
const int extrudePadding = (extrude ? 1: 0);
const bool listLayers = params.listLayers();
const bool listTags = params.listTags();
const bool listSlices = params.listSlices();
#ifdef ENABLE_UI
if (context->isUIAvailable() && askOverwrite) { if (context->isUIAvailable() && askOverwrite) {
if (!ask_overwrite(true, filename, if (!ask_overwrite(true, params.textureFilename(),
true, dataFilename)) true, params.dataFilename()))
return; // Do not overwrite return; // Do not overwrite
} }
#endif #endif
SelectedFrames selFrames;
Tag* tag = calculate_selected_frames(site, tagName, selFrames);
frame_t nframes = selFrames.size();
ASSERT(nframes > 0);
// If the user choose to render selected layers only, we can
// temporaly make them visible and hide the other ones.
RestoreVisibleLayers layersVisibility;
calculate_visible_layers(site, layerName, layersVisibility);
SelectedLayers selLayers;
if (layerName != kSelectedLayers) {
// TODO add a getLayerByName
for (Layer* layer : sprite->allLayers()) {
if (layer->name() == layerName) {
selLayers.insert(layer);
break;
}
}
}
if (bestFit) {
Fit fit = best_fit(sprite, nframes, borderPadding, shapePadding,
innerPadding + extrudePadding,
type);
columns = fit.columns;
rows = fit.rows;
width = fit.width;
height = fit.height;
}
int sheet_w = 0;
int sheet_h = 0;
switch (type) {
case app::SpriteSheetType::Horizontal:
columns = nframes;
rows = 1;
break;
case app::SpriteSheetType::Vertical:
columns = 1;
rows = nframes;
break;
case app::SpriteSheetType::Rows:
case app::SpriteSheetType::Columns:
if (width > 0) sheet_w = width;
if (height > 0) sheet_h = height;
break;
}
Fit fit = calculate_sheet_size(
sprite, nframes,
columns, rows,
borderPadding, shapePadding, innerPadding + extrudePadding);
if (sheet_w == 0) sheet_w = fit.width;
if (sheet_h == 0) sheet_h = fit.height;
DocExporter exporter; DocExporter exporter;
if (!filename.empty()) update_doc_exporter_from_params(site, params, exporter);
exporter.setTextureFilename(filename);
if (!dataFilename.empty()) {
exporter.setDataFilename(dataFilename);
exporter.setDataFormat(dataFormat);
}
exporter.setTextureWidth(sheet_w);
exporter.setTextureHeight(sheet_h);
exporter.setSpriteSheetType(type);
exporter.setBorderPadding(borderPadding);
exporter.setShapePadding(shapePadding);
exporter.setInnerPadding(innerPadding);
exporter.setTrimCels(trimCels);
exporter.setTrimByGrid(trimByGrid);
exporter.setExtrude(extrude);
if (listLayers) exporter.setListLayers(true);
if (listTags) exporter.setListTags(true);
if (listSlices) exporter.setListSlices(true);
exporter.addDocument(document, tag,
(!selLayers.empty() ? &selLayers: nullptr),
(!selFrames.empty() ? &selFrames: nullptr));
std::unique_ptr<Doc> newDocument(exporter.exportSheet(context)); std::unique_ptr<Doc> newDocument(exporter.exportSheet(context));
if (!newDocument) if (!newDocument)
@ -929,7 +962,7 @@ void ExportSpriteSheetCommand::onExecute(Context* context)
if (params.openGenerated()) { if (params.openGenerated()) {
// Setup a filename for the new document in case that user didn't // Setup a filename for the new document in case that user didn't
// save the file/specified one output filename. // save the file/specified one output filename.
if (filename.empty()) { if (params.textureFilename().empty()) {
std::string fn = document->filename(); std::string fn = document->filename();
std::string ext = base::get_file_extension(fn); std::string ext = base::get_file_extension(fn);
if (!ext.empty()) if (!ext.empty())
@ -939,7 +972,6 @@ void ExportSpriteSheetCommand::onExecute(Context* context)
base::join_path(base::get_file_path(fn), base::join_path(base::get_file_path(fn),
base::get_file_title(fn) + "-Sheet") + ext); base::get_file_title(fn) + "-Sheet") + ext);
} }
newDocument->setContext(context); newDocument->setContext(context);
newDocument.release(); newDocument.release();
} }

View File

@ -1,4 +1,5 @@
// Aseprite // Aseprite
// Copyright (C) 2019 Igara Studio S.A.
// Copyright (C) 2016-2018 David Capello // Copyright (C) 2016-2018 David Capello
// //
// This program is distributed under the terms of // This program is distributed under the terms of
@ -109,18 +110,18 @@ void fill_anidir_combobox(ui::ComboBox* anidir, doc::AniDir defAnidir)
anidir->setSelectedItemIndex(int(defAnidir)); anidir->setSelectedItemIndex(int(defAnidir));
} }
void calculate_visible_layers(Site& site, void calculate_visible_layers(const Site& site,
const std::string& layersValue, const std::string& layersValue,
RestoreVisibleLayers& layersVisibility) RestoreVisibleLayers& layersVisibility)
{ {
if (layersValue == kSelectedLayers) { if (layersValue == kSelectedLayers) {
if (!site.selectedLayers().empty()) { if (!site.selectedLayers().empty()) {
layersVisibility.showSelectedLayers( layersVisibility.showSelectedLayers(
site.sprite(), const_cast<Sprite*>(site.sprite()),
site.selectedLayers()); site.selectedLayers());
} }
else { else {
layersVisibility.showLayer(site.layer()); layersVisibility.showLayer(const_cast<Layer*>(site.layer()));
} }
} }
else if (layersValue != kAllFrames) { else if (layersValue != kAllFrames) {

View File

@ -1,4 +1,5 @@
// Aseprite // Aseprite
// Copyright (C) 2019 Igara Studio S.A.
// Copyright (C) 2016-2018 David Capello // Copyright (C) 2016-2018 David Capello
// //
// This program is distributed under the terms of // This program is distributed under the terms of
@ -55,7 +56,7 @@ namespace app {
void fill_frames_combobox(const doc::Sprite* sprite, ui::ComboBox* frames, const std::string& defFrame); void fill_frames_combobox(const doc::Sprite* sprite, ui::ComboBox* frames, const std::string& defFrame);
void fill_anidir_combobox(ui::ComboBox* anidir, doc::AniDir defAnidir); void fill_anidir_combobox(ui::ComboBox* anidir, doc::AniDir defAnidir);
void calculate_visible_layers(Site& site, void calculate_visible_layers(const Site& site,
const std::string& layersValue, const std::string& layersValue,
RestoreVisibleLayers& layersVisibility); RestoreVisibleLayers& layersVisibility);