commit 1dc4145306bf1730ed6f8824b83686f168c66218 parent d2b3f33ed902d67009a7c32ea9200049fbf9a79e Author: Shawn Laptiste <[email protected]> Date: Thu, 8 Sep 2022 15:45:39 -0400 Various bug fixes; Removed old code Change-Id: Id9cd49ed3cffd98ada99923a537b0dbf3ce351a0 Diffstat:
24 files changed, 345 insertions(+), 382 deletions(-)
diff --git a/Readme.md b/Readme.md @@ -21,7 +21,7 @@ ReaWwise is a REAPER extension that sound designers can use to transfer audio fi ### Manual installation 1. Go to the the [releases page](https://github.com/audiokinetic/ReaWwise/releases), then download the extension file (reaper_reawwise.dll for Windows or reaper_reawwise.dylib for Mac). -2. Copy the extension to the REAPER UserPlugins directory. +2. Copy the extension to the REAPER [UserPlugins directory](#userplugins-directory). 3. If REAPER is running, restart the application. The extension is available in the Extensions menu. @@ -48,16 +48,31 @@ ReaWwise is a REAPER extension that sound designers can use to transfer audio fi git clone https://github.com/audiokinetic/ReaWwise.git cd ReaWwise git submodule update --init -mkdir build -cmake --build ./build --target ReaWwise --config Release +cmake -B build +cmake --build build --target ReaWwise --config Release ``` -After the binary is built, move it to the REAPER UserPlugins directory. If REAPER is open, restart the application. The extension is available in the **Extensions** menu. +After the binary is built, move it to the REAPER [UserPlugins directory](#userplugins-directory). If REAPER is open, restart the application. The extension is available in the **Extensions** menu. + +## Basic Usage +1. Open your REAPER and Wwise projects. +2. Configure your REAPER render settings appropriately. The render settings determine which files will be transferred to Wwise. +3. Open ReaWwise and configure settings such as the Originals Subfolder (optional), Import Destination, Wwise Structures, and so on. +4. Preview the audio files and Wwise objects to be transfered in the Preview Panel. +4. Click **Transfer To Wwise** to transfer the audio files and create the corresponding Wwise objects. + +## UserPlugins Directory +- Windows: %appdata%\REAPER\UserPlugins +- Mac: ~/Library/Application Support/REAPER/UserPlugins ## Other Resources Refer to the [ReaWwise User Guide](https://audiokinetic.com/library/reawwise) for more information regarding usage, installation and troubleshooting of the extension. +For general questions about usage, refer to Audiokinetic's [Community Q&A](https://www.audiokinetic.com/qa/ask). + ## Contributing -The repository is not open to pull request but in the case of a bug report, bugfix or a suggestions, please feel free to [open an issue](https://github.com/this/ReaWwise/issues). +The repository is not open to pull request but in the case of a bug report, bugfix or a suggestions, please feel free to [open an issue](https://github.com/audiokinetic/ReaWwise/issues). + +Feature requests can also be submitted to the [feature request section](https://www.audiokinetic.com/qa/feature-requests/) of Audiokinetic's Community Q&A. Use ReaWwise as the Category when submitting a question. ## Legal Copyright © 2020 [Audiokinetic Inc.](https://audiokinetic.com) All rights reserved. diff --git a/src/extension/Extension.cpp b/src/extension/Extension.cpp @@ -3,6 +3,9 @@ #include "ExtensionWindow.h" #include "ReaperContext.h" +#include "Core/WaapiClient.h" +#include "Theme/CustomLookAndFeel.h" + #include <JSONHelpers.h> #include <juce_events/juce_events.h> #include <memory> @@ -727,7 +730,7 @@ namespace AK::ReaWwise if(!reaperContext->registerFunction("hookcommand", (void*)onHookCommand)) { return 0; - }; + } if(!reaperContext->registerFunction("hookcustommenu", (void*)onHookCustomMenu)) { diff --git a/src/extension/ExtensionWindow.cpp b/src/extension/ExtensionWindow.cpp @@ -22,7 +22,7 @@ namespace AK::ReaWwise juce::LookAndFeel::setDefaultLookAndFeel(&lookAndFeel); - mainContentComponent.reset(new WwiseTransfer::MainComponent(dawContext, JUCE_APPLICATION_NAME_STRING)); + auto mainContentComponent = new WwiseTransfer::MainComponent(dawContext, JUCE_APPLICATION_NAME_STRING); #ifdef WIN32 if(!mainContentComponent->hasScaleFactorOverride()) @@ -32,12 +32,17 @@ namespace AK::ReaWwise } #endif - setContentNonOwned(mainContentComponent.get(), true); + setContentOwned(mainContentComponent, true); centreWithSize(width, height); setResizable(true, true); setResizeLimits(minWidth, minHeight, (std::numeric_limits<int>::max)(), (std::numeric_limits<int>::max)()); } + ExtensionWindow::~ExtensionWindow() + { + juce::LookAndFeel::setDefaultLookAndFeel(nullptr); + } + int ExtensionWindow::getDesktopWindowStyleFlags() const { return juce::ComponentPeer::windowHasCloseButton | juce::ComponentPeer::windowHasTitleBar | @@ -49,4 +54,10 @@ namespace AK::ReaWwise { setVisible(false); } + + void ExtensionWindow::resized() + { + juce::ResizableWindow::resized(); + } + } // namespace AK::ReaWwise diff --git a/src/extension/ExtensionWindow.h b/src/extension/ExtensionWindow.h @@ -1,7 +1,7 @@ #pragma once #include "Core/DawContext.h" -#include "UI/MainComponent.h" +#include "Theme/CustomLookAndFeel.h" namespace AK::ReaWwise { @@ -9,12 +9,15 @@ namespace AK::ReaWwise { public: ExtensionWindow(WwiseTransfer::DawContext& dawContext); + ~ExtensionWindow() override; int getDesktopWindowStyleFlags() const override; void userTriedToCloseWindow() override; + protected: + void resized() override; + private: - std::unique_ptr<WwiseTransfer::MainComponent> mainContentComponent; WwiseTransfer::CustomLookAndFeel lookAndFeel; }; } // namespace AK::ReaWwise diff --git a/src/extension/ReaperContext.cpp b/src/extension/ReaperContext.cpp @@ -19,32 +19,6 @@ namespace AK::ReaWwise const juce::String applicationKey = "ReaWwise"; } // namespace ReaperContextConstants - struct RenderInfo - { - juce::String resolvedRenderPath; - juce::String resolvedRenderPathWithOriginalsSubfolder; - juce::String finalRenderPath; - }; - - struct TimeRange - { - double start{}; - double end{}; - - bool overlaps(const TimeRange& other) - { - return start <= other.end && end >= other.start; - } - }; - - struct ReaperMarker - { - const char* name; - bool isRegion; - TimeRange timeRange; - int index; - }; - ReaperContext::ReaperContext(reaper_plugin_info_t* pluginInfo) : pluginInfo(pluginInfo) , defaultRenderDirectory(juce::File::getSpecialLocation(juce::File::userDocumentsDirectory).getChildFile("REAPER Media")) @@ -81,9 +55,8 @@ namespace AK::ReaWwise auto projectInfo = getProjectInfo(); - auto applicationStateString = applicationState.toXmlString(); - auto applicationStateStringSize = juce::String(applicationStateString.getNumBytesAsUTF8()); - + const auto applicationStateString = applicationState.toXmlString(); + const auto applicationStateStringSize = juce::String(applicationStateString.getNumBytesAsUTF8()); if(setProjExtState(projectInfo.projectReference, applicationKey.toUTF8(), stateSizeKey.toUTF8(), applicationStateStringSize.toUTF8()) && setProjExtState(projectInfo.projectReference, applicationKey.toUTF8(), stateKey.toUTF8(), applicationStateString.toUTF8())) { @@ -98,27 +71,20 @@ namespace AK::ReaWwise { using namespace ReaperContextConstants; - juce::ValueTree state; - auto projectInfo = getProjectInfo(); - std::string buffer(ReaperContextConstants::defaultBufferSize, '\0'); + std::string buffer(defaultBufferSize, '\0'); getProjExtState(projectInfo.projectReference, applicationKey.toUTF8(), stateSizeKey.toUTF8(), &buffer[0], buffer.size()); - auto stateSize = std::strtoll(&buffer[0], nullptr, 10); - - if(stateSize != 0) - { - buffer.resize(stateSize); + const auto stateSize = std::strtoll(&buffer[0], nullptr, 10); + if(stateSize == 0) + return {}; - if(getProjExtState(projectInfo.projectReference, applicationKey.toUTF8(), stateKey.toUTF8(), &buffer[0], buffer.size())) - { - auto valueTree = juce::ValueTree::fromXml(buffer); - state = valueTree; - } - } + buffer.resize(stateSize); + if(getProjExtState(projectInfo.projectReference, applicationKey.toUTF8(), stateKey.toUTF8(), &buffer[0], buffer.size())) + return juce::ValueTree::fromXml(buffer); - return state; + return {}; } int ReaperContext::callerVersion() @@ -167,19 +133,67 @@ namespace AK::ReaWwise return false; } - void ReaperContext::renderImportItems() + void ReaperContext::renderItems() { main_OnCommand(42230, 0); } - std::vector<WwiseTransfer::Import::Item> ReaperContext::getImportItems(WwiseTransfer::Import::Options options) + std::vector<WwiseTransfer::Import::Item> ReaperContext::getItemsForImport(const WwiseTransfer::Import::Options& options) { - using namespace ReaperContextConstants; + std::vector<WwiseTransfer::Import::Item> importItems; + + auto importItemsForPreview = getItemsForPreview(options); + if(importItemsForPreview.size() == 0) + return importItems; + + auto projectInfo = getProjectInfo(); + + // In the following section, we retrieve render stats from reaper. We need to estimate the size of the buffer so that + // no heap corruption occurs when reaper tries to write to the buffer we give it. Unfortunately, there is no way + // for us to know the exact size of this buffer should be. And, it seems like reaper blindly writes to this buffer + // as we can not send it a buffer size. + + // Assuming that volume levels are in [-1000, 1000], this is the max length for an entry for one render would be: + // FILE:{audioFilePath};PEAK:-0000.000000;LRA:-0000.000000;LUFSMMAX:-0000.000000;LUFSSMAX:-0000.000000;LUFSI:-0000.000000; + // This is roughly 105 chars + the audioFileName + // Lets use 150 for extra chars just to leave some room for error. + const int extra = 150; + + juce::String renderDirectory = getProjectString(projectInfo.projectReference, "RENDER_FILE"); + + int totalBufferSize = (renderDirectory.length() + projectInfo.projectPath.length() + extra) * static_cast<int>(importItemsForPreview.size()); + for(const auto& item : importItemsForPreview) + { + totalBufferSize += item.audioFilePath.length(); + } + + juce::StringArray temp; + juce::String renderStats = getProjectString(projectInfo.projectReference, "RENDER_STATS", totalBufferSize); + temp.addTokens(renderStats, ";", ""); + + juce::StringArray finalRenderPaths; + + // Not all items in here are paths. Paths are prefixed with "FILE:" + for(auto item : temp) + { + if(item.startsWith("FILE:")) + finalRenderPaths.add(item.trimCharactersAtStart("FILE:")); + } + + for(int i = 0; i < importItemsForPreview.size(); ++i) + { + importItems.push_back({importItemsForPreview[i].path, importItemsForPreview[i].originalsSubFolder, importItemsForPreview[i].audioFilePath, finalRenderPaths[i]}); + } + + return importItems; + } + std::vector<WwiseTransfer::Import::PreviewItem> ReaperContext::getItemsForPreview(const WwiseTransfer::Import::Options& options) + { auto projectInfo = getProjectInfo(); - auto renderDirectoryPath = getProjectString(projectInfo.projectReference, "RENDER_FILE", defaultBufferSize); - auto renderPattern = getProjectString(projectInfo.projectReference, "RENDER_PATTERN", defaultBufferSize); + auto renderDirectoryPath = getProjectString(projectInfo.projectReference, "RENDER_FILE"); + auto renderPattern = getProjectString(projectInfo.projectReference, "RENDER_PATTERN"); // There are several scenarios where the render pattern could be empty // 1. When the project hasn't been saved (reaper uses "untitled") @@ -211,74 +225,32 @@ namespace AK::ReaWwise renderDirectory = defaultRenderDirectory.getChildFile(renderDirectoryPath); } - auto renderDirectoryPathPart = renderDirectoryPath.isNotEmpty() ? renderDirectoryPath + juce::File::getSeparatorString() : ""; - auto originalsSubfolderPathPart = options.originalsSubfolder.isNotEmpty() ? options.originalsSubfolder + juce::File::getSeparatorString() : ""; - - auto resolvedRenderPaths = getItemListFromRenderPattern(projectInfo.projectReference, renderDirectoryPathPart + renderPattern); - auto resolvedRenderPathsWithOriginalsSubfolder = getItemListFromRenderPattern(projectInfo.projectReference, renderDirectoryPathPart + originalsSubfolderPathPart + renderPattern); - + const auto originalsSubfolderPathPart = options.originalsSubfolder.isNotEmpty() ? options.originalsSubfolder + juce::File::getSeparatorString() : ""; + const auto resolvedRenderPaths = getItemListFromRenderPattern(projectInfo.projectReference, renderPattern, false); + const auto resolvedRenderPathsWithOriginalsSubfolder = getItemListFromRenderPattern(projectInfo.projectReference, originalsSubfolderPathPart + renderPattern, false); if(resolvedRenderPaths.size() != resolvedRenderPathsWithOriginalsSubfolder.size()) { juce::Logger::writeToLog("Reaper: Mismatch between resolvedRenderPaths and resolvedRenderPathsWithOriginalsSubfolder"); return {}; } - // Estimate the size of render stats so we can pass reaper the propper buffer size - int totalBufferSize = 0; - int extra = 150; // Each path may come with aditional render statistics (peak information). - for(const auto& resolvedRenderPath : resolvedRenderPaths) - { - totalBufferSize += resolvedRenderPath.length() + extra; - } - - // Get list of files that where rendered during the last render run using the current render settings and render pattern - // These file names may differ from the resolved render paths due to silent renaming. - juce::StringArray finalRenderPaths; - - if(totalBufferSize > 0) - { - juce::String renderStats = getProjectString(projectInfo.projectReference, "RENDER_STATS", totalBufferSize); - - juce::StringArray temp; - temp.addTokens(renderStats, ";", ""); - - // Not all items in here are paths. Paths are prefixed with "FILE:" - for(auto item : temp) - { - if(item.startsWith("FILE:")) - finalRenderPaths.add(item.trimCharactersAtStart("FILE:")); - } - } - - std::vector<WwiseTransfer::Import::Item> importItems; - - auto resolvePattern = options.importDestination + options.hierarchyMappingPath; - + const auto resolvePattern = options.importDestination + options.hierarchyMappingPath; std::vector<juce::String> resolvedObjectPaths = getItemListFromRenderPattern(projectInfo.projectReference, resolvePattern, false); - if(resolvedObjectPaths.size() != resolvedRenderPaths.size() || resolvedObjectPaths.size() != resolvedRenderPathsWithOriginalsSubfolder.size()) { juce::Logger::writeToLog("Reaper: Mismatch between resolvedObjectPaths, resolvedRenderPaths and resolvedRenderPathsWithOriginalsSubfolder"); return {}; } + std::vector<WwiseTransfer::Import::PreviewItem> importItems; for(int i = 0; i < resolvedObjectPaths.size(); ++i) { - juce::String objectPath(resolvedObjectPaths[i].upToLastOccurrenceOf(".", false, false)); - - juce::File parentDirectory = juce::File(resolvedRenderPathsWithOriginalsSubfolder[i]) - .getParentDirectory(); - - juce::String currentOriginalsSubfolder = parentDirectory == renderDirectory ? "" : parentDirectory.getRelativePathFrom(renderDirectory); - - auto objectName = WwiseTransfer::WwiseHelper::pathToObjectName(objectPath); - - importItems.emplace_back(objectName, - WwiseTransfer::Wwise::ObjectType::SoundSFX, - objectPath, - currentOriginalsSubfolder, - resolvedRenderPaths[i], - finalRenderPaths[i]); + const auto objectPath = resolvedObjectPaths[i].upToLastOccurrenceOf(".", false, false); + const auto resolvedRenderPath = renderDirectory.getChildFile(resolvedRenderPaths[i]).getFullPathName(); + const auto resolvedRenderPathWithOriginalsSubfolder = renderDirectory.getChildFile(resolvedRenderPathsWithOriginalsSubfolder[i]); + const auto parentDirectory = resolvedRenderPathWithOriginalsSubfolder.getParentDirectory().getRelativePathFrom(renderDirectory); + const auto originalsSubfolder = parentDirectory == "." ? "" : parentDirectory; + importItems.push_back({objectPath, originalsSubfolder, resolvedRenderPath}); } return importItems; @@ -286,54 +258,35 @@ namespace AK::ReaWwise juce::String ReaperContext::getProjectString(ReaProject* project, const char* key, int bufferSize) { - std::string buffer(bufferSize, '\0'); - - if(getSetProjectInfo_String(project, key, &buffer[0], false)) - { - auto end = buffer.find('\0'); - buffer.resize(end); - return buffer; - } + // getSetProjectInfo_String is unsafe since we cannot specify the size of the buffer that it must fill. + // There are a few cases where we are unable to predict the size required, so we use a large value and hope for the best. + std::string buffer((std::max)(bufferSize, ReaperContextConstants::defaultBufferSize), '\0'); + if(!getSetProjectInfo_String(project, key, &buffer[0], false)) + return {}; - return juce::String(); + return buffer; } ProjectInfo ReaperContext::getProjectInfo() { std::string buffer(ReaperContextConstants::defaultBufferSize, '\0'); - ReaProject* projectReference; - - // The buffer sent to enumProjects will contain the project path. It is a requirement that a project file ends with the .rpp (case insensitive) file extension. - while((projectReference = enumProjects(-1, &buffer[0], buffer.size())) && !juce::String(buffer).endsWithIgnoreCase(".rpp")) - { - if(buffer.size() > 10 << 20 || buffer.empty()) - { - jassertfalse; + // The buffer sent to enumProjects will contain the project path. + auto projectReference = enumProjects(-1, &buffer[0], buffer.size()); + if(!projectReference || buffer.empty()) + return {}; - return {}; - } - else - { - try - { - buffer.resize(2 * buffer.size()); - } - catch(std::bad_alloc&) - { - jassertfalse; - - return {}; - } - } - } + const juce::File projectFile(buffer); - juce::File projectFile(buffer); + // REAPER requires that a project file ends with the .rpp (case insensitive) file extension. + if(projectFile.getFileExtension().compareIgnoreCase(".rpp") != 0) + return {}; return { projectReference, projectFile.getFileNameWithoutExtension(), - projectFile.getFullPathName()}; + projectFile.getFullPathName() + }; } std::vector<juce::String> ReaperContext::getItemListFromRenderPattern(ReaProject* project, const juce::String& pattern, bool suppressIllegalPaths) diff --git a/src/extension/ReaperContext.h b/src/extension/ReaperContext.h @@ -23,8 +23,9 @@ namespace AK::ReaWwise juce::String getSessionName() override; bool saveState(juce::ValueTree applicationState) override; juce::ValueTree retrieveState() override; - void renderImportItems() override; - std::vector<WwiseTransfer::Import::Item> getImportItems(WwiseTransfer::Import::Options options) override; + void renderItems() override; + std::vector<WwiseTransfer::Import::PreviewItem> getItemsForPreview(const WwiseTransfer::Import::Options& options) override; + std::vector<WwiseTransfer::Import::Item> getItemsForImport(const WwiseTransfer::Import::Options& options) override; // Reaper api for public use bool isValid(); @@ -51,7 +52,7 @@ namespace AK::ReaWwise private: std::vector<juce::String> splitDoubleNullTerminatedString(const char*); std::vector<juce::String> getItemListFromRenderPattern(ReaProject* project, const juce::String& pattern, bool suppressIllegalPaths = true); - juce::String getProjectString(ReaProject* project, const char* key, int bufferSize); + juce::String getProjectString(ReaProject* project, const char* key, int bufferSize = 0); ProjectInfo getProjectInfo(); juce::File defaultRenderDirectory; diff --git a/src/shared/Core/DawContext.h b/src/shared/Core/DawContext.h @@ -15,7 +15,8 @@ namespace AK::WwiseTransfer virtual juce::String getSessionName() = 0; virtual bool saveState(juce::ValueTree applicationState) = 0; virtual juce::ValueTree retrieveState() = 0; - virtual void renderImportItems() = 0; - virtual std::vector<Import::Item> getImportItems(Import::Options options) = 0; + virtual void renderItems() = 0; + virtual std::vector<Import::PreviewItem> getItemsForPreview(const Import::Options& options) = 0; + virtual std::vector<Import::Item> getItemsForImport(const Import::Options& options) = 0; }; } // namespace AK::WwiseTransfer diff --git a/src/shared/Core/DawWatcher.cpp b/src/shared/Core/DawWatcher.cpp @@ -40,7 +40,7 @@ namespace AK::WwiseTransfer , languageSubfolder(appState, IDs::languageSubfolder, nullptr) { auto featureSupport = appState.getChildWithName(IDs::featureSupport); - waqlEnabled.referTo(featureSupport, IDs::sessionName, nullptr); + waqlEnabled.referTo(featureSupport, IDs::waqlEnabled, nullptr); applicationState.addListener(this); } @@ -80,9 +80,9 @@ namespace AK::WwiseTransfer if(importOptions && previewOptions) { - auto importItemsFromDaw = dawContext.getImportItems(*importOptions); + auto importItemsFromDaw = dawContext.getItemsForPreview(*importOptions); - auto currentImportItemsFromDawHash = ImportHelper::importItemsToHash(importItemsFromDaw); + auto currentImportItemsFromDawHash = ImportHelper::importPreviewItemsToHash(importItemsFromDaw); if(lastImportItemsFromDawHash != currentImportItemsFromDawHash || previewOptionsChanged) { @@ -113,7 +113,7 @@ namespace AK::WwiseTransfer auto name = WwiseHelper::pathToObjectName(pathWithoutType); auto type = WwiseHelper::pathToObjectType(ancestorPath); - child = ImportHelper::previewItemToValueTree(pathWithoutType, {name, type, Import::ObjectStatus::New, "", Import::WavStatus::Unknown}); + child = ImportHelper::previewItemNodeToValueTree(pathWithoutType, {name, type, Import::ObjectStatus::New, "", Import::WavStatus::Unknown}); currentNode.appendChild(child, nullptr); pathToValueTreeMapping[pathWithoutType] = child; @@ -144,7 +144,7 @@ namespace AK::WwiseTransfer wavStatus = Import::WavStatus::New; } - auto child = ImportHelper::previewItemToValueTree(pathWithoutType, {name, type, Import::ObjectStatus::New, originalsWav, wavStatus}); + auto child = ImportHelper::previewItemNodeToValueTree(pathWithoutType, {name, type, Import::ObjectStatus::New, originalsWav, wavStatus}); currentNode.appendChild(child, nullptr); pathToValueTreeMapping[pathWithoutType] = child; @@ -171,7 +171,7 @@ namespace AK::WwiseTransfer if(it != pathToValueTreeMappingLocal.end()) { - auto previewItem = ImportHelper::valueTreeToPreviewItem(it->second); + auto previewItem = ImportHelper::valueTreeToPreviewItemNode(it->second); if(existingObject.type != Wwise::ObjectType::Sound || previewOptions->containerNameExists == Import::ContainerNameExistsOption::UseExisting) previewItem.objectStatus = Import::ObjectStatus::NoChange; @@ -185,7 +185,7 @@ namespace AK::WwiseTransfer previewItem.type = existingObject.type; } - it->second.copyPropertiesFrom(ImportHelper::previewItemToValueTree(existingObject.path, previewItem), nullptr); + it->second.copyPropertiesFrom(ImportHelper::previewItemNodeToValueTree(existingObject.path, previewItem), nullptr); } } } diff --git a/src/shared/Core/WaapiClient.cpp b/src/shared/Core/WaapiClient.cpp @@ -270,7 +270,7 @@ namespace AK::WwiseTransfer return Unsubscribe(in_subscriptionId, out_result, in_timeoutMs); } - bool WaapiClient::isConnected() + bool WaapiClient::isConnected() const { return IsConnected(); } @@ -306,68 +306,6 @@ namespace AK::WwiseTransfer return status; } - Waapi::Response<Waapi::ObjectResponseSet> WaapiClient::getObjects(const std::set<juce::String>& objectPaths) - { - using namespace WwiseAuthoringAPI; - - Waapi::Response<Waapi::ObjectResponseSet> response; - - if(!objectPaths.empty()) - { - juce::String waql("where "); - - for(auto it = objectPaths.begin(); it != objectPaths.end(); ++it) - { - if(it != objectPaths.begin()) - waql << " or "; - waql << "path = \"" - << *it - << "\""; - } - - const auto args = AkJson::Map{ - { - "waql", - AkVariant{waql.toStdString()}, - }, - }; - - static const auto options = AkJson::Map{ - { - "return", - AkJson::Array{ - AkVariant{"id"}, - AkVariant{"name"}, - AkVariant{"type"}, - AkVariant{"path"}, - }, - }, - }; - - AkJson result; - response.status = call(WaapiCommands::objectGet, args, options, result); - - if(response.status) - { - if(result.HasKey("return")) - { - auto objects = result["return"].GetArray(); - - for(auto& object : objects) - { - response.result.emplace(object); - } - } - } - else - { - response.errorMessage << WaapiHelper::getErrorMessage(result); - } - } - - return response; - } - Waapi::Response<Waapi::ObjectResponseSet> WaapiClient::import(const std::vector<Waapi::ImportItemRequest>& importItemsRequest, Import::ContainerNameExistsOption containerNameExistsOption, const juce::String& objectLanguage) { using namespace WwiseAuthoringAPI; @@ -498,6 +436,7 @@ namespace AK::WwiseTransfer AkVariant{"name"}, AkVariant{"type"}, AkVariant{"path"}, + AkVariant{"workunitType"}, }, }, }; @@ -547,6 +486,7 @@ namespace AK::WwiseTransfer return response; } + Waapi::Response<std::vector<juce::String>> WaapiClient::getProjectLanguages() { using namespace WwiseAuthoringAPI; @@ -669,6 +609,7 @@ namespace AK::WwiseTransfer AkVariant{"name"}, AkVariant{"type"}, AkVariant{"path"}, + AkVariant{"workunitType"}, }, }, }; diff --git a/src/shared/Core/WaapiClient.h b/src/shared/Core/WaapiClient.h @@ -74,7 +74,7 @@ namespace AK::WwiseTransfer bool connect(const char* in_uri, unsigned int in_port, WwiseAuthoringAPI::disconnectHandler_t disconnectHandler = nullptr, int in_timeoutMs = -1); bool subscribe(const char* in_uri, const WwiseAuthoringAPI::AkJson& in_options, WampEventCallback in_callback, uint64_t& out_subscriptionId, WwiseAuthoringAPI::AkJson& out_result, int in_timeoutMs = -1); bool unsubscribe(const uint64_t& in_subscriptionId, WwiseAuthoringAPI::AkJson& out_result, int in_timeoutMs = -1); - bool isConnected(); + bool isConnected() const; void disconnect(); bool call(const char* in_uri, const WwiseAuthoringAPI::AkJson& in_args, const WwiseAuthoringAPI::AkJson& in_options, WwiseAuthoringAPI::AkJson& out_result, int in_timeoutMs = -1); bool call(const char* in_uri, const char* in_args, const char* in_options, std::string& out_result, int in_timeoutMs = -1); @@ -83,7 +83,6 @@ namespace AK::WwiseTransfer Waapi::Response<Waapi::ProjectInfo> getProjectInfo(); Waapi::Response<Waapi::ObjectResponse> getSelectedObject(); Waapi::Response<Waapi::ObjectResponseSet> pasteProperties(const Waapi::PastePropertiesRequest& pastePropertiesRequest); - Waapi::Response<Waapi::ObjectResponseSet> getObjects(const std::set<juce::String>& objectPaths); Waapi::Response<Waapi::ObjectResponseSet> import(const std::vector<Waapi::ImportItemRequest>& importItemsRequest, Import::ContainerNameExistsOption containerNameExistsOption, const juce::String& objectLanguage); Waapi::Response<Waapi::ObjectResponseSet> getObjectAncestorsAndDescendants(const juce::String& objectPath); Waapi::Response<Waapi::ObjectResponseSet> getObjectAncestorsAndDescendantsLegacy(const juce::String& objectPath); @@ -141,17 +140,6 @@ namespace AK::WwiseTransfer } template <typename Callback> - void getObjectsAsync(const std::set<juce::String>& objectPaths, const Callback& callback) - { - auto onJobExecute = [this, objectPaths = objectPaths]() - { - return getObjects(objectPaths); - }; - - threadPool.addJob(new AsyncJob(onJobExecute, callback), true); - } - - template <typename Callback> void getProjectLanguagesAsync(const Callback& callback) { auto onJobExecute = [this]() diff --git a/src/shared/Helpers/ImportHelper.h b/src/shared/Helpers/ImportHelper.h @@ -53,9 +53,9 @@ namespace AK::WwiseTransfer::ImportHelper valueTree[IDs::objectLanguage]); } - inline Import::PreviewItem valueTreeToPreviewItem(juce::ValueTree valueTree) + inline Import::PreviewItemNode valueTreeToPreviewItemNode(juce::ValueTree valueTree) { - return Import::PreviewItem{ + return Import::PreviewItemNode{ valueTree[IDs::objectName], juce::VariantConverter<Wwise::ObjectType>::fromVar(valueTree[IDs::objectType]), juce::VariantConverter<Import::ObjectStatus>::fromVar(valueTree[IDs::objectStatus]), @@ -64,7 +64,7 @@ namespace AK::WwiseTransfer::ImportHelper }; } - inline juce::ValueTree previewItemToValueTree(const juce::String& path, Import::PreviewItem previewItem) + inline juce::ValueTree previewItemNodeToValueTree(const juce::String& path, Import::PreviewItemNode previewItem) { juce::ValueTree valueTree(path); valueTree.setProperty(IDs::objectName, previewItem.name, nullptr); @@ -201,7 +201,7 @@ namespace AK::WwiseTransfer::ImportHelper } } - inline unsigned int importItemsToHash(const std::vector<Import::Item>& importItems) + inline unsigned int importPreviewItemsToHash(const std::vector<Import::PreviewItem>& importItems) { AK::FNVHash32 hash; diff --git a/src/shared/Model/Import.h b/src/shared/Model/Import.h @@ -40,43 +40,25 @@ namespace AK::WwiseTransfer::Import New }; - struct Item + struct PreviewItem { - Item(const juce::String& name, Wwise::ObjectType type, const juce::String& path, const juce::String& originalsSubFolder, const juce::String& audioFilePath, const juce::String& renderFilePath) - : name(name) - , type(type) - , path(path) - , originalsSubFolder(originalsSubFolder) - , audioFilePath(audioFilePath) - , renderFilePath(renderFilePath) - { - } - - juce::String name; - Wwise::ObjectType type; juce::String path; juce::String originalsSubFolder; juce::String audioFilePath; - juce::String renderFilePath; }; - struct PreviewItem + struct Item : public PreviewItem { - juce::String name; - Wwise::ObjectType type; - ObjectStatus objectStatus; - juce::String audioFilePath; - WavStatus wavStatus; + juce::String renderFilePath; }; - struct PreviewItemNode : public Item + struct PreviewItemNode { - PreviewItemNode(const Item& item) - : Item(item.name, item.type, item.path, item.originalsSubFolder, item.audioFilePath, item.renderFilePath) - { - } - - std::unordered_map<juce::String, PreviewItemNode> children; + juce::String name; + Wwise::ObjectType type{}; + ObjectStatus objectStatus{}; + juce::String audioFilePath; + WavStatus wavStatus{}; }; struct HierarchyMappingNode diff --git a/src/shared/Model/Waapi.h b/src/shared/Model/Waapi.h @@ -31,6 +31,15 @@ namespace AK::WwiseTransfer::Waapi , path(json.HasKey("path") ? json["path"].GetVariant().GetString() : "") , originalWavFilePath(json.HasKey("sound:originalWavFilePath") ? json["sound:originalWavFilePath"].GetVariant().GetString() : "") { + if(json.HasKey("workunitType")) + { + auto workUnitType = json["workunitType"].GetVariant().GetString(); + + if(path == "\\Actor-Mixer Hierarchy") + type = Wwise::ObjectType::ActorMixer; + else if(workUnitType == "folder") + type = Wwise::ObjectType::PhysicalFolder; + } } ObjectResponse() = default; @@ -47,7 +56,7 @@ namespace AK::WwiseTransfer::Waapi juce::String id; juce::String name; - Wwise::ObjectType type; + Wwise::ObjectType type{ Wwise::ObjectType::Unknown }; juce::String path; juce::String originalWavFilePath; }; diff --git a/src/shared/UI/AboutComponent.cpp b/src/shared/UI/AboutComponent.cpp @@ -28,6 +28,7 @@ namespace AK::WwiseTransfer AboutComponent::AboutComponent(const juce::String& applicationName) : copyRightSymbol(juce::CharPointer_UTF8("\xa9")) , wwiseIcon(juce::Drawable::createFromImageData(BinaryData::wwise_icon_svg, BinaryData::wwise_icon_svgSize)) + , tooltipWindow(this) { using namespace AboutComponentConstants; @@ -74,6 +75,11 @@ namespace AK::WwiseTransfer "Build: " + juce::String(BUILD_NUMBER)); } + AboutComponent::~AboutComponent() + { + setLookAndFeel(nullptr); + } + void AboutComponent::resized() { using namespace AboutComponentConstants; diff --git a/src/shared/UI/AboutComponent.h b/src/shared/UI/AboutComponent.h @@ -10,6 +10,7 @@ namespace AK::WwiseTransfer { public: AboutComponent(const juce::String& applicationName); + ~AboutComponent() override; void resized() override; void paint(juce::Graphics& g); @@ -27,6 +28,8 @@ namespace AK::WwiseTransfer juce::HyperlinkButton releaseNotesLink; juce::HyperlinkButton licensingLink; + juce::TooltipWindow tooltipWindow; + std::unique_ptr<juce::Drawable> wwiseIcon; }; } // namespace AK::WwiseTransfer diff --git a/src/shared/UI/ImportConflictsComponent.cpp b/src/shared/UI/ImportConflictsComponent.cpp @@ -48,6 +48,40 @@ namespace AK::WwiseTransfer audioFileNameExistsComboBox.getSelectedIdAsValue().referTo(audioFilenameExists.getPropertyAsValue()); applyTemplateComboBox.getSelectedIdAsValue().referTo(applyTemplate.getPropertyAsValue()); + auto containerNameExistsComboBoxOnChange = [this] + { + const char* tooltip = ""; + switch(static_cast<Import::ContainerNameExistsOption>(containerNameExistsComboBox.getSelectedId())) + { + case Import::ContainerNameExistsOption::UseExisting: + tooltip = "Retains the existing sounds but adds the transferred files as source audio " + "files. The new files do not become the main audio sources for the existing " + "objects unless they have identical names to the existing sources, and " + "therefore overwrite them."; + break; + + case Import::ContainerNameExistsOption::CreateNew: + tooltip = "Creates new sounds with increments appended to the names, for example " + "SoundName_01. The existing sounds are not affected, unless the existing audio " + "sources have the same names as the newly transferred files. In that case, the " + "new sources overwrite the existing ones."; + break; + + case Import::ContainerNameExistsOption::Replace: + tooltip = "Deletes and recreates the sounds, with the transferred files as sources."; + break; + + default: + jassertfalse; + break; + } + + containerNameExistsComboBox.setTooltip(tooltip); + }; + + containerNameExistsComboBoxOnChange(); + containerNameExistsComboBox.onChange = containerNameExistsComboBoxOnChange; + addAndMakeVisible(containerNameExistsLabel); addAndMakeVisible(audioFileNameExistsLabel); addAndMakeVisible(applyTemplateLabel); diff --git a/src/shared/UI/ImportControlsComponent.cpp b/src/shared/UI/ImportControlsComponent.cpp @@ -71,20 +71,21 @@ namespace AK::WwiseTransfer { using namespace ImportControlsComponentConstants; - juce::Logger::writeToLog("Sending render request to daw"); + // Disbale the import button while rendering + importButton.setEnabled(false); - dawContext.renderImportItems(); + juce::Logger::writeToLog("Sending render request to DAW"); + dawContext.renderItems(); - auto hierarchyMappingPath = ImportHelper::hierarchyMappingToPath(ImportHelper::valueTreeToHierarchyMappingNodeList(applicationState.getChildWithName(IDs::hierarchyMapping))); + const auto hierarchyMappingPath = ImportHelper::hierarchyMappingToPath(ImportHelper::valueTreeToHierarchyMappingNodeList(applicationState.getChildWithName(IDs::hierarchyMapping))); + const Import::Options opts(importDestination, originalsSubFolder, hierarchyMappingPath); + const auto importItems = dawContext.getItemsForImport(opts); - Import::Options opts(importDestination, originalsSubFolder, hierarchyMappingPath); - - auto importItems = dawContext.getImportItems(opts); - - auto showRenameWarning = false, showRenderFailed = false; - if(importItems.size()) + bool showRenameWarning = false; + bool showRenderFailed = false; + if(importItems.size() > 0) { - for(auto& importItem : importItems) + for(const auto& importItem : importItems) { if(importItem.renderFilePath.isEmpty()) { @@ -98,10 +99,24 @@ namespace AK::WwiseTransfer } } } + else + { + juce::Logger::writeToLog("No items to import."); + importButton.setEnabled(true); + return; + } - auto hierarchyMappingNodeList = ImportHelper::valueTreeToHierarchyMappingNodeList(hierarchyMapping); + if(showRenderFailed) + { + const juce::String message("One or more files failed to render."); + juce::Logger::writeToLog(message); + juce::AlertWindow::showMessageBoxAsync(juce::MessageBoxIconType::InfoIcon, "Import Aborted", message); + importButton.setEnabled(true); + return; + } - Import::Task::Options importTaskOptions{ + const auto hierarchyMappingNodeList = ImportHelper::valueTreeToHierarchyMappingNodeList(hierarchyMapping); + const Import::Task::Options importTaskOptions{ importItems, containerNameExistsOption, applyTemplateOption, @@ -114,56 +129,54 @@ namespace AK::WwiseTransfer auto onImportComplete = [this, importTaskOptions = importTaskOptions](const Import::Summary& importSummary) { showImportSummary(importSummary, importTaskOptions); + importButton.setEnabled(true); }; importTask.reset(new ImportTask(waapiClient, importTaskOptions, onImportComplete)); - if(showRenderFailed) + if(!showRenameWarning || !applicationProperties.getShowSilentIncrementWarning()) { - juce::String message("One or more files failed to render."); + juce::Logger::writeToLog("Importing files..."); + importTask->launchThread(); + return; + } - juce::Logger::writeToLog(message); + const juce::String message("Several file names where silently incremented to avoid overwriting during the render process."); + juce::Logger::writeToLog(message); - juce::AlertWindow::showMessageBoxAsync(juce::MessageBoxIconType::InfoIcon, "Import Aborted", message); - } - else if(showRenameWarning && applicationProperties.getShowSilentIncrementWarning()) + auto onDialogBtnClicked = [this, importItems = importItems](int result) { - juce::String message("Several file names where silently incremented to avoid overwriting during the render process."); - - juce::Logger::writeToLog(message); + applicationProperties.setShowSilentIncrementWarning(!showSilentIncrementWarningToggle.getToggleState()); - auto onDialogBtnClicked = [this, importItems = importItems](int result) + if(result == MessageBoxOption::Continue) { - applicationProperties.setShowSilentIncrementWarning(!showSilentIncrementWarningToggle.getToggleState()); + juce::Logger::writeToLog("Importing files..."); + importTask->launchThread(); + return; + } - if(result == MessageBoxOption::Continue) - { - importTask->launchThread(); - } - }; + juce::Logger::writeToLog("Import aborted."); + importTask.reset(); + importButton.setEnabled(true); + }; - auto messageBoxOptions = juce::MessageBoxOptions() - .withTitle("Action Required") - .withMessage(message) - .withButton("Continue") - .withButton("Cancel"); + auto messageBoxOptions = juce::MessageBoxOptions() + .withTitle("Action Required") + .withMessage(message) + .withButton("Continue") + .withButton("Cancel"); - juce::AlertWindow::showAsync(messageBoxOptions, onDialogBtnClicked); + juce::AlertWindow::showAsync(messageBoxOptions, onDialogBtnClicked); - auto modalManager = juce::ModalComponentManager::getInstance(); - auto alertWindow = dynamic_cast<juce::AlertWindow*>(modalManager->getModalComponent(0)); - alertWindow->addCustomComponent(&showSilentIncrementWarningToggle); + auto modalManager = juce::ModalComponentManager::getInstance(); + auto alertWindow = dynamic_cast<juce::AlertWindow*>(modalManager->getModalComponent(0)); + alertWindow->addCustomComponent(&showSilentIncrementWarningToggle); - // Reset and reposition the toggle button - showSilentIncrementWarningToggle.setToggleState(false, true); - auto bounds = showSilentIncrementWarningToggle.getBounds(); - bounds.setX(showSilentIncrementWarningToggleMarginLeft); - showSilentIncrementWarningToggle.setBounds(bounds); - } - else - { - importTask->launchThread(); - } + // Reset and reposition the toggle button + showSilentIncrementWarningToggle.setToggleState(false, true); + auto bounds = showSilentIncrementWarningToggle.getBounds(); + bounds.setX(showSilentIncrementWarningToggleMarginLeft); + showSilentIncrementWarningToggle.setBounds(bounds); } void ImportControlsComponent::showImportSummary(const Import::Summary& summary, const Import::Task::Options& importTaskOptions) diff --git a/src/shared/UI/ImportPreviewComponent.cpp b/src/shared/UI/ImportPreviewComponent.cpp @@ -234,7 +234,7 @@ namespace AK::WwiseTransfer { using namespace ImportPreviewComponentConstants; - auto previewItem = ImportHelper::valueTreeToPreviewItem(tree); + auto previewItem = ImportHelper::valueTreeToPreviewItemNode(tree); std::array<juce::String, 4> cellText{ previewItem.name, @@ -258,7 +258,7 @@ namespace AK::WwiseTransfer g.setFont(CustomLookAndFeelConstants::smallFontSize); - if(trueColumnWidth > 0) + if(trueColumnWidth > iconSize) { icon->drawWithin(g, juce::Rectangle<float>(iconSize, iconSize).reduced(iconPadding, iconPadding), juce::RectanglePlacement::centred, 1); @@ -312,7 +312,7 @@ namespace AK::WwiseTransfer juce::String ValueTreeItem::getComparisonTextForColumn(int column) { - auto previewItem = ImportHelper::valueTreeToPreviewItem(tree); + auto previewItem = ImportHelper::valueTreeToPreviewItemNode(tree); switch(column) { diff --git a/src/shared/UI/TruncatableTextEditor.cpp b/src/shared/UI/TruncatableTextEditor.cpp @@ -26,7 +26,6 @@ namespace AK::WwiseTransfer void TruncatableTextEditor::resized() { juce::TextEditor::resized(); - resetText(); } @@ -37,24 +36,28 @@ namespace AK::WwiseTransfer void TruncatableTextEditor::resetText() { - auto indent = getLeftIndent() * 4; - - juce::String text = valueToTruncate.getValue(); - - auto trueTextSpace = getWidth() - juce::Font().getStringWidth("...") - indent; + juce::String text{ "..." }; + const auto indent = getLeftIndent() * 4; + const auto ellipsisWidth = getFont().getStringWidth(text); + const auto width = getWidth(); + const auto trueTextSpace = width - ellipsisWidth - indent; - if(text.isNotEmpty() && juce::Font().getStringWidth(text) > trueTextSpace) + if(trueTextSpace > 0) { - auto midPoint = 0; - while(juce::Font().getStringWidth(text) > trueTextSpace) + text = valueToTruncate.getValue(); + if(text.isNotEmpty() && getFont().getStringWidth(text) > trueTextSpace) { - midPoint = juce::roundFloatToInt(float(text.length()) / 2.0f); - text = text.substring(0, midPoint) + text.substring(midPoint + 1); + int midPoint{}; + while(getFont().getStringWidth(text) > trueTextSpace) + { + midPoint = text.length() / 2; + text = text.substring(0, midPoint) + text.substring(midPoint + 1); + } + + text = text.substring(0, midPoint) + "..." + text.substring(midPoint); } - - text = text.substring(0, midPoint) + "..." + text.substring(midPoint); } setText(text); - }; + } } // namespace AK::WwiseTransfer diff --git a/src/standalone/StandaloneWindow.cpp b/src/standalone/StandaloneWindow.cpp @@ -28,6 +28,11 @@ namespace AK::WwiseTransfer setVisible(true); } + StandaloneWindow::~StandaloneWindow() + { + juce::LookAndFeel::setDefaultLookAndFeel(nullptr); + } + int StandaloneWindow::getDesktopWindowStyleFlags() const { return juce::ComponentPeer::windowHasCloseButton | juce::ComponentPeer::windowHasTitleBar | diff --git a/src/standalone/StandaloneWindow.h b/src/standalone/StandaloneWindow.h @@ -9,6 +9,7 @@ namespace AK::WwiseTransfer { public: StandaloneWindow(); + ~StandaloneWindow() override; int getDesktopWindowStyleFlags() const override; void userTriedToCloseWindow() override; diff --git a/src/standalone/StubContext.h b/src/standalone/StubContext.h @@ -10,12 +10,17 @@ namespace AK::WwiseTransfer : public DawContext { public: - std::vector<Import::Item> getImportItems(Import::Options options) override + std::vector<Import::PreviewItem> getItemsForPreview(const Import::Options& options) override { - return std::vector<Import::Item>{Import::Item("test", Wwise::ObjectType::SoundSFX, "\\A\\B\\C", juce::String(), juce::String(), juce::String())}; + return std::vector<Import::PreviewItem>{{"\\A\\B\\C", juce::String(), juce::String()}}; } - void renderImportItems() override + std::vector<Import::Item> getItemsForImport(const Import::Options& options) override + { + return std::vector<Import::Item>{{"\\A\\B\\C", juce::String(), juce::String(), juce::String()}}; + } + + void renderItems() override { } @@ -23,12 +28,12 @@ namespace AK::WwiseTransfer juce::String getSessionName() override { return juce::String(); - }; + } bool saveState(juce::ValueTree applicationState) override { return true; - }; + } juce::ValueTree retrieveState() override { diff --git a/src/test/ImportHelperTest.cpp b/src/test/ImportHelperTest.cpp @@ -17,7 +17,7 @@ namespace AK::WwiseTransfer::Test auto nodes = std::vector<Import::HierarchyMappingNode>(); auto testValuesVector = std::vector<TestTreeHierarchyValues>(); - for (int index = 0; index < indexLimit; index++) + for(int index = 0; index < indexLimit; index++) { auto testValues = TestTreeHierarchyValues(index); testValuesVector.emplace_back(testValues); @@ -27,7 +27,7 @@ namespace AK::WwiseTransfer::Test auto valueTrees = ImportHelper::hierachyMappingNodeListToValueTree(nodes); - for (int index = 0; index < indexLimit; index++) + for(int index = 0; index < indexLimit; index++) { testValueTreeEquality(valueTrees.getChild(index), testValuesVector[index]); } @@ -42,11 +42,11 @@ namespace AK::WwiseTransfer::Test testHierarchyMappingEquality(hierarchyMapping, testValues); } - TEST_CASE("valueTreeToPreviewItem") + TEST_CASE("valueTreeToPreviewItemNode") { auto testValues = TestTreePreviewItemValues(); auto valueTree = testValues.generateValueTree(); - auto previewItem = ImportHelper::valueTreeToPreviewItem(valueTree); + auto previewItem = ImportHelper::valueTreeToPreviewItemNode(valueTree); REQUIRE(previewItem.audioFilePath == testValues.audioFilePath); REQUIRE(previewItem.name == testValues.objectName); @@ -55,13 +55,13 @@ namespace AK::WwiseTransfer::Test REQUIRE(previewItem.wavStatus == testValues.wavStatus); } - TEST_CASE("previewItemToValueTree") + TEST_CASE("previewItemNodeToValueTree") { auto testValues = TestTreePreviewItemValues(); auto previewItem = testValues.generatePreviewItem(); auto testPath = "\\test\\path"; - auto valueTree = ImportHelper::previewItemToValueTree(testPath, previewItem); + auto valueTree = ImportHelper::previewItemNodeToValueTree(testPath, previewItem); REQUIRE(valueTree.getType().toString() == testPath); REQUIRE(valueTree.getProperty(IDs::objectName) == testValues.objectName); @@ -77,7 +77,7 @@ namespace AK::WwiseTransfer::Test auto valueTrees = juce::ValueTree("root"); auto testValuesVector = std::vector<TestTreeHierarchyValues>(); - for (int index = 0; index < indexLimit; index++) + for(int index = 0; index < indexLimit; index++) { auto testValues = TestTreeHierarchyValues(index); testValuesVector.emplace_back(testValues); @@ -87,7 +87,7 @@ namespace AK::WwiseTransfer::Test auto hierarchyList = ImportHelper::valueTreeToHierarchyMappingNodeList(valueTrees); - for (int index = 0; index < indexLimit; index++) + for(int index = 0; index < indexLimit; index++) { testHierarchyMappingEquality(hierarchyList[index], testValuesVector[index]); } @@ -99,7 +99,7 @@ namespace AK::WwiseTransfer::Test auto nodes = std::vector<Import::HierarchyMappingNode>(); auto testValuesVector = std::vector<TestTreeHierarchyValues>(); - for (int index = 0; index < indexLimit; index++) + for(int index = 0; index < indexLimit; index++) { auto testValues = TestTreeHierarchyValues(index); testValuesVector.emplace_back(testValues); @@ -113,7 +113,7 @@ namespace AK::WwiseTransfer::Test pathList.addTokens(fullPath, "\\", ""); pathList.removeEmptyStrings(); - for (int index = 0; index < pathList.size(); index++) + for(int index = 0; index < pathList.size(); index++) { REQUIRE(pathList[index].contains(testValuesVector[index].objectName)); REQUIRE(pathList[index].contains(WwiseHelper::objectTypeToReadableString(testValuesVector[index].objectType))); @@ -172,35 +172,34 @@ namespace AK::WwiseTransfer::Test REQUIRE(ImportHelper::wavStatusToReadableString(Import::WavStatus::Unknown) == ""); } - TEST_CASE("importItemsToHash") + TEST_CASE("importPreviewItemsToHash") { - SECTION("Equality Check") { auto itemCount = GENERATE(0, 1, 3, 1000); - auto testItems = std::vector<Import::Item>(); - for (int index = 0; index < itemCount; index++) + auto testItems = std::vector<Import::PreviewItem>(); + for(int index = 0; index < itemCount; index++) { - auto testValue = TestImportItemValues(index); - auto testItem = testValue.generateImportItem(); + auto testValue = TestImportPreviewItemValues(index); + auto testItem = testValue.generateImportPreviewItem(); testItems.emplace_back(testItem); } - REQUIRE(ImportHelper::importItemsToHash(testItems) == ImportHelper::importItemsToHash(testItems)); + REQUIRE(ImportHelper::importPreviewItemsToHash(testItems) == ImportHelper::importPreviewItemsToHash(testItems)); } SECTION("Difference Check") { auto evenIndex = GENERATE(2, 3, 2000, 2001); - auto testItems1 = std::vector<Import::Item>(); - auto testItems2 = std::vector<Import::Item>(); + auto testItems1 = std::vector<Import::PreviewItem>(); + auto testItems2 = std::vector<Import::PreviewItem>(); - for (int index = 0; index < evenIndex; index++) + for(int index = 0; index < evenIndex; index++) { - auto testValue = TestImportItemValues(index); - auto testItem = testValue.generateImportItem(); + auto testValue = TestImportPreviewItemValues(index); + auto testItem = testValue.generateImportPreviewItem(); - if (index % 2 == 0) + if(index % 2 == 0) { testItems1.emplace_back(testItem); } @@ -210,7 +209,7 @@ namespace AK::WwiseTransfer::Test } } - REQUIRE(ImportHelper::importItemsToHash(testItems1) != ImportHelper::importItemsToHash(testItems2)); + REQUIRE(ImportHelper::importPreviewItemsToHash(testItems1) != ImportHelper::importPreviewItemsToHash(testItems2)); } } } // namespace AK::WwiseTransfer::Test diff --git a/src/test/ImportHelperTest.h b/src/test/ImportHelperTest.h @@ -1,6 +1,6 @@ #include "Helpers/ImportHelper.h" -#include "Model/Import.h" #include "Model/IDs.h" +#include "Model/Import.h" #include <catch2/catch_all.hpp> #include <catch2/catch_test_macros.hpp> @@ -52,8 +52,7 @@ namespace AK::WwiseTransfer::Test propertyTemplatePathEnabled, propertyTemplatePathValid, propertyTemplatePathErrorMsg, - language - ); + language); return hierarchyMappingNode; } @@ -108,49 +107,37 @@ namespace AK::WwiseTransfer::Test return valueTree; } - inline Import::PreviewItem generatePreviewItem() const + inline Import::PreviewItemNode generatePreviewItem() const { return { objectName, objectType, objectStatus, audioFilePath, - wavStatus - }; + wavStatus}; } }; - struct TestImportItemValues + struct TestImportPreviewItemValues { - juce::String name; - Wwise::ObjectType type; juce::String path; juce::String originalsSubFolder; juce::String audioFilePath; - juce::String renderFilePath; - TestImportItemValues(int index = 0) + TestImportPreviewItemValues(int index = 0) { juce::String strIndex(index); - name = "TestName_" + strIndex; - type = Wwise::ObjectType::PhysicalFolder; path = "\\test\\item\\path_" + strIndex; originalsSubFolder = "testFolder"; audioFilePath = "\\test\\item\\audio\\path_" + strIndex; - renderFilePath = "\\test\\item\\render\\path_" + strIndex; } - inline Import::Item generateImportItem() const + inline Import::PreviewItem generateImportPreviewItem() const { - return - { - name, - type, + return { path, originalsSubFolder, - audioFilePath, - renderFilePath - }; + audioFilePath}; } };