metaforce/hecl/driver/ToolExtract.hpp

261 lines
8.9 KiB
C++
Raw Normal View History

2015-06-12 09:08:49 +00:00
#ifndef CTOOL_EXTRACT
#define CTOOL_EXTRACT
#include "ToolBase.hpp"
#include <stdio.h>
class ToolExtract final : public ToolBase
{
2015-07-10 05:28:33 +00:00
HECL::Database::IDataSpec::ExtractPassInfo m_einfo;
2015-07-20 23:27:22 +00:00
struct SpecExtractPass
{
const HECL::Database::DataSpecEntry* m_entry;
std::unique_ptr<HECL::Database::IDataSpec> m_instance;
SpecExtractPass(const HECL::Database::DataSpecEntry* entry, HECL::Database::IDataSpec* instance)
: m_entry(entry), m_instance(instance) {}
2015-07-22 19:14:50 +00:00
SpecExtractPass(const SpecExtractPass& other) = delete;
SpecExtractPass(SpecExtractPass&& other) = default;
2015-07-20 23:27:22 +00:00
};
std::vector<SpecExtractPass> m_specPasses;
2015-07-10 05:28:33 +00:00
std::vector<HECL::Database::IDataSpec::ExtractReport> m_reps;
2015-07-28 23:54:54 +00:00
std::unique_ptr<HECL::Database::Project> m_fallbackProj;
HECL::Database::Project* m_useProj;
2015-06-12 09:08:49 +00:00
public:
ToolExtract(const ToolPassInfo& info)
: ToolBase(info)
{
2015-07-10 05:28:33 +00:00
if (!m_info.args.size())
LogModule.report(LogVisor::FatalError, "hecl extract needs a source path as its first argument");
2015-07-28 02:25:33 +00:00
if (!info.project)
{
/* Get name from input file and init project there */
2015-07-28 23:54:54 +00:00
HECL::SystemString baseFile = info.args[0];
2015-07-28 02:25:33 +00:00
size_t slashPos = baseFile.rfind(_S('/'));
if (slashPos == HECL::SystemString::npos)
slashPos = baseFile.rfind(_S('\\'));
if (slashPos != HECL::SystemString::npos)
2015-07-28 23:54:54 +00:00
baseFile.assign(baseFile.begin() + slashPos + 1, baseFile.end());
size_t dotPos = baseFile.rfind(_S('.'));
if (dotPos != HECL::SystemString::npos)
baseFile.assign(baseFile.begin(), baseFile.begin() + dotPos);
if (baseFile.empty())
LogModule.report(LogVisor::FatalError, "hecl extract must be ran within a project directory");
size_t ErrorRef = LogVisor::ErrorCount;
2015-08-05 22:59:59 +00:00
HECL::SystemString rootDir = info.cwd + '/' + baseFile;
HECL::ProjectRootPath newProjRoot(rootDir);
2015-07-28 23:54:54 +00:00
newProjRoot.makeDir();
m_fallbackProj.reset(new HECL::Database::Project(newProjRoot));
if (LogVisor::ErrorCount > ErrorRef)
2015-08-05 22:59:59 +00:00
LogModule.report(LogVisor::FatalError, "unable to init project at '%s'", rootDir.c_str());
LogModule.report(LogVisor::Info, _S("initialized project at '%s/.hecl'"), rootDir.c_str());
2015-07-28 23:54:54 +00:00
m_useProj = m_fallbackProj.get();
2015-07-28 02:25:33 +00:00
}
2015-07-28 23:54:54 +00:00
else
m_useProj = info.project;
2015-07-28 02:25:33 +00:00
2015-07-13 06:30:20 +00:00
m_einfo.srcpath = m_info.args[0];
2015-07-10 05:28:33 +00:00
m_einfo.extractArgs.reserve(info.args.size() - 1);
2015-07-18 04:35:01 +00:00
m_einfo.force = info.force;
2015-07-10 05:28:33 +00:00
for (std::vector<HECL::SystemString>::const_iterator it=info.args.begin() + 1;
it != info.args.end();
++it)
2015-07-13 06:30:20 +00:00
m_einfo.extractArgs.push_back(*it);
2015-07-10 05:28:33 +00:00
2015-07-08 04:26:29 +00:00
for (const HECL::Database::DataSpecEntry* entry : HECL::Database::DATA_SPEC_REGISTRY)
{
2015-07-28 23:54:54 +00:00
HECL::Database::IDataSpec* ds = entry->m_factory(*m_useProj, HECL::Database::TOOL_EXTRACT);
2015-07-12 04:26:49 +00:00
if (ds)
{
2015-08-05 22:59:59 +00:00
if (ds->canExtract(m_einfo, m_reps))
2015-07-20 23:27:22 +00:00
m_specPasses.emplace_back(entry, ds);
2015-07-12 04:26:49 +00:00
else
delete ds;
}
2015-07-08 04:26:29 +00:00
}
2015-06-12 09:08:49 +00:00
}
static void Help(HelpOutput& help)
{
help.secHead(_S("NAME"));
help.beginWrap();
help.wrap(_S("hecl-extract - Extract objects from supported package/image formats\n"));
help.endWrap();
help.secHead(_S("SYNOPSIS"));
help.beginWrap();
2015-07-08 04:26:29 +00:00
help.wrap(_S("hecl extract <packagefile> [<subnode>...]\n"));
2015-06-12 09:08:49 +00:00
help.endWrap();
help.secHead(_S("DESCRIPTION"));
help.beginWrap();
2015-07-22 19:14:50 +00:00
help.wrap(_S("This command recursively extracts all or part of a dataspec-supported ")
_S("package format. Each object is decoded to a working format and added to the project.\n\n"));
2015-06-12 09:08:49 +00:00
help.endWrap();
help.secHead(_S("OPTIONS"));
help.optionHead(_S("<packagefile>[/<subnode>...]"), _S("input file"));
help.beginWrap();
2015-07-22 19:14:50 +00:00
help.wrap(_S("Specifies the package file or disc image to source data from. ")
_S("An optional subnode specifies a named hierarchical-node specific ")
_S("to the game architecture (levels/areas)."));
2015-06-12 09:08:49 +00:00
help.endWrap();
}
HECL::SystemString toolName() const {return _S("extract");}
2015-07-10 05:28:33 +00:00
static void _recursivePrint(int level, HECL::Database::IDataSpec::ExtractReport& rep)
{
for (int l=0 ; l<level ; ++l)
HECL::Printf(_S(" "));
2015-07-22 19:14:50 +00:00
#if _WIN32
HECL::Printf(_S("%s"), rep.name.c_str());
#else
2015-07-10 05:28:33 +00:00
if (XTERM_COLOR)
2015-07-12 04:26:49 +00:00
HECL::Printf(_S("" BOLD "%s" NORMAL ""), rep.name.c_str());
2015-07-10 05:28:33 +00:00
else
2015-07-12 04:26:49 +00:00
HECL::Printf(_S("%s"), rep.name.c_str());
2015-07-22 19:14:50 +00:00
#endif
2015-07-12 04:26:49 +00:00
if (rep.desc.size())
HECL::Printf(_S(" [%s]"), rep.desc.c_str());
HECL::Printf(_S("\n"));
2015-07-10 05:28:33 +00:00
for (HECL::Database::IDataSpec::ExtractReport& child : rep.childOpts)
_recursivePrint(level + 1, child);
}
2015-06-12 09:08:49 +00:00
int run()
{
2015-07-20 23:27:22 +00:00
if (m_specPasses.empty())
2015-07-10 05:28:33 +00:00
{
2015-07-22 19:14:50 +00:00
#if _WIN32
HECL::Printf(_S("NOTHING TO EXTRACT\n"));
#else
2015-07-10 05:28:33 +00:00
if (XTERM_COLOR)
HECL::Printf(_S("" RED BOLD "NOTHING TO EXTRACT" NORMAL "\n"));
else
HECL::Printf(_S("NOTHING TO EXTRACT\n"));
2015-07-22 19:14:50 +00:00
#endif
2015-07-10 05:28:33 +00:00
return -1;
}
2015-07-22 19:14:50 +00:00
#if _WIN32
HECL::Printf(_S("ABOUT TO EXTRACT:\n"));
#else
2015-07-10 05:28:33 +00:00
if (XTERM_COLOR)
HECL::Printf(_S("" GREEN BOLD "ABOUT TO EXTRACT:" NORMAL "\n"));
else
HECL::Printf(_S("ABOUT TO EXTRACT:\n"));
2015-07-22 19:14:50 +00:00
#endif
2015-07-10 05:28:33 +00:00
for (HECL::Database::IDataSpec::ExtractReport& rep : m_reps)
2015-07-13 06:30:20 +00:00
{
2015-07-10 05:28:33 +00:00
_recursivePrint(0, rep);
2015-07-13 06:30:20 +00:00
HECL::Printf(_S("\n"));
}
2015-07-10 05:28:33 +00:00
2015-07-22 19:14:50 +00:00
#if _WIN32
HECL::Printf(_S("\nContinue? (Y/N) "));
#else
2015-07-10 05:28:33 +00:00
if (XTERM_COLOR)
2015-07-12 04:26:49 +00:00
HECL::Printf(_S("\n" BLUE BOLD "Continue?" NORMAL " (Y/N) "));
2015-07-10 05:28:33 +00:00
else
2015-07-12 04:26:49 +00:00
HECL::Printf(_S("\nContinue? (Y/N) "));
2015-07-22 19:14:50 +00:00
#endif
2015-07-10 05:28:33 +00:00
int ch;
while ((ch = getchar()))
{
if (ch == 'n' || ch == 'N')
return 0;
if (ch == 'y' || ch == 'Y')
break;
}
2015-07-20 23:27:22 +00:00
for (SpecExtractPass& ds : m_specPasses)
{
2015-07-22 19:14:50 +00:00
#if _WIN32
HECL::Printf(_S("Using DataSpec %s:\n"), ds.m_entry->m_name);
#else
2015-07-20 23:27:22 +00:00
if (XTERM_COLOR)
2015-07-22 19:14:50 +00:00
HECL::Printf(_S("" MAGENTA BOLD "Using DataSpec %s:" NORMAL "\n"), ds.m_entry->m_name);
2015-07-20 23:27:22 +00:00
else
2015-07-22 19:14:50 +00:00
HECL::Printf(_S("Using DataSpec %s:\n"), ds.m_entry->m_name);
#endif
2015-07-20 23:27:22 +00:00
int lineIdx = 0;
2015-08-05 22:59:59 +00:00
ds.m_instance->doExtract(m_einfo,
2015-07-20 23:27:22 +00:00
[&lineIdx](const HECL::SystemChar* message, int lidx, float factor)
{
2015-07-22 19:14:50 +00:00
#ifndef _WIN32
2015-07-20 23:27:22 +00:00
if (XTERM_COLOR)
HECL::Printf(_S("" HIDE_CURSOR ""));
2015-07-22 19:14:50 +00:00
#endif
2015-07-20 23:27:22 +00:00
if (lidx > lineIdx)
{
HECL::Printf(_S("\n "));
lineIdx = lidx;
}
else
HECL::Printf(_S(" "));
int width = HECL::ConsoleWidth();
int half = width / 2 - 2;
size_t messageLen = HECL::StrLen(message);
if (messageLen > half)
HECL::Printf(_S("%.*s... "), half-3, message);
else
{
HECL::Printf(_S("%s"), message);
for (int i=half-messageLen ; i>=0 ; --i)
HECL::Printf(_S(" "));
}
2015-07-22 19:14:50 +00:00
#ifndef _WIN32
2015-07-20 23:27:22 +00:00
if (XTERM_COLOR)
{
size_t blocks = half - 7;
size_t filled = blocks * factor;
size_t rem = blocks - filled;
HECL::Printf(_S("" BOLD "%3d%% ["), (int)(factor * 100.0));
for (int b=0 ; b<filled ; ++b)
HECL::Printf(_S("#"), message);
for (int b=0 ; b<rem ; ++b)
HECL::Printf(_S("-"), message);
HECL::Printf(_S("]" NORMAL ""));
}
else
{
2015-07-22 19:14:50 +00:00
#endif
2015-07-20 23:27:22 +00:00
size_t blocks = half - 7;
size_t filled = blocks * factor;
size_t rem = blocks - filled;
HECL::Printf(_S("%3d%% ["), (int)(factor * 100.0));
for (int b=0 ; b<filled ; ++b)
HECL::Printf(_S("#"), message);
for (int b=0 ; b<rem ; ++b)
HECL::Printf(_S("-"), message);
HECL::Printf(_S("]"));
2015-07-22 19:14:50 +00:00
#ifndef _WIN32
2015-07-20 23:27:22 +00:00
}
2015-07-22 19:14:50 +00:00
#endif
2015-07-20 23:27:22 +00:00
HECL::Printf(_S("\r"));
2015-07-22 19:14:50 +00:00
#ifndef _WIN32
2015-07-20 23:27:22 +00:00
if (XTERM_COLOR)
HECL::Printf(_S("" SHOW_CURSOR ""));
2015-07-22 19:14:50 +00:00
#endif
2015-07-28 23:54:54 +00:00
fflush(stdout);
2015-07-20 23:27:22 +00:00
});
HECL::Printf(_S("\n\n"));
}
2015-07-10 05:28:33 +00:00
2015-06-12 09:08:49 +00:00
return 0;
}
};
#endif // CTOOL_EXTRACT