mirror of
https://github.com/AxioDL/amuse.git
synced 2025-12-08 21:17:49 +00:00
Clipboard support and various bug fixes
This commit is contained in:
@@ -1,5 +1,9 @@
|
||||
#include "amuse/AudioGroup.hpp"
|
||||
#include "amuse/AudioGroupData.hpp"
|
||||
#include <regex>
|
||||
#include <athena/FileReader.hpp>
|
||||
|
||||
using namespace std::literals;
|
||||
|
||||
namespace amuse
|
||||
{
|
||||
@@ -20,6 +24,15 @@ void AudioGroup::assign(SystemStringView groupPath)
|
||||
m_proj = AudioGroupProject::CreateAudioGroupProject(groupPath);
|
||||
m_samp = nullptr;
|
||||
}
|
||||
void AudioGroup::assign(const AudioGroup& data, SystemStringView groupPath)
|
||||
{
|
||||
/* Reverse order when loading intermediates */
|
||||
m_groupPath = groupPath;
|
||||
m_sdir = AudioGroupSampleDirectory::CreateAudioGroupSampleDirectory(groupPath);
|
||||
m_pool = AudioGroupPool::CreateAudioGroupPool(groupPath);
|
||||
m_proj = AudioGroupProject::CreateAudioGroupProject(data.getProj());
|
||||
m_samp = nullptr;
|
||||
}
|
||||
|
||||
const SampleEntry* AudioGroup::getSample(SampleId sfxId) const
|
||||
{
|
||||
@@ -93,4 +106,249 @@ void AudioGroup::makeCompressedVersion(SampleId sfxId, const SampleEntry* sample
|
||||
m_sdir._extractCompressed(sfxId, *sample->m_data, m_groupPath, sample->m_data->m_looseData.get(), true);
|
||||
}
|
||||
}
|
||||
|
||||
void AudioGroupDatabase::renameSample(SampleId id, std::string_view str)
|
||||
{
|
||||
SystemString oldBasePath = getSampleBasePath(id);
|
||||
SampleId::CurNameDB->rename(id, str);
|
||||
SystemString newBasePath = getSampleBasePath(id);
|
||||
Rename((oldBasePath + _S(".wav")).c_str(), (newBasePath + _S(".wav")).c_str());
|
||||
Rename((oldBasePath + _S(".dsp")).c_str(), (newBasePath + _S(".dsp")).c_str());
|
||||
Rename((oldBasePath + _S(".vadpcm")).c_str(), (newBasePath + _S(".vadpcm")).c_str());
|
||||
}
|
||||
|
||||
void AudioGroupDatabase::deleteSample(SampleId id)
|
||||
{
|
||||
SystemString basePath = getSampleBasePath(id);
|
||||
Unlink((basePath + _S(".wav")).c_str());
|
||||
Unlink((basePath + _S(".dsp")).c_str());
|
||||
Unlink((basePath + _S(".vadpcm")).c_str());
|
||||
}
|
||||
|
||||
void AudioGroupDatabase::copySampleInto(const SystemString& basePath, const SystemString& newBasePath)
|
||||
{
|
||||
Copy((basePath + _S(".wav")).c_str(), (newBasePath + _S(".wav")).c_str());
|
||||
Copy((basePath + _S(".dsp")).c_str(), (newBasePath + _S(".dsp")).c_str());
|
||||
Copy((basePath + _S(".vadpcm")).c_str(), (newBasePath + _S(".vadpcm")).c_str());
|
||||
}
|
||||
|
||||
void AudioGroupDatabase::_recursiveRenameMacro(SoundMacroId id, std::string_view str, int& macroIdx,
|
||||
std::unordered_set<SoundMacroId>& renamedIds)
|
||||
{
|
||||
if (renamedIds.find(id) != renamedIds.cend())
|
||||
return;
|
||||
if (const SoundMacro* macro = getPool().soundMacro(id))
|
||||
{
|
||||
if (!strncmp(SoundMacroId::CurNameDB->resolveNameFromId(id).data(), "macro", 5))
|
||||
{
|
||||
std::string macroName("macro"sv);
|
||||
if (macroIdx)
|
||||
{
|
||||
char num[16];
|
||||
snprintf(num, 16, "%d", macroIdx);
|
||||
macroName += num;
|
||||
}
|
||||
macroName += '_';
|
||||
macroName += str;
|
||||
++macroIdx;
|
||||
SoundMacroId::CurNameDB->rename(id, macroName);
|
||||
renamedIds.insert(id);
|
||||
int sampleIdx = 0;
|
||||
for (const auto& cmd : macro->m_cmds)
|
||||
{
|
||||
switch (cmd->Isa())
|
||||
{
|
||||
case SoundMacro::CmdOp::StartSample:
|
||||
{
|
||||
SoundMacro::CmdStartSample* ss = static_cast<SoundMacro::CmdStartSample*>(cmd.get());
|
||||
if (!strncmp(SampleId::CurNameDB->resolveNameFromId(ss->sample.id).data(), "sample", 6))
|
||||
{
|
||||
std::string sampleName("sample"sv);
|
||||
if (sampleIdx)
|
||||
{
|
||||
char num[16];
|
||||
snprintf(num, 16, "%d", sampleIdx);
|
||||
sampleName += num;
|
||||
}
|
||||
sampleName += '_';
|
||||
sampleName += macroName;
|
||||
++sampleIdx;
|
||||
renameSample(ss->sample.id, sampleName);
|
||||
}
|
||||
}
|
||||
case SoundMacro::CmdOp::SplitKey:
|
||||
_recursiveRenameMacro(static_cast<SoundMacro::CmdSplitKey*>(cmd.get())->macro, str, macroIdx, renamedIds);
|
||||
break;
|
||||
case SoundMacro::CmdOp::SplitVel:
|
||||
_recursiveRenameMacro(static_cast<SoundMacro::CmdSplitVel*>(cmd.get())->macro, str, macroIdx, renamedIds);
|
||||
break;
|
||||
case SoundMacro::CmdOp::Goto:
|
||||
_recursiveRenameMacro(static_cast<SoundMacro::CmdGoto*>(cmd.get())->macro, str, macroIdx, renamedIds);
|
||||
break;
|
||||
case SoundMacro::CmdOp::PlayMacro:
|
||||
_recursiveRenameMacro(static_cast<SoundMacro::CmdPlayMacro*>(cmd.get())->macro, str, macroIdx, renamedIds);
|
||||
break;
|
||||
case SoundMacro::CmdOp::SplitMod:
|
||||
_recursiveRenameMacro(static_cast<SoundMacro::CmdSplitMod*>(cmd.get())->macro, str, macroIdx, renamedIds);
|
||||
break;
|
||||
case SoundMacro::CmdOp::SplitRnd:
|
||||
_recursiveRenameMacro(static_cast<SoundMacro::CmdSplitRnd*>(cmd.get())->macro, str, macroIdx, renamedIds);
|
||||
break;
|
||||
case SoundMacro::CmdOp::GoSub:
|
||||
_recursiveRenameMacro(static_cast<SoundMacro::CmdGoSub*>(cmd.get())->macro, str, macroIdx, renamedIds);
|
||||
break;
|
||||
case SoundMacro::CmdOp::TrapEvent:
|
||||
_recursiveRenameMacro(static_cast<SoundMacro::CmdTrapEvent*>(cmd.get())->macro, str, macroIdx, renamedIds);
|
||||
break;
|
||||
case SoundMacro::CmdOp::SendMessage:
|
||||
_recursiveRenameMacro(static_cast<SoundMacro::CmdSendMessage*>(cmd.get())->macro, str, macroIdx, renamedIds);
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static const std::regex DefineGRPEntry(R"(#define\s+GRP(\S+)\s+(\S+))",
|
||||
std::regex::ECMAScript|std::regex::optimize);
|
||||
static const std::regex DefineSNGEntry(R"(#define\s+SNG(\S+)\s+(\S+))",
|
||||
std::regex::ECMAScript|std::regex::optimize);
|
||||
static const std::regex DefineSFXEntry(R"(#define\s+SFX(\S+)\s+(\S+))",
|
||||
std::regex::ECMAScript|std::regex::optimize);
|
||||
|
||||
void AudioGroupDatabase::importCHeader(std::string_view header)
|
||||
{
|
||||
setIdDatabases();
|
||||
std::match_results<std::string_view::const_iterator> dirMatch;
|
||||
auto begin = header.cbegin();
|
||||
auto end = header.cend();
|
||||
while (std::regex_search(begin, end, dirMatch, DefineGRPEntry))
|
||||
{
|
||||
std::string key = dirMatch[1].str();
|
||||
std::string value = dirMatch[2].str();
|
||||
char* endPtr;
|
||||
amuse::ObjectId id;
|
||||
id.id = strtoul(value.c_str(), &endPtr, 0);
|
||||
if (endPtr == value.c_str())
|
||||
continue;
|
||||
GroupId::CurNameDB->rename(id, key);
|
||||
begin = dirMatch.suffix().first;
|
||||
}
|
||||
begin = header.cbegin();
|
||||
end = header.cend();
|
||||
while (std::regex_search(begin, end, dirMatch, DefineSNGEntry))
|
||||
{
|
||||
std::string key = dirMatch[1].str();
|
||||
std::string value = dirMatch[2].str();
|
||||
char* endPtr;
|
||||
amuse::ObjectId id;
|
||||
id.id = strtoul(value.c_str(), &endPtr, 0);
|
||||
if (endPtr == value.c_str())
|
||||
continue;
|
||||
SongId::CurNameDB->rename(id, key);
|
||||
begin = dirMatch.suffix().first;
|
||||
}
|
||||
begin = header.cbegin();
|
||||
end = header.cend();
|
||||
std::unordered_set<SoundMacroId> renamedMacroIDs;
|
||||
while (std::regex_search(begin, end, dirMatch, DefineSFXEntry))
|
||||
{
|
||||
std::string key = dirMatch[1].str();
|
||||
std::string value = dirMatch[2].str();
|
||||
char* endPtr;
|
||||
amuse::ObjectId id;
|
||||
id.id = strtoul(value.c_str(), &endPtr, 0);
|
||||
if (endPtr == value.c_str())
|
||||
continue;
|
||||
SFXId::CurNameDB->rename(id, key);
|
||||
int macroIdx = 0;
|
||||
for (auto& sfxGrp : getProj().sfxGroups())
|
||||
{
|
||||
for (auto& sfx : sfxGrp.second->m_sfxEntries)
|
||||
{
|
||||
if (sfx.first == id)
|
||||
{
|
||||
ObjectId sfxObjId = sfx.second.objId;
|
||||
if (sfxObjId == ObjectId() || sfxObjId.id & 0xc000)
|
||||
continue;
|
||||
_recursiveRenameMacro(sfxObjId, key, macroIdx, renamedMacroIDs);
|
||||
}
|
||||
}
|
||||
}
|
||||
begin = dirMatch.suffix().first;
|
||||
}
|
||||
}
|
||||
|
||||
static void WriteDefineLine(std::string& ret, std::string_view typeStr, std::string_view name, ObjectId id)
|
||||
{
|
||||
ret += "#define "sv;
|
||||
ret += typeStr;
|
||||
ret += name;
|
||||
ret += ' ';
|
||||
char idStr[16];
|
||||
snprintf(idStr, 16, "%d", id.id);
|
||||
ret += idStr;
|
||||
ret += '\n';
|
||||
}
|
||||
|
||||
std::string AudioGroupDatabase::exportCHeader(std::string_view projectName, std::string_view groupName) const
|
||||
{
|
||||
setIdDatabases();
|
||||
std::string ret;
|
||||
ret += "/* Auto-generated Amuse Defines\n"
|
||||
" *\n"
|
||||
" * Project: "sv;
|
||||
ret += projectName;
|
||||
ret += "\n"
|
||||
" * Subproject: "sv;
|
||||
ret += groupName;
|
||||
ret += "\n"
|
||||
" * Date: "sv;
|
||||
time_t curTime = time(nullptr);
|
||||
struct tm curTm;
|
||||
localtime_r(&curTime, &curTm);
|
||||
char curTmStr[26];
|
||||
asctime_r(&curTm, curTmStr);
|
||||
ret += curTmStr;
|
||||
ret += "\n"
|
||||
" */\n\n\n"sv;
|
||||
|
||||
for (const auto& sg : SortUnorderedMap(getProj().songGroups()))
|
||||
{
|
||||
auto name = amuse::GroupId::CurNameDB->resolveNameFromId(sg.first);
|
||||
WriteDefineLine(ret, "GRP"sv, name, sg.first);
|
||||
}
|
||||
for (const auto& sg : SortUnorderedMap(getProj().sfxGroups()))
|
||||
{
|
||||
auto name = amuse::GroupId::CurNameDB->resolveNameFromId(sg.first);
|
||||
WriteDefineLine(ret, "GRP"sv, name, sg.first);
|
||||
}
|
||||
|
||||
ret += "\n\n"sv;
|
||||
|
||||
std::unordered_set<amuse::SongId> songIds;
|
||||
for (const auto& sg : getProj().songGroups())
|
||||
for (const auto& song : sg.second->m_midiSetups)
|
||||
songIds.insert(song.first);
|
||||
for (amuse::SongId id : SortUnorderedSet(songIds))
|
||||
{
|
||||
auto name = amuse::SongId::CurNameDB->resolveNameFromId(id);
|
||||
WriteDefineLine(ret, "SNG"sv, name, id);
|
||||
}
|
||||
|
||||
ret += "\n\n"sv;
|
||||
|
||||
for (const auto& sg : SortUnorderedMap(getProj().sfxGroups()))
|
||||
{
|
||||
for (const auto& sfx : SortUnorderedMap(sg.second.get()->m_sfxEntries))
|
||||
{
|
||||
auto name = amuse::SFXId::CurNameDB->resolveNameFromId(sfx.first);
|
||||
WriteDefineLine(ret, "SFX"sv, name, sfx.first.id);
|
||||
}
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user