commit 9176d7766f7a7e80ca0598b8001938c3cc8b5471 parent 7872f5e4bc537122af8a4a72747919725ac5de19 Author: jatinchowdhury18 <[email protected]> Date: Wed, 11 Dec 2019 16:06:48 -0800 Major plugin updates, moving toward version 2 (#9) * Major plugin updates, moving toward version 2 * Version 2: general fixes, fix Linux compilation error, pluginval issue * Make pluginval timeout longer for Linux Diffstat:
85 files changed, 1324 insertions(+), 2090 deletions(-)
diff --git a/.gitignore b/.gitignore @@ -12,8 +12,8 @@ Refs #JUCE generated content Builds BinaryData.h -BinaryData.cpp -BinaryData2.cpp +BinaryData*.cpp +JuceLibraryCode #Windows Temp Cache Files [Tt]humbs.db diff --git a/.travis.yml b/.travis.yml @@ -94,6 +94,31 @@ script: # ./NewMixer --unit-tests # fi + # Pluginval + - | + if [[ $TRAVIS_OS_NAME == 'osx' ]]; then + echo "Validating plugin" + cd $TRAVIS_BUILD_DIR/ + curl -L "https://github.com/Tracktion/pluginval/releases/download/latest_release/pluginval_macOS.zip" -o pluginval.zip + unzip pluginval + pluginval.app/Contents/MacOS/pluginval --strictness-level 8 --validate-in-process --validate "$TRAVIS_BUILD_DIR/Plugin/Builds/MacOSX/build/Release/CHOWTapeModel.vst3" || exit 1 + pluginval.app/Contents/MacOS/pluginval --strictness-level 8 --validate-in-process --validate "$TRAVIS_BUILD_DIR/Plugin/Builds/MacOSX/build/Release/CHOWTapeModel.component" || exit 1 + fi + - | + if [[ $TRAVIS_OS_NAME == 'windows' ]]; then + echo "Validating plugin" + cd $TRAVIS_BUILD_DIR/ + powershell -Command "[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12; Invoke-WebRequest https://github.com/Tracktion/pluginval/releases/download/latest_release/pluginval_Windows.zip -OutFile pluginval.zip" + powershell -Command "Expand-Archive pluginval.zip -DestinationPath ." + ./pluginval.exe --strictness-level 8 --validate-in-process --validate "$TRAVIS_BUILD_DIR/Plugin/Builds/VisualStudio2017/x64/Debug/VST3/CHOWTapeModel.vst3" + fi + - | + if [[ $TRAVIS_OS_NAME == 'linux' ]]; then + curl -L "https://github.com/Tracktion/pluginval/releases/download/latest_release/pluginval_Linux.zip" -o pluginval.zip + unzip pluginval + ./pluginval --strictness-level 8 --validate-in-process --timeout-ms 300000 --validate "$TRAVIS_BUILD_DIR/Plugin/Builds/LinuxMakefile/build/CHOWTapeModel.so" || exit 1 + fi + - echo "SUCCESS" # after_success: # upload mac bins diff --git a/Plugin/CHOWTapeModel.jucer b/Plugin/CHOWTapeModel.jucer @@ -1,62 +1,62 @@ <?xml version="1.0" encoding="UTF-8"?> -<JUCERPROJECT id="uS0OjK" name="CHOWTapeModel" projectType="audioplug" jucerVersion="5.4.4" - pluginFormats="buildAU,buildStandalone,buildVST,buildVST3"> - <MAINGROUP id="oKBSK1" name="CHOWTapeModel"> - <GROUP id="{7659EE80-6477-48A0-C7B6-CD8900735F10}" name="Source"> - <GROUP id="{6EA40328-703D-5E93-4DC1-47040482016F}" name="GUI Components"> - <FILE id="C8pGDO" name="BiasControls.cpp" compile="1" resource="0" - file="Source/GUI Components/BiasControls.cpp"/> - <FILE id="PORdVN" name="BiasControls.h" compile="0" resource="0" file="Source/GUI Components/BiasControls.h"/> - <FILE id="jubA6Z" name="LossControls.cpp" compile="1" resource="0" +<JUCERPROJECT id="jDoXPz" name="CHOWTapeModel" projectType="audioplug" jucerVersion="5.4.4" + version="2.0.0" pluginFormats="buildAU,buildStandalone,buildVST,buildVST3" + cppLanguageStandard="17" companyName="chowdsp" companyEmail="[email protected]"> + <MAINGROUP id="pXbPvR" name="CHOWTapeModel"> + <GROUP id="{0178B10A-4A61-796A-5AB2-915D32AF6EEE}" name="Source"> + <GROUP id="{95079A80-7AE7-EAEE-9302-0BC2E901D700}" name="GUI Components"> + <FILE id="U6ldtW" name="HysteresisControls.cpp" compile="1" resource="0" + file="Source/GUI Components/HysteresisControls.cpp"/> + <FILE id="pgwXn0" name="HysteresisControls.h" compile="0" resource="0" + file="Source/GUI Components/HysteresisControls.h"/> + <FILE id="yCHTFN" name="LossControls.cpp" compile="1" resource="0" file="Source/GUI Components/LossControls.cpp"/> - <FILE id="KrcGEd" name="LossControls.h" compile="0" resource="0" file="Source/GUI Components/LossControls.h"/> - <FILE id="S8Xybc" name="MainControls.cpp" compile="1" resource="0" + <FILE id="qv9Pbi" name="LossControls.h" compile="0" resource="0" file="Source/GUI Components/LossControls.h"/> + <FILE id="mpMg4X" name="MainControls.cpp" compile="1" resource="0" file="Source/GUI Components/MainControls.cpp"/> - <FILE id="zXwghi" name="MainControls.h" compile="0" resource="0" file="Source/GUI Components/MainControls.h"/> - <FILE id="E0NC0D" name="TimingControls.cpp" compile="1" resource="0" + <FILE id="A4qTd6" name="MainControls.h" compile="0" resource="0" file="Source/GUI Components/MainControls.h"/> + <FILE id="mSX91i" name="TimingControls.cpp" compile="1" resource="0" file="Source/GUI Components/TimingControls.cpp"/> - <FILE id="vA4R0k" name="TimingControls.h" compile="0" resource="0" + <FILE id="Gd9QVZ" name="TimingControls.h" compile="0" resource="0" file="Source/GUI Components/TimingControls.h"/> </GROUP> - <GROUP id="{7867F016-80C7-7749-B604-ED042C040FC7}" name="GUI Extras"> - <FILE id="xd7cQs" name="ChowSlider.h" compile="0" resource="0" file="Source/GUI Extras/ChowSlider.h"/> - <FILE id="mQFlPP" name="MyLNF.h" compile="0" resource="0" file="Source/GUI Extras/MyLNF.h"/> + <GROUP id="{25D57C01-20AB-46DC-E9B3-80021429D612}" name="GUI Extras"> + <FILE id="ddmwEV" name="ChowSlider.h" compile="0" resource="0" file="Source/GUI Extras/ChowSlider.h"/> + <FILE id="gFXfuJ" name="MyLNF.h" compile="0" resource="0" file="Source/GUI Extras/MyLNF.h"/> </GROUP> - <GROUP id="{D2422983-A0E9-6A14-2092-2381CB1F3E7F}" name="Processors"> - <GROUP id="{6480D4F8-23CE-4932-1E51-CF3CF4FCC37F}" name="Timing Effects"> - <FILE id="sF9JlS" name="TimingEffect.cpp" compile="1" resource="0" - file="Source/Processors/Timing Effects/TimingEffect.cpp"/> - <FILE id="vDhRoW" name="TimingEffect.h" compile="0" resource="0" file="Source/Processors/Timing Effects/TimingEffect.h"/> - </GROUP> - <GROUP id="{99069F32-1399-FF98-7C4A-19F8E855C2AA}" name="Loss Effects"> - <FILE id="aEdcW7" name="LossEffects.cpp" compile="1" resource="0" file="Source/Processors/Loss Effects/LossEffects.cpp"/> - <FILE id="TFnfhT" name="LossEffects.h" compile="0" resource="0" file="Source/Processors/Loss Effects/LossEffects.h"/> - <FILE id="MSDyaq" name="LossEffectsFilter.cpp" compile="1" resource="0" - file="Source/Processors/Loss Effects/LossEffectsFilter.cpp"/> - <FILE id="C8T3dH" name="LossEffectsFilter.h" compile="0" resource="0" - file="Source/Processors/Loss Effects/LossEffectsFilter.h"/> - </GROUP> - <GROUP id="{E8FF20D6-35DC-19EF-2A0D-F8363CDBE1AE}" name="Hysteresis"> - <FILE id="sjUZ6l" name="HysteresisProcessing.cpp" compile="1" resource="0" + <GROUP id="{43BBFC88-4D0A-01B8-2635-3748470B94F4}" name="Processors"> + <GROUP id="{6052B1B0-83EF-DBFA-991C-FC0B47A949C9}" name="Hysteresis"> + <FILE id="Qe4tlV" name="HysteresisProcessing.cpp" compile="1" resource="0" file="Source/Processors/Hysteresis/HysteresisProcessing.cpp"/> - <FILE id="DrklCU" name="HysteresisProcessing.h" compile="0" resource="0" + <FILE id="OYS18C" name="HysteresisProcessing.h" compile="0" resource="0" file="Source/Processors/Hysteresis/HysteresisProcessing.h"/> - <FILE id="S8e9Mq" name="HysteresisProcessor.cpp" compile="1" resource="0" + <FILE id="Afas5v" name="HysteresisProcessor.cpp" compile="1" resource="0" file="Source/Processors/Hysteresis/HysteresisProcessor.cpp"/> - <FILE id="zCtaDA" name="HysteresisProcessor.h" compile="0" resource="0" + <FILE id="TRDp2E" name="HysteresisProcessor.h" compile="0" resource="0" file="Source/Processors/Hysteresis/HysteresisProcessor.h"/> </GROUP> - <FILE id="kdFIkd" name="GainProcessor.h" compile="0" resource="0" file="Source/Processors/GainProcessor.h"/> - <FILE id="eGTZ2w" name="ProcessorBase.h" compile="0" resource="0" file="Source/Processors/ProcessorBase.h"/> + <GROUP id="{37F4BCFA-28D3-CD4D-17AF-3C696E7EC8DA}" name="Loss_Effects"> + <FILE id="gJA2Gi" name="FIRFilter.h" compile="0" resource="0" file="Source/Processors/Loss_Effects/FIRFilter.h"/> + <FILE id="ZNErgZ" name="LossFilter.h" compile="0" resource="0" file="Source/Processors/Loss_Effects/LossFilter.h"/> + </GROUP> + <GROUP id="{0C000F30-53FD-3EFB-FAEA-6321B08AE56A}" name="Timing_Effects"> + <FILE id="kv0dir" name="DelayProcessor.cpp" compile="1" resource="0" + file="Source/Processors/Timing_Effects/DelayProcessor.cpp"/> + <FILE id="ZrrIc3" name="DelayProcessor.h" compile="0" resource="0" + file="Source/Processors/Timing_Effects/DelayProcessor.h"/> + <FILE id="cVSIAR" name="Flutter.cpp" compile="1" resource="0" file="Source/Processors/Timing_Effects/Flutter.cpp"/> + <FILE id="Wz3lz6" name="Flutter.h" compile="0" resource="0" file="Source/Processors/Timing_Effects/Flutter.h"/> + </GROUP> + <FILE id="zwLvQ9" name="GainProcessor.h" compile="0" resource="0" file="Source/Processors/GainProcessor.h"/> </GROUP> - <FILE id="y330sp" name="PluginProcessor.cpp" compile="1" resource="0" + <FILE id="zPJjtw" name="PluginProcessor.cpp" compile="1" resource="0" file="Source/PluginProcessor.cpp"/> - <FILE id="FvqAFZ" name="PluginProcessor.h" compile="0" resource="0" + <FILE id="JDgaoP" name="PluginProcessor.h" compile="0" resource="0" file="Source/PluginProcessor.h"/> - <FILE id="Z0ozGS" name="PluginEditor.cpp" compile="1" resource="0" + <FILE id="pmhzjP" name="PluginEditor.cpp" compile="1" resource="0" file="Source/PluginEditor.cpp"/> - <FILE id="IAaXYH" name="PluginEditor.h" compile="0" resource="0" file="Source/PluginEditor.h"/> + <FILE id="HqaJqZ" name="PluginEditor.h" compile="0" resource="0" file="Source/PluginEditor.h"/> </GROUP> </MAINGROUP> <EXPORTFORMATS> @@ -66,20 +66,20 @@ <CONFIGURATION isDebug="0" name="Release"/> </CONFIGURATIONS> <MODULEPATHS> - <MODULEPATH id="juce_audio_basics" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_audio_devices" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_audio_formats" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_audio_plugin_client" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_audio_processors" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_audio_utils" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_core" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_cryptography" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_data_structures" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_events" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_graphics" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_gui_basics" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_gui_extra" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_opengl" path="../../../ShiftyLoop/juce/modules"/> + <MODULEPATH id="juce_audio_basics" path="Juce/modules"/> + <MODULEPATH id="juce_audio_devices" path="Juce/modules"/> + <MODULEPATH id="juce_audio_formats" path="Juce/modules"/> + <MODULEPATH id="juce_audio_plugin_client" path="Juce/modules"/> + <MODULEPATH id="juce_audio_processors" path="Juce/modules"/> + <MODULEPATH id="juce_audio_utils" path="Juce/modules"/> + <MODULEPATH id="juce_core" path="Juce/modules"/> + <MODULEPATH id="juce_cryptography" path="Juce/modules"/> + <MODULEPATH id="juce_data_structures" path="Juce/modules"/> + <MODULEPATH id="juce_events" path="Juce/modules"/> + <MODULEPATH id="juce_graphics" path="Juce/modules"/> + <MODULEPATH id="juce_gui_basics" path="Juce/modules"/> + <MODULEPATH id="juce_gui_extra" path="Juce/modules"/> + <MODULEPATH id="juce_opengl" path="Juce/modules"/> <MODULEPATH id="juce_dsp" path="Juce/modules"/> </MODULEPATHS> </XCODE_MAC> @@ -87,23 +87,22 @@ <CONFIGURATIONS> <CONFIGURATION isDebug="1" name="Debug"/> <CONFIGURATION isDebug="0" name="Release"/> - <CONFIGURATION isDebug="0" name="Release (win32)" winArchitecture="Win32"/> </CONFIGURATIONS> <MODULEPATHS> - <MODULEPATH id="juce_audio_basics" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_audio_devices" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_audio_formats" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_audio_plugin_client" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_audio_processors" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_audio_utils" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_core" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_cryptography" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_data_structures" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_events" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_graphics" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_gui_basics" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_gui_extra" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_opengl" path="../../../ShiftyLoop/juce/modules"/> + <MODULEPATH id="juce_audio_basics" path="Juce/modules"/> + <MODULEPATH id="juce_audio_devices" path="Juce/modules"/> + <MODULEPATH id="juce_audio_formats" path="Juce/modules"/> + <MODULEPATH id="juce_audio_plugin_client" path="Juce/modules"/> + <MODULEPATH id="juce_audio_processors" path="Juce/modules"/> + <MODULEPATH id="juce_audio_utils" path="Juce/modules"/> + <MODULEPATH id="juce_core" path="Juce/modules"/> + <MODULEPATH id="juce_cryptography" path="Juce/modules"/> + <MODULEPATH id="juce_data_structures" path="Juce/modules"/> + <MODULEPATH id="juce_events" path="Juce/modules"/> + <MODULEPATH id="juce_graphics" path="Juce/modules"/> + <MODULEPATH id="juce_gui_basics" path="Juce/modules"/> + <MODULEPATH id="juce_gui_extra" path="Juce/modules"/> + <MODULEPATH id="juce_opengl" path="Juce/modules"/> <MODULEPATH id="juce_dsp" path="Juce/modules"/> </MODULEPATHS> </VS2017> @@ -113,41 +112,41 @@ <CONFIGURATION isDebug="0" name="Release"/> </CONFIGURATIONS> <MODULEPATHS> - <MODULEPATH id="juce_audio_basics" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_audio_devices" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_audio_formats" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_audio_plugin_client" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_audio_processors" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_audio_utils" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_core" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_cryptography" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_data_structures" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_events" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_graphics" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_gui_basics" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_gui_extra" path="../../../ShiftyLoop/juce/modules"/> - <MODULEPATH id="juce_opengl" path="../../../ShiftyLoop/juce/modules"/> + <MODULEPATH id="juce_audio_basics" path="Juce/modules"/> + <MODULEPATH id="juce_audio_devices" path="Juce/modules"/> + <MODULEPATH id="juce_audio_formats" path="Juce/modules"/> + <MODULEPATH id="juce_audio_plugin_client" path="Juce/modules"/> + <MODULEPATH id="juce_audio_processors" path="Juce/modules"/> + <MODULEPATH id="juce_audio_utils" path="Juce/modules"/> + <MODULEPATH id="juce_core" path="Juce/modules"/> + <MODULEPATH id="juce_cryptography" path="Juce/modules"/> + <MODULEPATH id="juce_data_structures" path="Juce/modules"/> + <MODULEPATH id="juce_events" path="Juce/modules"/> + <MODULEPATH id="juce_graphics" path="Juce/modules"/> + <MODULEPATH id="juce_gui_basics" path="Juce/modules"/> + <MODULEPATH id="juce_gui_extra" path="Juce/modules"/> + <MODULEPATH id="juce_opengl" path="Juce/modules"/> <MODULEPATH id="juce_dsp" path="Juce/modules"/> </MODULEPATHS> </LINUX_MAKE> </EXPORTFORMATS> <MODULES> - <MODULE id="juce_audio_basics" showAllCode="1" useLocalCopy="0" useGlobalPath="1"/> - <MODULE id="juce_audio_devices" showAllCode="1" useLocalCopy="0" useGlobalPath="1"/> - <MODULE id="juce_audio_formats" showAllCode="1" useLocalCopy="0" useGlobalPath="1"/> + <MODULE id="juce_audio_basics" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> + <MODULE id="juce_audio_devices" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> + <MODULE id="juce_audio_formats" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> <MODULE id="juce_audio_plugin_client" showAllCode="1" useLocalCopy="0" - useGlobalPath="1"/> - <MODULE id="juce_audio_processors" showAllCode="1" useLocalCopy="0" useGlobalPath="1"/> - <MODULE id="juce_audio_utils" showAllCode="1" useLocalCopy="0" useGlobalPath="1"/> - <MODULE id="juce_core" showAllCode="1" useLocalCopy="0" useGlobalPath="1"/> - <MODULE id="juce_cryptography" showAllCode="1" useLocalCopy="0" useGlobalPath="1"/> - <MODULE id="juce_data_structures" showAllCode="1" useLocalCopy="0" useGlobalPath="1"/> - <MODULE id="juce_dsp" showAllCode="1" useLocalCopy="0" useGlobalPath="1"/> - <MODULE id="juce_events" showAllCode="1" useLocalCopy="0" useGlobalPath="1"/> - <MODULE id="juce_graphics" showAllCode="1" useLocalCopy="0" useGlobalPath="1"/> - <MODULE id="juce_gui_basics" showAllCode="1" useLocalCopy="0" useGlobalPath="1"/> - <MODULE id="juce_gui_extra" showAllCode="1" useLocalCopy="0" useGlobalPath="1"/> - <MODULE id="juce_opengl" showAllCode="1" useLocalCopy="0" useGlobalPath="1"/> + useGlobalPath="0"/> + <MODULE id="juce_audio_processors" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> + <MODULE id="juce_audio_utils" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> + <MODULE id="juce_core" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> + <MODULE id="juce_cryptography" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> + <MODULE id="juce_data_structures" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> + <MODULE id="juce_dsp" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> + <MODULE id="juce_events" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> + <MODULE id="juce_graphics" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> + <MODULE id="juce_gui_basics" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> + <MODULE id="juce_gui_extra" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> + <MODULE id="juce_opengl" showAllCode="1" useLocalCopy="0" useGlobalPath="0"/> </MODULES> <LIVE_SETTINGS> <WINDOWS/> diff --git a/Plugin/JuceLibraryCode/AppConfig.h b/Plugin/JuceLibraryCode/AppConfig.h @@ -1,480 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - - There's a section below where you can add your own custom code safely, and the - Projucer will preserve the contents of that block, but the best way to change - any of these definitions is by using the Projucer's project settings. - - Any commented-out settings will assume their default values. - -*/ - -#pragma once - -//============================================================================== -// [BEGIN_USER_CODE_SECTION] - -// (You can add your own code in this section, and the Projucer will not overwrite it) - -// [END_USER_CODE_SECTION] - -/* - ============================================================================== - - In accordance with the terms of the JUCE 5 End-Use License Agreement, the - JUCE Code in SECTION A cannot be removed, changed or otherwise rendered - ineffective unless you have a JUCE Indie or Pro license, or are using JUCE - under the GPL v3 license. - - End User License Agreement: www.juce.com/juce-5-licence - - ============================================================================== -*/ - -// BEGIN SECTION A - -#ifndef JUCE_DISPLAY_SPLASH_SCREEN - #define JUCE_DISPLAY_SPLASH_SCREEN 0 -#endif - -#ifndef JUCE_REPORT_APP_USAGE - #define JUCE_REPORT_APP_USAGE 0 -#endif - -// END SECTION A - -#define JUCE_USE_DARK_SPLASH_SCREEN 1 - -//============================================================================== -#define JUCE_MODULE_AVAILABLE_juce_audio_basics 1 -#define JUCE_MODULE_AVAILABLE_juce_audio_devices 1 -#define JUCE_MODULE_AVAILABLE_juce_audio_formats 1 -#define JUCE_MODULE_AVAILABLE_juce_audio_plugin_client 1 -#define JUCE_MODULE_AVAILABLE_juce_audio_processors 1 -#define JUCE_MODULE_AVAILABLE_juce_audio_utils 1 -#define JUCE_MODULE_AVAILABLE_juce_core 1 -#define JUCE_MODULE_AVAILABLE_juce_cryptography 1 -#define JUCE_MODULE_AVAILABLE_juce_data_structures 1 -#define JUCE_MODULE_AVAILABLE_juce_dsp 1 -#define JUCE_MODULE_AVAILABLE_juce_events 1 -#define JUCE_MODULE_AVAILABLE_juce_graphics 1 -#define JUCE_MODULE_AVAILABLE_juce_gui_basics 1 -#define JUCE_MODULE_AVAILABLE_juce_gui_extra 1 -#define JUCE_MODULE_AVAILABLE_juce_opengl 1 - -#define JUCE_GLOBAL_MODULE_SETTINGS_INCLUDED 1 - -//============================================================================== -// juce_audio_devices flags: - -#ifndef JUCE_USE_WINRT_MIDI - //#define JUCE_USE_WINRT_MIDI 0 -#endif - -#ifndef JUCE_ASIO - //#define JUCE_ASIO 0 -#endif - -#ifndef JUCE_WASAPI - //#define JUCE_WASAPI 1 -#endif - -#ifndef JUCE_WASAPI_EXCLUSIVE - //#define JUCE_WASAPI_EXCLUSIVE 0 -#endif - -#ifndef JUCE_DIRECTSOUND - //#define JUCE_DIRECTSOUND 1 -#endif - -#ifndef JUCE_ALSA - //#define JUCE_ALSA 1 -#endif - -#ifndef JUCE_JACK - //#define JUCE_JACK 0 -#endif - -#ifndef JUCE_BELA - //#define JUCE_BELA 0 -#endif - -#ifndef JUCE_USE_ANDROID_OBOE - //#define JUCE_USE_ANDROID_OBOE 0 -#endif - -#ifndef JUCE_USE_ANDROID_OPENSLES - //#define JUCE_USE_ANDROID_OPENSLES 0 -#endif - -#ifndef JUCE_DISABLE_AUDIO_MIXING_WITH_OTHER_APPS - //#define JUCE_DISABLE_AUDIO_MIXING_WITH_OTHER_APPS 0 -#endif - -//============================================================================== -// juce_audio_formats flags: - -#ifndef JUCE_USE_FLAC - //#define JUCE_USE_FLAC 1 -#endif - -#ifndef JUCE_USE_OGGVORBIS - //#define JUCE_USE_OGGVORBIS 1 -#endif - -#ifndef JUCE_USE_MP3AUDIOFORMAT - //#define JUCE_USE_MP3AUDIOFORMAT 0 -#endif - -#ifndef JUCE_USE_LAME_AUDIO_FORMAT - //#define JUCE_USE_LAME_AUDIO_FORMAT 0 -#endif - -#ifndef JUCE_USE_WINDOWS_MEDIA_FORMAT - //#define JUCE_USE_WINDOWS_MEDIA_FORMAT 1 -#endif - -//============================================================================== -// juce_audio_plugin_client flags: - -#ifndef JUCE_VST3_CAN_REPLACE_VST2 - #define JUCE_VST3_CAN_REPLACE_VST2 0 -#endif - -#ifndef JUCE_FORCE_USE_LEGACY_PARAM_IDS - //#define JUCE_FORCE_USE_LEGACY_PARAM_IDS 0 -#endif - -#ifndef JUCE_FORCE_LEGACY_PARAMETER_AUTOMATION_TYPE - //#define JUCE_FORCE_LEGACY_PARAMETER_AUTOMATION_TYPE 0 -#endif - -#ifndef JUCE_USE_STUDIO_ONE_COMPATIBLE_PARAMETERS - //#define JUCE_USE_STUDIO_ONE_COMPATIBLE_PARAMETERS 1 -#endif - -#ifndef JUCE_STANDALONE_FILTER_WINDOW_USE_KIOSK_MODE - //#define JUCE_STANDALONE_FILTER_WINDOW_USE_KIOSK_MODE 0 -#endif - -//============================================================================== -// juce_audio_processors flags: - -#ifndef JUCE_PLUGINHOST_VST - //#define JUCE_PLUGINHOST_VST 0 -#endif - -#ifndef JUCE_PLUGINHOST_VST3 - //#define JUCE_PLUGINHOST_VST3 0 -#endif - -#ifndef JUCE_PLUGINHOST_AU - //#define JUCE_PLUGINHOST_AU 0 -#endif - -#ifndef JUCE_PLUGINHOST_LADSPA - //#define JUCE_PLUGINHOST_LADSPA 0 -#endif - -//============================================================================== -// juce_audio_utils flags: - -#ifndef JUCE_USE_CDREADER - //#define JUCE_USE_CDREADER 0 -#endif - -#ifndef JUCE_USE_CDBURNER - //#define JUCE_USE_CDBURNER 0 -#endif - -//============================================================================== -// juce_core flags: - -#ifndef JUCE_FORCE_DEBUG - //#define JUCE_FORCE_DEBUG 0 -#endif - -#ifndef JUCE_LOG_ASSERTIONS - //#define JUCE_LOG_ASSERTIONS 0 -#endif - -#ifndef JUCE_CHECK_MEMORY_LEAKS - //#define JUCE_CHECK_MEMORY_LEAKS 1 -#endif - -#ifndef JUCE_DONT_AUTOLINK_TO_WIN32_LIBRARIES - //#define JUCE_DONT_AUTOLINK_TO_WIN32_LIBRARIES 0 -#endif - -#ifndef JUCE_INCLUDE_ZLIB_CODE - //#define JUCE_INCLUDE_ZLIB_CODE 1 -#endif - -#ifndef JUCE_USE_CURL - //#define JUCE_USE_CURL 1 -#endif - -#ifndef JUCE_LOAD_CURL_SYMBOLS_LAZILY - //#define JUCE_LOAD_CURL_SYMBOLS_LAZILY 0 -#endif - -#ifndef JUCE_CATCH_UNHANDLED_EXCEPTIONS - //#define JUCE_CATCH_UNHANDLED_EXCEPTIONS 0 -#endif - -#ifndef JUCE_ALLOW_STATIC_NULL_VARIABLES - //#define JUCE_ALLOW_STATIC_NULL_VARIABLES 0 -#endif - -#ifndef JUCE_STRICT_REFCOUNTEDPOINTER - #define JUCE_STRICT_REFCOUNTEDPOINTER 1 -#endif - -//============================================================================== -// juce_dsp flags: - -#ifndef JUCE_ASSERTION_FIRFILTER - //#define JUCE_ASSERTION_FIRFILTER 1 -#endif - -#ifndef JUCE_DSP_USE_INTEL_MKL - //#define JUCE_DSP_USE_INTEL_MKL 0 -#endif - -#ifndef JUCE_DSP_USE_SHARED_FFTW - //#define JUCE_DSP_USE_SHARED_FFTW 0 -#endif - -#ifndef JUCE_DSP_USE_STATIC_FFTW - //#define JUCE_DSP_USE_STATIC_FFTW 0 -#endif - -#ifndef JUCE_DSP_ENABLE_SNAP_TO_ZERO - //#define JUCE_DSP_ENABLE_SNAP_TO_ZERO 1 -#endif - -//============================================================================== -// juce_events flags: - -#ifndef JUCE_EXECUTE_APP_SUSPEND_ON_IOS_BACKGROUND_TASK - //#define JUCE_EXECUTE_APP_SUSPEND_ON_IOS_BACKGROUND_TASK 0 -#endif - -//============================================================================== -// juce_graphics flags: - -#ifndef JUCE_USE_COREIMAGE_LOADER - //#define JUCE_USE_COREIMAGE_LOADER 1 -#endif - -#ifndef JUCE_USE_DIRECTWRITE - //#define JUCE_USE_DIRECTWRITE 1 -#endif - -#ifndef JUCE_DISABLE_COREGRAPHICS_FONT_SMOOTHING - //#define JUCE_DISABLE_COREGRAPHICS_FONT_SMOOTHING 0 -#endif - -//============================================================================== -// juce_gui_basics flags: - -#ifndef JUCE_ENABLE_REPAINT_DEBUGGING - //#define JUCE_ENABLE_REPAINT_DEBUGGING 0 -#endif - -#ifndef JUCE_USE_XRANDR - //#define JUCE_USE_XRANDR 1 -#endif - -#ifndef JUCE_USE_XINERAMA - //#define JUCE_USE_XINERAMA 1 -#endif - -#ifndef JUCE_USE_XSHM - //#define JUCE_USE_XSHM 1 -#endif - -#ifndef JUCE_USE_XRENDER - //#define JUCE_USE_XRENDER 0 -#endif - -#ifndef JUCE_USE_XCURSOR - //#define JUCE_USE_XCURSOR 1 -#endif - -#ifndef JUCE_WIN_PER_MONITOR_DPI_AWARE - //#define JUCE_WIN_PER_MONITOR_DPI_AWARE 1 -#endif - -//============================================================================== -// juce_gui_extra flags: - -#ifndef JUCE_WEB_BROWSER - //#define JUCE_WEB_BROWSER 1 -#endif - -#ifndef JUCE_ENABLE_LIVE_CONSTANT_EDITOR - //#define JUCE_ENABLE_LIVE_CONSTANT_EDITOR 0 -#endif - -//============================================================================== -// Audio plugin settings.. - -#ifndef JucePlugin_Build_VST - #define JucePlugin_Build_VST 1 -#endif -#ifndef JucePlugin_Build_VST3 - #define JucePlugin_Build_VST3 1 -#endif -#ifndef JucePlugin_Build_AU - #define JucePlugin_Build_AU 1 -#endif -#ifndef JucePlugin_Build_AUv3 - #define JucePlugin_Build_AUv3 0 -#endif -#ifndef JucePlugin_Build_RTAS - #define JucePlugin_Build_RTAS 0 -#endif -#ifndef JucePlugin_Build_AAX - #define JucePlugin_Build_AAX 0 -#endif -#ifndef JucePlugin_Build_Standalone - #define JucePlugin_Build_Standalone 1 -#endif -#ifndef JucePlugin_Build_Unity - #define JucePlugin_Build_Unity 0 -#endif -#ifndef JucePlugin_Enable_IAA - #define JucePlugin_Enable_IAA 0 -#endif -#ifndef JucePlugin_Name - #define JucePlugin_Name "CHOWTapeModel" -#endif -#ifndef JucePlugin_Desc - #define JucePlugin_Desc "CHOWTapeModel" -#endif -#ifndef JucePlugin_Manufacturer - #define JucePlugin_Manufacturer "yourcompany" -#endif -#ifndef JucePlugin_ManufacturerWebsite - #define JucePlugin_ManufacturerWebsite "" -#endif -#ifndef JucePlugin_ManufacturerEmail - #define JucePlugin_ManufacturerEmail "" -#endif -#ifndef JucePlugin_ManufacturerCode - #define JucePlugin_ManufacturerCode 0x4d616e75 // 'Manu' -#endif -#ifndef JucePlugin_PluginCode - #define JucePlugin_PluginCode 0x5573306f // 'Us0o' -#endif -#ifndef JucePlugin_IsSynth - #define JucePlugin_IsSynth 0 -#endif -#ifndef JucePlugin_WantsMidiInput - #define JucePlugin_WantsMidiInput 0 -#endif -#ifndef JucePlugin_ProducesMidiOutput - #define JucePlugin_ProducesMidiOutput 0 -#endif -#ifndef JucePlugin_IsMidiEffect - #define JucePlugin_IsMidiEffect 0 -#endif -#ifndef JucePlugin_EditorRequiresKeyboardFocus - #define JucePlugin_EditorRequiresKeyboardFocus 0 -#endif -#ifndef JucePlugin_Version - #define JucePlugin_Version 1.0.0 -#endif -#ifndef JucePlugin_VersionCode - #define JucePlugin_VersionCode 0x10000 -#endif -#ifndef JucePlugin_VersionString - #define JucePlugin_VersionString "1.0.0" -#endif -#ifndef JucePlugin_VSTUniqueID - #define JucePlugin_VSTUniqueID JucePlugin_PluginCode -#endif -#ifndef JucePlugin_VSTCategory - #define JucePlugin_VSTCategory kPlugCategEffect -#endif -#ifndef JucePlugin_Vst3Category - #define JucePlugin_Vst3Category "Fx" -#endif -#ifndef JucePlugin_AUMainType - #define JucePlugin_AUMainType 'aufx' -#endif -#ifndef JucePlugin_AUSubType - #define JucePlugin_AUSubType JucePlugin_PluginCode -#endif -#ifndef JucePlugin_AUExportPrefix - #define JucePlugin_AUExportPrefix CHOWTapeModelAU -#endif -#ifndef JucePlugin_AUExportPrefixQuoted - #define JucePlugin_AUExportPrefixQuoted "CHOWTapeModelAU" -#endif -#ifndef JucePlugin_AUManufacturerCode - #define JucePlugin_AUManufacturerCode JucePlugin_ManufacturerCode -#endif -#ifndef JucePlugin_CFBundleIdentifier - #define JucePlugin_CFBundleIdentifier com.yourcompany.CHOWTapeModel -#endif -#ifndef JucePlugin_RTASCategory - #define JucePlugin_RTASCategory 0 -#endif -#ifndef JucePlugin_RTASManufacturerCode - #define JucePlugin_RTASManufacturerCode JucePlugin_ManufacturerCode -#endif -#ifndef JucePlugin_RTASProductId - #define JucePlugin_RTASProductId JucePlugin_PluginCode -#endif -#ifndef JucePlugin_RTASDisableBypass - #define JucePlugin_RTASDisableBypass 0 -#endif -#ifndef JucePlugin_RTASDisableMultiMono - #define JucePlugin_RTASDisableMultiMono 0 -#endif -#ifndef JucePlugin_AAXIdentifier - #define JucePlugin_AAXIdentifier com.yourcompany.CHOWTapeModel -#endif -#ifndef JucePlugin_AAXManufacturerCode - #define JucePlugin_AAXManufacturerCode JucePlugin_ManufacturerCode -#endif -#ifndef JucePlugin_AAXProductId - #define JucePlugin_AAXProductId JucePlugin_PluginCode -#endif -#ifndef JucePlugin_AAXCategory - #define JucePlugin_AAXCategory 0 -#endif -#ifndef JucePlugin_AAXDisableBypass - #define JucePlugin_AAXDisableBypass 0 -#endif -#ifndef JucePlugin_AAXDisableMultiMono - #define JucePlugin_AAXDisableMultiMono 0 -#endif -#ifndef JucePlugin_IAAType - #define JucePlugin_IAAType 0x61757278 // 'aurx' -#endif -#ifndef JucePlugin_IAASubType - #define JucePlugin_IAASubType JucePlugin_PluginCode -#endif -#ifndef JucePlugin_IAAName - #define JucePlugin_IAAName "yourcompany: CHOWTapeModel" -#endif -#ifndef JucePlugin_VSTNumMidiInputs - #define JucePlugin_VSTNumMidiInputs 16 -#endif -#ifndef JucePlugin_VSTNumMidiOutputs - #define JucePlugin_VSTNumMidiOutputs 16 -#endif - -//============================================================================== -#ifndef JUCE_STANDALONE_APPLICATION - #if defined(JucePlugin_Name) && defined(JucePlugin_Build_Standalone) - #define JUCE_STANDALONE_APPLICATION JucePlugin_Build_Standalone - #else - #define JUCE_STANDALONE_APPLICATION 0 - #endif -#endif diff --git a/Plugin/JuceLibraryCode/JuceHeader.h b/Plugin/JuceLibraryCode/JuceHeader.h @@ -1,48 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - - This is the header file that your files should include in order to get all the - JUCE library headers. You should avoid including the JUCE headers directly in - your own source files, because that wouldn't pick up the correct configuration - options for your app. - -*/ - -#pragma once - -#include "AppConfig.h" - -#include <juce_audio_basics/juce_audio_basics.h> -#include <juce_audio_devices/juce_audio_devices.h> -#include <juce_audio_formats/juce_audio_formats.h> -#include <juce_audio_plugin_client/juce_audio_plugin_client.h> -#include <juce_audio_processors/juce_audio_processors.h> -#include <juce_audio_utils/juce_audio_utils.h> -#include <juce_core/juce_core.h> -#include <juce_cryptography/juce_cryptography.h> -#include <juce_data_structures/juce_data_structures.h> -#include <juce_dsp/juce_dsp.h> -#include <juce_events/juce_events.h> -#include <juce_graphics/juce_graphics.h> -#include <juce_gui_basics/juce_gui_basics.h> -#include <juce_gui_extra/juce_gui_extra.h> -#include <juce_opengl/juce_opengl.h> - - -#if ! DONT_SET_USING_JUCE_NAMESPACE - // If your code uses a lot of JUCE classes, then this will obviously save you - // a lot of typing, but can be disabled by setting DONT_SET_USING_JUCE_NAMESPACE. - using namespace juce; -#endif - -#if ! JUCE_DONT_DECLARE_PROJECTINFO -namespace ProjectInfo -{ - const char* const projectName = "CHOWTapeModel"; - const char* const companyName = ""; - const char* const versionString = "1.0.0"; - const int versionNumber = 0x10000; -} -#endif diff --git a/Plugin/JuceLibraryCode/ReadMe.txt b/Plugin/JuceLibraryCode/ReadMe.txt @@ -1,12 +0,0 @@ - - Important Note!! - ================ - -The purpose of this folder is to contain files that are auto-generated by the Projucer, -and ALL files in this folder will be mercilessly DELETED and completely re-written whenever -the Projucer saves your project. - -Therefore, it's a bad idea to make any manual changes to the files in here, or to -put any of your own files in here if you don't want to lose them. (Of course you may choose -to add the folder's contents to your version-control system so that you can re-merge your own -modifications after the Projucer has saved its changes). diff --git a/Plugin/JuceLibraryCode/include_juce_audio_basics.cpp b/Plugin/JuceLibraryCode/include_juce_audio_basics.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_basics/juce_audio_basics.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_basics.mm b/Plugin/JuceLibraryCode/include_juce_audio_basics.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_basics/juce_audio_basics.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_devices.cpp b/Plugin/JuceLibraryCode/include_juce_audio_devices.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_devices/juce_audio_devices.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_devices.mm b/Plugin/JuceLibraryCode/include_juce_audio_devices.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_devices/juce_audio_devices.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_formats.cpp b/Plugin/JuceLibraryCode/include_juce_audio_formats.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_formats/juce_audio_formats.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_formats.mm b/Plugin/JuceLibraryCode/include_juce_audio_formats.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_formats/juce_audio_formats.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_AAX.cpp b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_AAX.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_AAX.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_AAX.mm b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_AAX.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_AAX.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_AU.r b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_AU.r @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client_AU.r> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_AU_1.mm b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_AU_1.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_AU_1.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_AU_2.mm b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_AU_2.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_AU_2.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_AUv3.mm b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_AUv3.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_AUv3.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_RTAS.r b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_RTAS.r @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client_RTAS.r> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_RTAS_1.cpp b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_RTAS_1.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_RTAS_1.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_RTAS_2.cpp b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_RTAS_2.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_RTAS_2.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_RTAS_3.cpp b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_RTAS_3.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_RTAS_3.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_RTAS_4.cpp b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_RTAS_4.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_RTAS_4.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_RTAS_utils.cpp b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_RTAS_utils.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_RTAS_utils.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_RTAS_utils.mm b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_RTAS_utils.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_RTAS_utils.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_Standalone.cpp b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_Standalone.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_Standalone.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_Unity.cpp b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_Unity.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_Unity.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_VST2.cpp b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_VST2.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_VST2.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_VST3.cpp b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_VST3.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_VST3.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_VST_utils.mm b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_VST_utils.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_VST_utils.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_utils.cpp b/Plugin/JuceLibraryCode/include_juce_audio_plugin_client_utils.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_plugin_client/juce_audio_plugin_client_utils.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_processors.cpp b/Plugin/JuceLibraryCode/include_juce_audio_processors.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_processors/juce_audio_processors.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_processors.mm b/Plugin/JuceLibraryCode/include_juce_audio_processors.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_processors/juce_audio_processors.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_utils.cpp b/Plugin/JuceLibraryCode/include_juce_audio_utils.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_utils/juce_audio_utils.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_audio_utils.mm b/Plugin/JuceLibraryCode/include_juce_audio_utils.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_audio_utils/juce_audio_utils.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_core.cpp b/Plugin/JuceLibraryCode/include_juce_core.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_core/juce_core.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_core.mm b/Plugin/JuceLibraryCode/include_juce_core.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_core/juce_core.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_cryptography.cpp b/Plugin/JuceLibraryCode/include_juce_cryptography.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_cryptography/juce_cryptography.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_cryptography.mm b/Plugin/JuceLibraryCode/include_juce_cryptography.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_cryptography/juce_cryptography.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_data_structures.cpp b/Plugin/JuceLibraryCode/include_juce_data_structures.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_data_structures/juce_data_structures.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_data_structures.mm b/Plugin/JuceLibraryCode/include_juce_data_structures.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_data_structures/juce_data_structures.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_dsp.cpp b/Plugin/JuceLibraryCode/include_juce_dsp.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_dsp/juce_dsp.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_dsp.mm b/Plugin/JuceLibraryCode/include_juce_dsp.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_dsp/juce_dsp.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_events.cpp b/Plugin/JuceLibraryCode/include_juce_events.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_events/juce_events.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_events.mm b/Plugin/JuceLibraryCode/include_juce_events.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_events/juce_events.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_graphics.cpp b/Plugin/JuceLibraryCode/include_juce_graphics.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_graphics/juce_graphics.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_graphics.mm b/Plugin/JuceLibraryCode/include_juce_graphics.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_graphics/juce_graphics.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_gui_basics.cpp b/Plugin/JuceLibraryCode/include_juce_gui_basics.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_gui_basics/juce_gui_basics.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_gui_basics.mm b/Plugin/JuceLibraryCode/include_juce_gui_basics.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_gui_basics/juce_gui_basics.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_gui_extra.cpp b/Plugin/JuceLibraryCode/include_juce_gui_extra.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_gui_extra/juce_gui_extra.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_gui_extra.mm b/Plugin/JuceLibraryCode/include_juce_gui_extra.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_gui_extra/juce_gui_extra.mm> diff --git a/Plugin/JuceLibraryCode/include_juce_opengl.cpp b/Plugin/JuceLibraryCode/include_juce_opengl.cpp @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_opengl/juce_opengl.cpp> diff --git a/Plugin/JuceLibraryCode/include_juce_opengl.mm b/Plugin/JuceLibraryCode/include_juce_opengl.mm @@ -1,9 +0,0 @@ -/* - - IMPORTANT! This file is auto-generated each time you save your - project - if you alter its contents, your changes may be overwritten! - -*/ - -#include "AppConfig.h" -#include <juce_opengl/juce_opengl.mm> diff --git a/Plugin/Source/GUI Components/BiasControls.cpp b/Plugin/Source/GUI Components/BiasControls.cpp @@ -1,52 +0,0 @@ -#include "BiasControls.h" -#include "../PluginEditor.h" - -BiasControls::BiasControls (ChowtapeModelAudioProcessor& proc) : - processor (proc) -{ - ChowtapeModelAudioProcessorEditor::createSlider (biasFreqSlide, processor.biasFreq, myLNF, this, String ("kHz"), 1.0f); - biasFreqSlide.setEnabled (false); - //biasFreqSlide.setSkewFactorFromMidPoint (55.0); - - ChowtapeModelAudioProcessorEditor::createSlider (biasGainSlide, processor.biasGain, myLNF, this); - //biasGainSlide.setSkewFactorFromMidPoint (14.0); - - ChowtapeModelAudioProcessorEditor::createLabel (biasFreqLabel, processor.biasFreq, this); - ChowtapeModelAudioProcessorEditor::createLabel (biasGainLabel, processor.biasGain, this); -} - -void BiasControls::paint (Graphics& g) -{ - g.setColour (Colours::antiquewhite); - g.setFont (Font ((float) nameHeight).boldened()); - - g.drawFittedText ("Biasing Parameters:", Rectangle<int> (xOffset, yOffset, width, labelHeight), - Justification::centredLeft, 1); -} - -void BiasControls::resized() -{ - biasFreqLabel.setBounds (0, 2 * yOffset + labelY, sliderWidth, labelHeight); - biasFreqSlide.setBounds (0, 2 * yOffset + sliderY, sliderWidth, sliderWidth); - - biasGainLabel.setBounds (biasFreqSlide.getRight(), 2 * yOffset + labelY, sliderWidth, labelHeight); - biasGainSlide.setBounds (biasFreqSlide.getRight(), 2 * yOffset + sliderY, sliderWidth, sliderWidth); -} - -void BiasControls::sliderValueChanged (Slider* slider) -{ - if (AudioParameterFloat* param = ChowtapeModelAudioProcessorEditor::getParamForSlider (slider, processor)) - *param = (float) slider->getValue(); -} - -void BiasControls::sliderDragStarted(Slider* slider) -{ - if (AudioParameterFloat* param = ChowtapeModelAudioProcessorEditor::getParamForSlider (slider, processor)) - param->beginChangeGesture(); -} - -void BiasControls::sliderDragEnded(Slider* slider) -{ - if (AudioParameterFloat* param = ChowtapeModelAudioProcessorEditor::getParamForSlider (slider, processor)) - param->endChangeGesture(); -} diff --git a/Plugin/Source/GUI Components/BiasControls.h b/Plugin/Source/GUI Components/BiasControls.h @@ -1,35 +0,0 @@ -#ifndef BIASCONTROLS_H_INCLUDED -#define BIASCONTROLS_H_INCLUDED - -#include "../PluginProcessor.h" -#include "../GUI Extras/ChowSlider.h" -#include "../GUI Extras/MyLNF.h" - -class BiasControls : public Component, - public Slider::Listener -{ -public: - BiasControls (ChowtapeModelAudioProcessor& proc); - - void paint (Graphics&) override; - void resized() override; - -private: - void sliderValueChanged (Slider* slider) override; - void sliderDragStarted (Slider* slider) override; - void sliderDragEnded (Slider* slider) override; - - MyLNF myLNF; - - ChowtapeModelAudioProcessor& processor; - - ChowSlider biasFreqSlide; - ChowSlider biasGainSlide; - - Label biasFreqLabel; - Label biasGainLabel; - - JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (BiasControls) -}; - -#endif //BIASCONTROLS_H_INCLUDED diff --git a/Plugin/Source/GUI Components/HysteresisControls.cpp b/Plugin/Source/GUI Components/HysteresisControls.cpp @@ -0,0 +1,37 @@ +#include "HysteresisControls.h" +#include "../PluginEditor.h" + +HysteresisControls::HysteresisControls (ChowtapeModelAudioProcessor& proc) : + processor (proc) +{ + auto createSliderAnbdLabel = [=, &proc] (FullSlider& slider, String id) + { + ChowtapeModelAudioProcessorEditor::createSlider (slider.slider, proc.getVTS(), id, slider.attach, *this, myLNF); + ChowtapeModelAudioProcessorEditor::createLabel (slider.label, slider.slider.getName(), this); + }; + + createSliderAnbdLabel (driveSlide, "drive"); + createSliderAnbdLabel (satSlide, "sat"); + createSliderAnbdLabel (widthSlide, "width"); +} + +void HysteresisControls::paint (Graphics& g) +{ + g.setColour (Colours::antiquewhite); + g.setFont (Font ((float) nameHeight).boldened()); + + g.drawFittedText ("Hysteresis Parameters:", Rectangle<int> (xOffset, yOffset, width, labelHeight), + Justification::centredLeft, 1); +} + +void HysteresisControls::resized() +{ + driveSlide.label.setBounds (0, 2 * yOffset + labelY, sliderWidth, labelHeight); + driveSlide.slider.setBounds (0, 2 * yOffset + sliderY, sliderWidth, sliderWidth); + + satSlide.label.setBounds (driveSlide.slider.getRight(), 2 * yOffset + labelY, sliderWidth, labelHeight); + satSlide.slider.setBounds (driveSlide.slider.getRight(), 2 * yOffset + sliderY, sliderWidth, sliderWidth); + + widthSlide.label.setBounds (satSlide.slider.getRight(), 2 * yOffset + labelY, sliderWidth, labelHeight); + widthSlide.slider.setBounds (satSlide.slider.getRight(), 2 * yOffset + sliderY, sliderWidth, sliderWidth); +} diff --git a/Plugin/Source/GUI Components/HysteresisControls.h b/Plugin/Source/GUI Components/HysteresisControls.h @@ -0,0 +1,37 @@ +#ifndef BIASCONTROLS_H_INCLUDED +#define BIASCONTROLS_H_INCLUDED + +#include "../PluginProcessor.h" +#include "../GUI Extras/ChowSlider.h" +#include "../GUI Extras/MyLNF.h" + +using SliderAttachment = AudioProcessorValueTreeState::SliderAttachment; + +struct FullSlider +{ + Slider slider; + Label label; + std::unique_ptr<SliderAttachment> attach; +}; + +class HysteresisControls : public Component +{ +public: + HysteresisControls (ChowtapeModelAudioProcessor& proc); + + void paint (Graphics&) override; + void resized() override; + +private: + MyLNF myLNF; + + ChowtapeModelAudioProcessor& processor; + + FullSlider driveSlide; + FullSlider satSlide; + FullSlider widthSlide; + + JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (HysteresisControls) +}; + +#endif //BIASCONTROLS_H_INCLUDED diff --git a/Plugin/Source/GUI Components/LossControls.cpp b/Plugin/Source/GUI Components/LossControls.cpp @@ -4,13 +4,18 @@ LossControls::LossControls (ChowtapeModelAudioProcessor& proc) : processor (proc) { - ChowtapeModelAudioProcessorEditor::createSlider (spacingSlider, processor.tapeSpacing, myLNF, this, String ("um")); - ChowtapeModelAudioProcessorEditor::createSlider (thicknessSlider, processor.tapeThickness, myLNF, this, String ("um")); - ChowtapeModelAudioProcessorEditor::createSlider (gapSlider, processor.gapWidth, myLNF, this, String ("um")); - - ChowtapeModelAudioProcessorEditor::createLabel (spacingLabel, processor.tapeSpacing, this); - ChowtapeModelAudioProcessorEditor::createLabel (thicknessLabel, processor.tapeThickness, this); - ChowtapeModelAudioProcessorEditor::createLabel (gapLabel, processor.gapWidth, this); + std::function<String (double)> mmFunc = [] (double value) { return String (value * 1000.0f, 2, true) + " mm"; }; + + auto createSliderAndLabel = [=, &proc] (FullSlider& slider, String id, String suffix= {}, std::function<String (double)> textFromValue = {}) + { + ChowtapeModelAudioProcessorEditor::createSlider (slider.slider, proc.getVTS(), id, slider.attach, *this, myLNF, suffix, {}, textFromValue); + ChowtapeModelAudioProcessorEditor::createLabel (slider.label, slider.slider.getName(), this); + }; + + createSliderAndLabel (speedSlider, "speed", " ips"); + createSliderAndLabel (spacingSlider, "spacing", "", mmFunc); + createSliderAndLabel (thicknessSlider, "thick", "", mmFunc); + createSliderAndLabel (gapSlider, "gap", "", mmFunc); } void LossControls::paint (Graphics& g) @@ -18,36 +23,23 @@ void LossControls::paint (Graphics& g) g.setColour (Colours::antiquewhite); g.setFont (Font ((float) nameHeight).boldened()); - g.drawFittedText ("Loss Parameters:", Rectangle<int> (xOffset, yOffset, width, labelHeight), + g.drawFittedText ("Playhead Parameters:", Rectangle<int> (xOffset, yOffset, width, labelHeight), Justification::centredLeft, 1); } void LossControls::resized() { - spacingLabel.setBounds (0, 2 * yOffset + labelY, sliderWidth, labelHeight); - spacingSlider.setBounds (0, 2 * yOffset + sliderY, sliderWidth, sliderWidth); - - thicknessLabel.setBounds (spacingSlider.getRight(), 2 * yOffset + labelY, sliderWidth, labelHeight); - thicknessSlider.setBounds (spacingSlider.getRight(), 2 * yOffset + sliderY, sliderWidth, sliderWidth); + const int smallSliderWidth = 90; - gapLabel.setBounds (thicknessSlider.getRight(), 2 * yOffset + labelY, sliderWidth, labelHeight); - gapSlider.setBounds (thicknessSlider.getRight(), 2 * yOffset + sliderY, sliderWidth, sliderWidth); -} + spacingSlider.label.setBounds (0, 2 * yOffset + labelY, smallSliderWidth, labelHeight); + spacingSlider.slider.setBounds (0, 2 * yOffset + sliderY, smallSliderWidth, sliderWidth); -void LossControls::sliderValueChanged (Slider* slider) -{ - if (AudioParameterFloat* param = ChowtapeModelAudioProcessorEditor::getParamForSlider (slider, processor)) - *param = (float) slider->getValue(); -} + thicknessSlider.label.setBounds (spacingSlider.slider.getRight(), 2 * yOffset + labelY, smallSliderWidth, labelHeight); + thicknessSlider.slider.setBounds (spacingSlider.slider.getRight(), 2 * yOffset + sliderY, smallSliderWidth, sliderWidth); -void LossControls::sliderDragStarted(Slider* slider) -{ - if (AudioParameterFloat* param = ChowtapeModelAudioProcessorEditor::getParamForSlider (slider, processor)) - param->beginChangeGesture(); -} + gapSlider.label.setBounds (thicknessSlider.slider.getRight(), 2 * yOffset + labelY, smallSliderWidth, labelHeight); + gapSlider.slider.setBounds (thicknessSlider.slider.getRight(), 2 * yOffset + sliderY, smallSliderWidth, sliderWidth); -void LossControls::sliderDragEnded(Slider* slider) -{ - if (AudioParameterFloat* param = ChowtapeModelAudioProcessorEditor::getParamForSlider (slider, processor)) - param->endChangeGesture(); + speedSlider.label.setBounds (gapSlider.slider.getRight(), 2 * yOffset + labelY, smallSliderWidth, labelHeight); + speedSlider.slider.setBounds (gapSlider.slider.getRight(), 2 * yOffset + sliderY, smallSliderWidth, sliderWidth); } diff --git a/Plugin/Source/GUI Components/LossControls.h b/Plugin/Source/GUI Components/LossControls.h @@ -4,9 +4,11 @@ #include "../PluginProcessor.h" #include "../GUI Extras/ChowSlider.h" #include "../GUI Extras/MyLNF.h" +#include "HysteresisControls.h" -class LossControls : public Component, - public Slider::Listener +using SliderAttachment = AudioProcessorValueTreeState::SliderAttachment; + +class LossControls : public Component { public: LossControls (ChowtapeModelAudioProcessor& proc); @@ -15,21 +17,14 @@ public: void resized() override; private: - void sliderValueChanged (Slider* slider) override; - void sliderDragStarted (Slider* slider) override; - void sliderDragEnded (Slider* slider) override; - MyLNF myLNF; ChowtapeModelAudioProcessor& processor; - ChowSlider spacingSlider; - ChowSlider thicknessSlider; - ChowSlider gapSlider; - - Label spacingLabel; - Label thicknessLabel; - Label gapLabel; + FullSlider speedSlider; + FullSlider spacingSlider; + FullSlider thicknessSlider; + FullSlider gapSlider; JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (LossControls) }; diff --git a/Plugin/Source/GUI Components/MainControls.cpp b/Plugin/Source/GUI Components/MainControls.cpp @@ -4,18 +4,19 @@ MainControls::MainControls (ChowtapeModelAudioProcessor& proc) : processor (proc) { - ChowtapeModelAudioProcessorEditor::createSlider (gainInKnob, processor.inGain, myLNF, this, String ("dB")); - ChowtapeModelAudioProcessorEditor::createSlider (gainOutKnob, processor.outGain, myLNF, this, String ("dB")); - ChowtapeModelAudioProcessorEditor::createComboBox (oversampling, processor.overSampling, this); - ChowtapeModelAudioProcessorEditor::createComboBox (tapeSpeed, processor.tapeSpeed, this); - //ChowtapeModelAudioProcessorEditor::createComboBox (tapeType, processor.tapeType, this); - oversampling.setEnabled (false); - - ChowtapeModelAudioProcessorEditor::createLabel (inGainLabel, processor.inGain, this); - ChowtapeModelAudioProcessorEditor::createLabel (outGainLabel, processor.outGain, this); - ChowtapeModelAudioProcessorEditor::createLabel (oversampleLabel, processor.overSampling, this); - ChowtapeModelAudioProcessorEditor::createLabel (speedLabel, processor.tapeSpeed, this); - //ChowtapeModelAudioProcessorEditor::createLabel (typeLabel, processor.tapeType, this); + ChowtapeModelAudioProcessorEditor::createSlider (gainInKnob, proc.getVTS(), "ingain", gainInAttach, *this, myLNF, " dB"); + ChowtapeModelAudioProcessorEditor::createSlider (gainOutKnob, proc.getVTS(), "outgain", gainOutAttach, *this, myLNF, " dB"); + ChowtapeModelAudioProcessorEditor::createComboBox (oversampling, proc.getVTS(), "os", osAttach, this, StringArray ({"2x", "4x", "8x"})); + // ChowtapeModelAudioProcessorEditor::createComboBox (oversampling, processor.overSampling, this); + // ChowtapeModelAudioProcessorEditor::createComboBox (tapeSpeed, processor.tapeSpeed, this); + // //ChowtapeModelAudioProcessorEditor::createComboBox (tapeType, processor.tapeType, this); + // oversampling.setEnabled (false); + + ChowtapeModelAudioProcessorEditor::createLabel (inGainLabel, gainInKnob.getName(), this); + ChowtapeModelAudioProcessorEditor::createLabel (outGainLabel, gainOutKnob.getName(), this); + ChowtapeModelAudioProcessorEditor::createLabel (oversampleLabel, oversampling.getName(), this); + // ChowtapeModelAudioProcessorEditor::createLabel (speedLabel, processor.tapeSpeed, this); + // //ChowtapeModelAudioProcessorEditor::createLabel (typeLabel, processor.tapeType, this); } void MainControls::paint (Graphics&) {} @@ -28,36 +29,12 @@ void MainControls::resized() oversampleLabel.setBounds (gainInKnob.getRight() - 7 * xOffset, 2 * labelY + yOffset, tapeWidth, labelHeight); oversampling.setBounds (gainInKnob.getRight(), oversampleLabel.getBottom(), overWidth, boxHeight); - speedLabel.setBounds (oversampling.getRight(), 2 * labelY + yOffset, tapeWidth, labelHeight); - tapeSpeed.setBounds (oversampling.getRight() + 2 * xOffset, speedLabel.getBottom(), tapeWidth, boxHeight); - - typeLabel.setBounds (gainInKnob.getRight(), oversampling.getBottom() + yOffset, typeWidth, labelHeight); - tapeType.setBounds (gainInKnob.getRight(), typeLabel.getBottom(), typeWidth, boxHeight); - - outGainLabel.setBounds (tapeSpeed.getRight(), labelY, sliderWidth, labelHeight); - gainOutKnob.setBounds (tapeSpeed.getRight(), sliderY, sliderWidth, sliderWidth); -} - -void MainControls::sliderValueChanged (Slider* slider) -{ - if (AudioParameterFloat* param = ChowtapeModelAudioProcessorEditor::getParamForSlider (slider, processor)) - *param = (float) slider->getValue(); -} - -void MainControls::sliderDragStarted(Slider* slider) -{ - if (AudioParameterFloat* param = ChowtapeModelAudioProcessorEditor::getParamForSlider (slider, processor)) - param->beginChangeGesture(); -} + // speedLabel.setBounds (oversampling.getRight(), 2 * labelY + yOffset, tapeWidth, labelHeight); + // tapeSpeed.setBounds (oversampling.getRight() + 2 * xOffset, speedLabel.getBottom(), tapeWidth, boxHeight); + // + // typeLabel.setBounds (gainInKnob.getRight(), oversampling.getBottom() + yOffset, typeWidth, labelHeight); + // tapeType.setBounds (gainInKnob.getRight(), typeLabel.getBottom(), typeWidth, boxHeight); -void MainControls::sliderDragEnded(Slider* slider) -{ - if (AudioParameterFloat* param = ChowtapeModelAudioProcessorEditor::getParamForSlider (slider, processor)) - param->endChangeGesture(); -} - -void MainControls::comboBoxChanged (ComboBox* box) -{ - if (AudioParameterChoice* param = ChowtapeModelAudioProcessorEditor::getParamForBox (box, processor)) - *param = box->getSelectedItemIndex(); + outGainLabel.setBounds (getWidth() - sliderWidth, labelY, sliderWidth, labelHeight); + gainOutKnob.setBounds (getWidth() - sliderWidth, sliderY, sliderWidth, sliderWidth); } diff --git a/Plugin/Source/GUI Components/MainControls.h b/Plugin/Source/GUI Components/MainControls.h @@ -5,9 +5,10 @@ #include "../GUI Extras/ChowSlider.h" #include "../GUI Extras/MyLNF.h" -class MainControls : public Component, - public Slider::Listener, - public ComboBox::Listener +using SliderAttachment = AudioProcessorValueTreeState::SliderAttachment; +using ComboBoxAttachment = AudioProcessorValueTreeState::ComboBoxAttachment; + +class MainControls : public Component { public: MainControls (ChowtapeModelAudioProcessor& proc); @@ -16,27 +17,27 @@ public: void resized() override; private: - void comboBoxChanged (ComboBox* box) override; - void sliderValueChanged (Slider* slider) override; - void sliderDragStarted (Slider* slider) override; - void sliderDragEnded (Slider* slider) override; - MyLNF myLNF; ChowtapeModelAudioProcessor& processor; - ChowSlider gainInKnob; - ChowSlider gainOutKnob; + Slider gainInKnob; + Slider gainOutKnob; + + std::unique_ptr<SliderAttachment> gainInAttach; + std::unique_ptr<SliderAttachment> gainOutAttach; ComboBox oversampling; - ComboBox tapeSpeed; - ComboBox tapeType; + std::unique_ptr<ComboBoxAttachment> osAttach; + + // ComboBox tapeSpeed; + // ComboBox tapeType; Label inGainLabel; Label outGainLabel; Label oversampleLabel; - Label speedLabel; - Label typeLabel; + // Label speedLabel; + // Label typeLabel; JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (MainControls) }; diff --git a/Plugin/Source/GUI Components/TimingControls.cpp b/Plugin/Source/GUI Components/TimingControls.cpp @@ -4,10 +4,14 @@ TimingControls::TimingControls (ChowtapeModelAudioProcessor& proc) : processor (proc) { - ChowtapeModelAudioProcessorEditor::createSlider (flutterDepthSlide, processor.flutterDepth, myLNF, this); - flutterDepthSlide.setSkewFactorFromMidPoint (1.0); - - ChowtapeModelAudioProcessorEditor::createLabel (flutterDepthLabel, processor.flutterDepth, this); + auto createSliderAndLabel = [=, &proc] (FullSlider& slider, String id) + { + ChowtapeModelAudioProcessorEditor::createSlider (slider.slider, proc.getVTS(), id, slider.attach, *this, myLNF); + ChowtapeModelAudioProcessorEditor::createLabel (slider.label, slider.slider.getName(), this); + }; + + createSliderAndLabel (depthSlide, "depth"); + createSliderAndLabel (rateSlide, "rate"); } void TimingControls::paint (Graphics& g) @@ -15,31 +19,15 @@ void TimingControls::paint (Graphics& g) g.setColour (Colours::antiquewhite); g.setFont (Font ((float) nameHeight).boldened()); - g.drawFittedText ("Timing Parameters:", Rectangle<int> (xOffset, yOffset, width, labelHeight), + g.drawFittedText ("Flutter Parameters:", Rectangle<int> (xOffset, yOffset, width, labelHeight), Justification::centredLeft, 1); } void TimingControls::resized() { - flutterDepthLabel.setBounds (0, 2 * yOffset + labelY, sliderWidth, labelHeight); - flutterDepthSlide.setBounds (0, 2 * yOffset + sliderY, sliderWidth, sliderWidth); -} - -void TimingControls::sliderValueChanged (Slider* slider) -{ - if (AudioParameterFloat* param = ChowtapeModelAudioProcessorEditor::getParamForSlider (slider, processor)) - *param = (float) slider->getValue(); -} - -void TimingControls::sliderDragStarted(Slider* slider) -{ - if (AudioParameterFloat* param = ChowtapeModelAudioProcessorEditor::getParamForSlider (slider, processor)) - param->beginChangeGesture(); -} + rateSlide.label.setBounds (0, 2 * yOffset + labelY, sliderWidth, labelHeight); + rateSlide.slider.setBounds (0, 2 * yOffset + sliderY, sliderWidth, sliderWidth); -void TimingControls::sliderDragEnded(Slider* slider) -{ - if (AudioParameterFloat* param = ChowtapeModelAudioProcessorEditor::getParamForSlider (slider, processor)) - param->endChangeGesture(); + depthSlide.label.setBounds (rateSlide.slider.getRight(), 2 * yOffset + labelY, sliderWidth, labelHeight); + depthSlide.slider.setBounds (rateSlide.slider.getRight(), 2 * yOffset + sliderY, sliderWidth, sliderWidth); } - diff --git a/Plugin/Source/GUI Components/TimingControls.h b/Plugin/Source/GUI Components/TimingControls.h @@ -4,9 +4,9 @@ #include "../PluginProcessor.h" #include "../GUI Extras/ChowSlider.h" #include "../GUI Extras/MyLNF.h" +#include "HysteresisControls.h" -class TimingControls : public Component, - public Slider::Listener +class TimingControls : public Component { public: TimingControls (ChowtapeModelAudioProcessor& proc); @@ -15,16 +15,12 @@ public: void resized() override; private: - void sliderValueChanged (Slider* slider) override; - void sliderDragStarted (Slider* slider) override; - void sliderDragEnded (Slider* slider) override; - MyLNF myLNF; ChowtapeModelAudioProcessor& processor; - ChowSlider flutterDepthSlide; - Label flutterDepthLabel; + FullSlider rateSlide; + FullSlider depthSlide; JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (TimingControls) }; diff --git a/Plugin/Source/PluginEditor.cpp b/Plugin/Source/PluginEditor.cpp @@ -8,7 +8,7 @@ ChowtapeModelAudioProcessorEditor::ChowtapeModelAudioProcessorEditor (ChowtapeMo mainControls.reset (new MainControls (processor)); addAndMakeVisible (mainControls.get()); - biasControls.reset (new BiasControls (processor)); + biasControls.reset (new HysteresisControls (processor)); addAndMakeVisible (biasControls.get()); lossControls.reset (new LossControls (processor)); @@ -24,47 +24,52 @@ ChowtapeModelAudioProcessorEditor::~ChowtapeModelAudioProcessorEditor() { } -void ChowtapeModelAudioProcessorEditor::createSlider (ChowSlider& slide, AudioParameterFloat* param, LookAndFeel& lnf, - Component* comp, String suffix, float step){ - slide.setName(param->name); - slide.setRange(param->range.start, param->range.end, step); - slide.setDefaultValue (param->convertFrom0to1 (dynamic_cast<AudioProcessorParameterWithID*> (param)->getDefaultValue())); - slide.setValue(*param); - - slide.setLookAndFeel (&lnf); - slide.setSliderStyle(Slider::RotaryHorizontalVerticalDrag); - slide.setColour(Slider::rotarySliderFillColourId, Colours::black); - slide.setColour(Slider::rotarySliderOutlineColourId, Colours::darkred); - slide.setColour(Slider::thumbColourId, Colours::antiquewhite); - slide.setTextBoxStyle(Slider::TextBoxBelow, false, 80, 20); - slide.setColour(Slider::textBoxTextColourId, Colours::antiquewhite); - slide.setColour(Slider::textBoxOutlineColourId, Colours::antiquewhite); - - if (suffix.isNotEmpty()) - slide.setTextValueSuffix (" " + suffix); - - slide.addListener (dynamic_cast<Slider::Listener*> (comp)); - comp->addAndMakeVisible (slide); +void ChowtapeModelAudioProcessorEditor::createSlider (Slider& slider, AudioProcessorValueTreeState& vts, String paramID, + std::unique_ptr<SliderAttachment>& attachment, Component& comp, LookAndFeel& myLNF, + String suffix, std::function<void()> onValueChange, + std::function<String (double)> textFromValue, std::function<double (String)> valueFromText) +{ + comp.addAndMakeVisible (slider); + attachment.reset (new SliderAttachment (vts, paramID, slider)); + + slider.setName (vts.getParameter (paramID)->name); + slider.textFromValueFunction = textFromValue; + slider.valueFromTextFunction = valueFromText; + slider.setNumDecimalPlacesToDisplay (2); + slider.setTextValueSuffix (suffix); + slider.onValueChange = onValueChange; + + slider.setLookAndFeel (&myLNF); + slider.setSliderStyle(Slider::RotaryHorizontalVerticalDrag); + slider.setColour(Slider::rotarySliderFillColourId, Colours::black); + slider.setColour(Slider::rotarySliderOutlineColourId, Colours::darkred); + slider.setColour(Slider::thumbColourId, Colours::antiquewhite); + slider.setTextBoxStyle(Slider::TextBoxBelow, false, 80, 20); + slider.setColour(Slider::textBoxTextColourId, Colours::antiquewhite); + slider.setColour(Slider::textBoxOutlineColourId, Colours::antiquewhite); } -void ChowtapeModelAudioProcessorEditor::createComboBox (ComboBox& box, AudioParameterChoice* choice, Component* comp) +void ChowtapeModelAudioProcessorEditor::createComboBox (ComboBox& box, AudioProcessorValueTreeState& vts, String paramID, + std::unique_ptr<ComboBoxAttachment>& attachment, Component* comp, + StringArray choices) { - box.setName (choice->name); - box.addItemList (choice->getAllValueStrings(), 1); - box.setSelectedItemIndex (*choice); + attachment.reset (new ComboBoxAttachment (vts, paramID, box)); + + box.setName (vts.getParameter (paramID)->name); + box.addItemList (choices, 1); + box.setSelectedItemIndex ((int) *vts.getRawParameterValue (paramID), dontSendNotification); box.setColour (ComboBox::backgroundColourId, Colours::black); box.setColour (ComboBox::outlineColourId, Colours::saddlebrown); box.setColour (ComboBox::textColourId, Colours::antiquewhite); box.setColour (ComboBox::arrowColourId, Colours::antiquewhite); - box.addListener (dynamic_cast<ComboBox::Listener*> (comp)); comp->addAndMakeVisible (box); } -void ChowtapeModelAudioProcessorEditor::createLabel (Label& label, AudioProcessorParameterWithID* param, Component* comp) +void ChowtapeModelAudioProcessorEditor::createLabel (Label& label, String name, Component* comp) { - label.setText (param->name, dontSendNotification); + label.setText (name, dontSendNotification); label.setJustificationType (Justification::centred); label.setColour (Label::textColourId, Colours::antiquewhite); label.setFont (17.0f); @@ -93,37 +98,3 @@ void ChowtapeModelAudioProcessorEditor::resized() lossControls->setBounds (0, 2 * sectionHeight, width, sectionHeight); timingControls->setBounds (0, 3 * sectionHeight, width, sectionHeight); } - -AudioParameterFloat* ChowtapeModelAudioProcessorEditor::getParamForSlider (Slider* slider, ChowtapeModelAudioProcessor& proc) -{ - if (proc.inGain->name == slider->getName()) - return proc.inGain; - else if (proc.outGain->name == slider->getName()) - return proc.outGain; - else if (proc.biasFreq->name == slider->getName()) - return proc.biasFreq; - else if (proc.biasGain->name == slider->getName()) - return proc.biasGain; - else if (proc.tapeSpacing->name == slider->getName()) - return proc.tapeSpacing; - else if (proc.tapeThickness->name == slider->getName()) - return proc.tapeThickness; - else if (proc.gapWidth->name == slider->getName()) - return proc.gapWidth; - else if (proc.flutterDepth->name == slider->getName()) - return proc.flutterDepth; - else - return nullptr; -} - -AudioParameterChoice* ChowtapeModelAudioProcessorEditor::getParamForBox (ComboBox* box, ChowtapeModelAudioProcessor& proc) -{ - if (proc.overSampling->name == box->getName()) - return proc.overSampling; - else if (proc.tapeSpeed->name == box->getName()) - return proc.tapeSpeed; - else if (proc.tapeType->name == box->getName()) - return proc.tapeType; - else - return nullptr; -} diff --git a/Plugin/Source/PluginEditor.h b/Plugin/Source/PluginEditor.h @@ -3,7 +3,7 @@ #include "PluginProcessor.h" #include "GUI Extras/MyLNF.h" #include "GUI Extras/ChowSlider.h" -#include "GUI Components/BiasControls.h" +#include "GUI Components/HysteresisControls.h" #include "GUI Components/MainControls.h" #include "GUI Components/LossControls.h" #include "GUI Components/TimingControls.h" @@ -25,7 +25,7 @@ enum sliderWidth = 110, sliderY = 25, - overWidth = 65, + overWidth = 90, tapeWidth = 90, typeWidth = 150, speedWidth = 150, @@ -42,21 +42,19 @@ public: void paint (Graphics&) override; void resized() override; - static void createSlider (ChowSlider& slide, AudioParameterFloat* param, LookAndFeel& lnf, - Component* comp, String suffix = String(), float step = 0.1f); - static void createComboBox (ComboBox& box, AudioParameterChoice* choice, Component* comp); - static void createLabel (Label& label, AudioProcessorParameterWithID* param, Component* comp); - - static AudioParameterFloat* getParamForSlider (Slider* slider, ChowtapeModelAudioProcessor& proc); - static AudioParameterChoice* getParamForBox (ComboBox* box, ChowtapeModelAudioProcessor& proc); + static void createSlider (Slider& slider, AudioProcessorValueTreeState& vts, String paramID, + std::unique_ptr<SliderAttachment>& attachment, Component& comp, LookAndFeel& myLNF, + String suffix = "", std::function<void()> onValueChange = {}, + std::function<String (double)> textFromValue = {}, std::function<double (String)> valueFromText = {}); + static void createComboBox (ComboBox& box, AudioProcessorValueTreeState& vts, String paramID, + std::unique_ptr<ComboBoxAttachment>& attachment, Component* comp, StringArray choices); + static void createLabel (Label& label, String name, Component* comp); private: - MyLNF myLNF; - ChowtapeModelAudioProcessor& processor; std::unique_ptr<MainControls> mainControls; - std::unique_ptr<BiasControls> biasControls; + std::unique_ptr<HysteresisControls> biasControls; std::unique_ptr<LossControls> lossControls; std::unique_ptr<TimingControls> timingControls; diff --git a/Plugin/Source/PluginProcessor.cpp b/Plugin/Source/PluginProcessor.cpp @@ -1,3 +1,13 @@ +/* + ============================================================================== + + This file was auto-generated! + + It contains the basic framework code for a JUCE plugin processor. + + ============================================================================== +*/ + #include "PluginProcessor.h" #include "PluginEditor.h" @@ -11,86 +21,32 @@ ChowtapeModelAudioProcessor::ChowtapeModelAudioProcessor() #endif .withOutput ("Output", AudioChannelSet::stereo(), true) #endif - ) + ), #endif + vts (*this, nullptr, Identifier ("Parameters"), createParameterLayout()), + hysteresis (vts), + flutter (vts) { - //Main Controls - addParameter (inGain = new AudioParameterFloat (String ("inGain"), String ("Input Gain"), -30.0f, 30.0f, 0.0f)); - inGain->addListener (this); - - addParameter (outGain = new AudioParameterFloat (String ("outGain"), String ("Output Gain"), -30.0f, 30.0f, 0.0f)); - outGain->addListener (this); - - addParameter (overSampling = new AudioParameterChoice (String ("overSampling"), String ("Upsample"), - StringArray ({ "2x", "4x", "8x", "16x" }), 3)); - overSampling->addListener (this); - - addParameter (tapeSpeed = new AudioParameterChoice (String ("tapeSpeed"), String ("Speed"), - StringArray ({ "3.75 ips", "7.5 ips", "15 ips" }), 1)); - tapeSpeed->addListener (this); - - addParameter (tapeType = new AudioParameterChoice (String ("tapeType"), String ("Tape Type"), - StringArray ({ "Iron Oxide", "Chromium Oxide" }), 0)); - tapeType->addListener(this); - - //Bias Controls - addParameter (biasFreq = new AudioParameterFloat (String ("biasFreq"), String ("Bias Frequency"), 45.0f, 55.0f, 50.0)); - biasFreq->addListener (this); - - addParameter (biasGain = new AudioParameterFloat (String ("biasGain"), String ("Bias Gain"), 0.0f, 10.0f, 5.0f)); - biasGain->addListener (this); - - //Loss Controls - addParameter (tapeSpacing = new AudioParameterFloat (String ("tapeSpacing"), String ("Spacing"), 0.0f, 50.0f, 0.001f)); - tapeSpacing->addListener (this); - - addParameter (tapeThickness = new AudioParameterFloat (String ("tapeThickness"), String ("Thickness"), 0.0f, 50.0f, 10.0f)); - tapeThickness->addListener (this); - - addParameter (gapWidth = new AudioParameterFloat (String ("gapWidth"), String ("Gap Width"), 2.5f, 12.0f, 3.0f)); - gapWidth->addListener (this); - - //Timing Controls - addParameter (flutterDepth = new AudioParameterFloat (String ("flutterDepth"), String ("Flutter Depth"), 0.0f, 5.0f, 1.0f)); - flutterDepth->addListener (this); - - lossEffects.setSpeed (*tapeSpeed); - hysteresis.setOverSamplingFactor (*overSampling); - hysteresis.setBiasFreq (*biasFreq); - timingEffect.setDepth (*flutterDepth); + for (int ch = 0; ch < 2; ++ch) + lossFilter[ch].reset (new LossFilter (vts)); } ChowtapeModelAudioProcessor::~ChowtapeModelAudioProcessor() { } -void ChowtapeModelAudioProcessor::parameterValueChanged (int paramIndex, float /*newValue*/) +AudioProcessorValueTreeState::ParameterLayout ChowtapeModelAudioProcessor::createParameterLayout() { - if (paramIndex == inGain->getParameterIndex()) - inGainProc.setGain (Decibels::decibelsToGain ((float) *inGain)); - else if (paramIndex == outGain->getParameterIndex()) - outGainProc.setGain (Decibels::decibelsToGain ((float) *outGain)); - //else if (paramIndex == overSampling->getParameterIndex()) - // hysteresis.setOverSamplingFactor (*overSampling); - else if (paramIndex == tapeSpeed->getParameterIndex()) - { - lossEffects.setSpeed (*tapeSpeed); - timingEffect.setTapeSpeed (*tapeSpeed); - } - //else if (paramIndex == tapeType->getParameterIndex()) - // return; //@TODO - //else if (paramIndex == biasFreq->getParameterIndex()) - // hysteresis.setBiasFreq (*biasFreq); - else if (paramIndex == biasGain->getParameterIndex()) - hysteresis.setBiasGain (*biasGain); - else if (paramIndex == tapeSpacing->getParameterIndex()) - lossEffects.setSpacing (*tapeSpacing); - else if (paramIndex == tapeThickness->getParameterIndex()) - lossEffects.setThickness (*tapeThickness); - else if (paramIndex == gapWidth->getParameterIndex()) - lossEffects.setGap (*gapWidth); - else if (paramIndex == flutterDepth->getParameterIndex()) - timingEffect.setDepth (*flutterDepth); + std::vector<std::unique_ptr<RangedAudioParameter>> params; + + params.push_back (std::make_unique<AudioParameterFloat> ("ingain", "Input Gain", -30.0f, 6.0f, 0.0f)); + params.push_back (std::make_unique<AudioParameterFloat> ("outgain", "Output Gain", -30.0f, 6.0f, 0.0f)); + + HysteresisProcessor::createParameterLayout (params); + LossFilter::createParameterLayout (params); + Flutter::createParameterLayout (params); + + return { params.begin(), params.end() }; } //============================================================================== @@ -142,36 +98,35 @@ int ChowtapeModelAudioProcessor::getCurrentProgram() return 0; } -void ChowtapeModelAudioProcessor::setCurrentProgram (int /*index*/) +void ChowtapeModelAudioProcessor::setCurrentProgram (int index) { } -const String ChowtapeModelAudioProcessor::getProgramName (int /*index*/) +const String ChowtapeModelAudioProcessor::getProgramName (int index) { return {}; } -void ChowtapeModelAudioProcessor::changeProgramName (int /*index*/, const String& /*newName*/) +void ChowtapeModelAudioProcessor::changeProgramName (int index, const String& newName) { } //============================================================================== void ChowtapeModelAudioProcessor::prepareToPlay (double sampleRate, int samplesPerBlock) { - inGainProc.prepareToPlay (sampleRate, samplesPerBlock); + inGain.prepareToPlay (sampleRate, samplesPerBlock); hysteresis.prepareToPlay (sampleRate, samplesPerBlock); - timingEffect.prepareToPlay (sampleRate, samplesPerBlock); - lossEffects.prepareToPlay (sampleRate, samplesPerBlock); - outGainProc.prepareToPlay (sampleRate, samplesPerBlock); + + for (int ch = 0; ch < 2; ++ch) + lossFilter[ch]->prepare ((float) sampleRate, samplesPerBlock); + + flutter.prepareToPlay (sampleRate, samplesPerBlock); + outGain.prepareToPlay (sampleRate, samplesPerBlock); } void ChowtapeModelAudioProcessor::releaseResources() { - inGainProc.releaseResources(); hysteresis.releaseResources(); - timingEffect.releaseResources(); - lossEffects.releaseResources(); - outGainProc.releaseResources(); } #ifndef JucePlugin_PreferredChannelConfigurations @@ -202,15 +157,18 @@ void ChowtapeModelAudioProcessor::processBlock (AudioBuffer<float>& buffer, Midi { ScopedNoDenormals noDenormals; - inGainProc.processBlock (buffer, midiMessages); + inGain.setGain (Decibels::decibelsToGain (*vts.getRawParameterValue ("ingain"))); + outGain.setGain (Decibels::decibelsToGain (*vts.getRawParameterValue ("outgain"))); + inGain.processBlock (buffer, midiMessages); hysteresis.processBlock (buffer, midiMessages); + + flutter.processBlock (buffer, midiMessages); - timingEffect.processBlock (buffer, midiMessages); - - lossEffects.processBlock (buffer, midiMessages); + for (int ch = 0; ch < buffer.getNumChannels(); ++ch) + lossFilter[ch]->processBlock (buffer.getWritePointer (ch), buffer.getNumSamples()); - outGainProc.processBlock (buffer, midiMessages); + outGain.processBlock (buffer, midiMessages); } //============================================================================== @@ -227,37 +185,18 @@ AudioProcessorEditor* ChowtapeModelAudioProcessor::createEditor() //============================================================================== void ChowtapeModelAudioProcessor::getStateInformation (MemoryBlock& destData) { - std::unique_ptr<XmlElement> xml (new XmlElement ("ChowTapeXmlData")); - - xml->setAttribute ("inGain", (double) *inGain); - xml->setAttribute ("outGain", (double) *outGain); - xml->setAttribute ("tapeSpeed", tapeSpeed->getIndex()); - xml->setAttribute ("biasGain", (double) *biasGain); - xml->setAttribute ("tapeSpacing", (double) *tapeSpacing); - xml->setAttribute ("tapeThickness", (double) *tapeThickness); - xml->setAttribute ("gapWidth", (double) *gapWidth); - xml->setAttribute ("flutterDepth", (double) *flutterDepth); - + auto state = vts.copyState(); + std::unique_ptr<XmlElement> xml (state.createXml()); copyXmlToBinary (*xml, destData); } void ChowtapeModelAudioProcessor::setStateInformation (const void* data, int sizeInBytes) { std::unique_ptr<XmlElement> xmlState (getXmlFromBinary (data, sizeInBytes)); + if (xmlState.get() != nullptr) - { - if (xmlState->hasTagName ("ChowTapeXmlData")) - { - *inGain = (float) xmlState->getDoubleAttribute ("inGain", 0.0); - *outGain = (float) xmlState->getDoubleAttribute ("outGain", 0.0); - *tapeSpeed = xmlState->getIntAttribute ("tapeSpeed", 0); - *biasGain= (float) xmlState->getDoubleAttribute ("biasGain", 0.0); - *tapeSpacing = (float) xmlState->getDoubleAttribute ("tapeSpacing", 0.0); - *tapeThickness = (float) xmlState->getDoubleAttribute ("tapeThickness", 0.0); - *gapWidth= (float) xmlState->getDoubleAttribute ("gapWidth", 0.0); - *flutterDepth= (float) xmlState->getDoubleAttribute ("flutterDepth", 0.0); - } - } + if (xmlState->hasTagName (vts.state.getType())) + vts.replaceState (ValueTree::fromXml (*xmlState)); } //============================================================================== diff --git a/Plugin/Source/PluginProcessor.h b/Plugin/Source/PluginProcessor.h @@ -1,16 +1,25 @@ +/* + ============================================================================== + + This file was auto-generated! + + It contains the basic framework code for a JUCE plugin processor. + + ============================================================================== +*/ + #pragma once #include "../JuceLibraryCode/JuceHeader.h" -#include "Processors/Hysteresis/HysteresisProcessor.h" #include "Processors/GainProcessor.h" -#include "Processors/Loss Effects/LossEffectsFilter.h" -#include "Processors/Timing Effects/TimingEffect.h" +#include "Processors/Hysteresis/HysteresisProcessor.h" +#include "Processors/Loss_Effects/LossFilter.h" +#include "Processors/Timing_Effects/Flutter.h" //============================================================================== /** */ -class ChowtapeModelAudioProcessor : public AudioProcessor, - public AudioProcessorParameter::Listener +class ChowtapeModelAudioProcessor : public AudioProcessor { public: //============================================================================== @@ -50,31 +59,17 @@ public: void getStateInformation (MemoryBlock& destData) override; void setStateInformation (const void* data, int sizeInBytes) override; - AudioParameterFloat* inGain; - AudioParameterFloat* outGain; - AudioParameterChoice* overSampling; - AudioParameterChoice* tapeSpeed; - AudioParameterChoice* tapeType; - - AudioParameterFloat* biasFreq; - AudioParameterFloat* biasGain; - - AudioParameterFloat* tapeSpacing; - AudioParameterFloat* tapeThickness; - AudioParameterFloat* gapWidth; - - AudioParameterFloat* flutterDepth; - - void parameterValueChanged (int paramIndex, float /*newValue*/) override; - void parameterGestureChanged (int /*paramIndex*/, bool /*gestureIsStarting*/) override {} + AudioProcessorValueTreeState::ParameterLayout createParameterLayout(); + AudioProcessorValueTreeState& getVTS() { return vts; } private: - HysteresisProcessor hysteresis; - LossEffectsFilter lossEffects; - TimingEffect timingEffect; + AudioProcessorValueTreeState vts; - GainProcessor inGainProc; - GainProcessor outGainProc; + GainProcessor inGain; + HysteresisProcessor hysteresis; + std::unique_ptr<LossFilter> lossFilter[2]; + Flutter flutter; + GainProcessor outGain; //============================================================================== JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ChowtapeModelAudioProcessor) diff --git a/Plugin/Source/Processors/GainProcessor.h b/Plugin/Source/Processors/GainProcessor.h @@ -1,21 +1,19 @@ #ifndef GAINPROCESSOR_H_INCLUDED #define GAINPROCESSOR_H_INCLUDED -#include "ProcessorBase.h" +#include "JuceHeader.h" -class GainProcessor : public ProcessorBase +class GainProcessor { public: - GainProcessor() : ProcessorBase (String ("Gain Processor")) {} + GainProcessor() {} - void prepareToPlay (double sampleRate, int maximumExpectedSamplesPerBlock) override + void prepareToPlay (double /*sampleRate*/, int /*maximumExpectedSamplesPerBlock*/) { - setRateAndBufferSizeDetails (sampleRate, maximumExpectedSamplesPerBlock); oldGain = 0.0f; } - void releaseResources() override {} - void processBlock (AudioBuffer<float>& buffer, MidiBuffer& /*midiMessages*/) override + void processBlock (AudioBuffer<float>& buffer, MidiBuffer& /*midiMessages*/) { if (curGain != oldGain) { diff --git a/Plugin/Source/Processors/Hysteresis/HysteresisProcessing.cpp b/Plugin/Source/Processors/Hysteresis/HysteresisProcessing.cpp @@ -1,62 +1,94 @@ #include "HysteresisProcessing.h" #include <math.h> +inline int sign (float x) +{ + return (x > 0.0f) - (x < 0.0f); +} + HysteresisProcessing::HysteresisProcessing() { +} + +void HysteresisProcessing::reset() +{ + M_n1 = 0.0f; + H_n1 = 0.0f; + H_d_n1 = 0.0f; + coth = 0.0f; + nearZero = false; } -float HysteresisProcessing::cothApprox (float x) +void HysteresisProcessing::cook (float drive, float width, float sat) { - const auto xSquare = x * x; + M_s = 0.5f + 1.5f * (1.0f - sat); + a = M_s / (0.01f + 6.0f * drive); + c = sqrtf (1.0f - width) - 0.01f; + + nc = 1.0f - c; + M_s_oa = M_s / a; + M_s_oa_tc = c * M_s_oa; + M_s_oa_tc_talpha = alpha * M_s_oa_tc; +} + +inline float HysteresisProcessing::cothApprox (float x) +{ + const auto exp_2x = expf (2.0f * x); // expf_approx (2.0f * x); - return (1.0f + (xSquare / (3.0f + (xSquare / (5.0f + (xSquare / (7.0f))))))) / x; + return (exp_2x + 1.0f) / (exp_2x - 1.0f); + + // const auto xSquare = x * x; + + // return (1.0f + (xSquare / (3.0f + (xSquare / (5.0f + (xSquare / (7.0f))))))) / x; + // return (1.0f + (xSquare / (3.0f + (xSquare / (5.0f))))) / x; + // return (1.0f + (xSquare / 3.0f)) / x; } -float HysteresisProcessing::langevin (float x) +inline float HysteresisProcessing::langevin (float x) { - if (nearZero) + if (! nearZero) return (coth) - (1.0f / x); else return (x / 3.0f); } -float HysteresisProcessing::langevinD (float x) +inline float HysteresisProcessing::langevinD (float x) { - if (nearZero) + if (! nearZero) return (1.0f / (x * x)) - (coth * coth) + 1.0f; else return (1.0f / 3.0f); } -float HysteresisProcessing::deriv (float x_n, float x_n1, float x_d_n1) +inline float HysteresisProcessing::deriv (float x_n, float x_n1, float x_d_n1) { - return ((2.0f * fs) * (x_n - x_n1)) - x_d_n1; + return (twoFs * (x_n - x_n1)) - x_d_n1; } -float HysteresisProcessing::hysteresisFunc (float M, float H, float H_d) +inline float HysteresisProcessing::hysteresisFunc (float M, float H, float H_d) { - const float Q = (H + alpha * M) / a; - coth = cothApprox (Q); - nearZero = std::abs (Q) > (float) (10e-4); - - const float M_diff = M_s * langevin (Q) - M; - - const float delta = H_d > 0 ? 1.0f : -1.0f; - const float delta_M = std::signbit (delta) == std::signbit (M_diff) ? 1.0f : 0.0f; - - const float L_prime = langevinD (Q); - - //const float denominator = 1 - (c * alpha * (M_s / a) * L_prime); - // - //const float t1_num = (1 - c) * delta_M * M_diff; - //const float t1_den = ((1 - c) * delta * k) - (alpha * M_diff); - //const float t1 = (t1_num / t1_den) * H_d; - // - //const float t2 = c * (M_s / a) * H_d * L_prime; - // - //return (t1 + t2) / denominator; - return (H_d * (((1 - c) * delta_M * M_diff) / (((1 - c) * delta * k) - (alpha * M_diff))) + (c * (M_s / a) * H_d * L_prime)) / (1 - (c * alpha * (M_s / a) * L_prime)); + Q = (H + alpha * M) / a; + coth = 1.0f / dsp::FastMathApproximations::tanh (Q); + nearZero = Q < 0.001f && Q > -0.001f; + + M_diff = M_s * langevin (Q) - M; + + delta = (float) ((H_d >= 0.0f) - (H_d < 0.0f)); + delta_M = (float) (sign (delta) == sign (M_diff)); + + L_prime = langevinD (Q); + + // const float denominator = 1 - (c * alpha * (M_s / a) * L_prime); + // + // const float t1_num = (1 - c) * delta_M * M_diff; + // const float t1_den = ((1 - c) * delta * k) - (alpha * M_diff); + // const float t1 = (t1_num / t1_den) * H_d; + // + // const float t2 = c * (M_s / a) * H_d * L_prime; + // + // return (t1 + t2) / denominator; + return H_d * (((nc * delta_M * M_diff) / ((nc * delta * k) - (alpha * M_diff))) + (M_s_oa_tc * L_prime)) / (1.0f - (M_s_oa_tc_talpha * L_prime)); } float HysteresisProcessing::M_n (float prevM, float k1, float k2, float k3, float k4) @@ -66,18 +98,23 @@ float HysteresisProcessing::M_n (float prevM, float k1, float k2, float k3, floa float HysteresisProcessing::process (float H) { - const float H_d = deriv (H, H_n1, H_d_n1); + float H_d = deriv (H, H_n1, H_d_n1); const float k1 = T * hysteresisFunc (M_n1, H_n1, H_d_n1); const float k2 = T * hysteresisFunc (M_n1 + (k1 / 2.0f), (H + H_n1) / 2.0f, (H_d + H_d_n1) / 2.0f); - const float k3 = T * hysteresisFunc (M_n1 + (k2 / 2.0f), (H + H_n1) / 2.0f, (H_d + H_d_n1) / 2.0f); - const float k4 = T * hysteresisFunc (M_n1 + k3, H, H_d); - const float M = M_n (M_n1, k1, k2, k3, k4); + float M = M_n1 + k2; + + if (std::isnan (M)) + { + M = 0.0f; + H = 0.0f; + H_d = 0.0f; + } M_n1 = M; H_n1 = H; H_d_n1 = H_d; - return M / M_s; + return M; } diff --git a/Plugin/Source/Processors/Hysteresis/HysteresisProcessing.h b/Plugin/Source/Processors/Hysteresis/HysteresisProcessing.h @@ -3,43 +3,61 @@ #include "JuceHeader.h" +/* + Hysteresis processing for a model of a Sowter 3575f Transformer. + For more information on the DSP happening here, see: + https://ccrma.stanford.edu/~jatin/420/tape/TapeModel_DAFx.pdf +*/ class HysteresisProcessing { public: HysteresisProcessing(); + /* Process a single sample */ float process (float H); - void setSampleRate (float newSR) { fs = newSR; T = 1.0f / fs; } + void reset(); + void setSampleRate (float newSR) { fs = newSR; T = 1.0f / fs; twoFs = 2.0f * fs; } + + void cook (float drive, float width, float sat); private: - float cothApprox (float x); + /* Continued fraction approximation for hyperbolic cotangent */ + inline float cothApprox (float x); - float langevin (float x); - float langevinD (float x); - float deriv (float x_n, float x_n1, float x_d_n1); + inline float langevin (float x); // Langevin function + inline float langevinD (float x); // Derivative of Langevin function + inline float deriv (float x_n, float x_n1, float x_d_n1); // Derivative by trapezoidal rule - float hysteresisFunc (float M, float H, float H_d); + inline float hysteresisFunc (float M, float H, float H_d); float M_n (float M_n1, float k1, float k2, float k3, float k4); float fs = 48000.0f; float T = 1.0f / fs; - const float M_s = 350000.0f; - const float a = (float) 2.2e4; + float M_s = (float) 1; + float a = M_s / 4.0f; const float alpha = (float) 1.6e-3; - const float k = (float) 27.0e3; - const float c = (float) 1.7e-1; + const float k = 0.47875f; + float c = (float) 1.7e-1; + + // Save calculations + float twoFs = 2.0f * fs; + float nc = 1-c; + float M_s_oa = M_s / a; + float M_s_oa_tc = c * M_s / a; + float M_s_oa_tc_talpha = alpha * c * M_s / a; float M_n1 = 0.0f; float H_n1 = 0.0f; float H_d_n1 = 0.0f; + // temp vars + float Q, M_diff, delta, delta_M, L_prime; + float coth = 0.0f; - // float tanHyp = 0.0f; - // float tanhRecip = 0.0f; bool nearZero = false; - //JUCE_DECLARE_NONCOPYABLE_WITH_LEAK_DETECTOR (HysteresisProcessing) + // JUCE_DECLARE_NONCOPYABLE_WITH_LEAK_DETECTOR (HysteresisProcessing) }; #endif diff --git a/Plugin/Source/Processors/Hysteresis/HysteresisProcessor.cpp b/Plugin/Source/Processors/Hysteresis/HysteresisProcessor.cpp @@ -1,62 +1,156 @@ #include "HysteresisProcessor.h" -HysteresisProcessor::HysteresisProcessor() : ProcessorBase ("HysteresisProcessor") +enum { - overSample2.reset (new dsp::Oversampling<float> (2, 1, dsp::Oversampling<float>::filterHalfBandPolyphaseIIR)); - overSample4.reset (new dsp::Oversampling<float> (2, 2, dsp::Oversampling<float>::filterHalfBandPolyphaseIIR)); - overSample8.reset (new dsp::Oversampling<float> (2, 3, dsp::Oversampling<float>::filterHalfBandPolyphaseIIR)); - overSample16.reset (new dsp::Oversampling<float> (2, 4, dsp::Oversampling<float>::filterHalfBandPolyphaseIIR)); + reset = 1048576, + numSteps = 500, +}; + +HysteresisProcessor::HysteresisProcessor (AudioProcessorValueTreeState& vts) +{ + driveParam = vts.getRawParameterValue ("drive"); + satParam = vts.getRawParameterValue ("sat"); + widthParam = vts.getRawParameterValue ("width"); + osParam = vts.getRawParameterValue ("os"); + + overSample[0].reset (new dsp::Oversampling<float> (2, 1, dsp::Oversampling<float>::filterHalfBandPolyphaseIIR)); + overSample[1].reset (new dsp::Oversampling<float> (2, 2, dsp::Oversampling<float>::filterHalfBandPolyphaseIIR)); + overSample[2].reset (new dsp::Oversampling<float> (2, 3, dsp::Oversampling<float>::filterHalfBandPolyphaseIIR)); + + for (int ch = 0; ch < 2; ++ch) + { + drive[ch].reset (numSteps); + width[ch].reset (numSteps); + sat[ch].reset (numSteps); + makeup[ch].reset (numSteps); + } } -void HysteresisProcessor::prepareToPlay (double sampleRate, int samplesPerBlock) +void HysteresisProcessor::createParameterLayout (std::vector<std::unique_ptr<RangedAudioParameter>>& params) { - setRateAndBufferSizeDetails (sampleRate, samplesPerBlock); + params.push_back (std::make_unique<AudioParameterFloat> ("drive", "Drive", 0.0f, 1.0f, 0.5f)); + params.push_back (std::make_unique<AudioParameterFloat> ("sat", "Saturation", 0.0f, 1.0f, 0.5f)); + params.push_back (std::make_unique<AudioParameterFloat> ("width", "Bias", 0.0f, 1.0f, 0.5f)); - hProcs[0].setSampleRate ((float) (sampleRate * overSamplingFactor)); - hProcs[1].setSampleRate ((float) (sampleRate * overSamplingFactor)); + params.push_back (std::make_unique<AudioParameterChoice> ("os", "Oversampling", StringArray ({"2x", "4x", "8x"}), 0)); +} - if (overSamplingFactor == 16) - overSample16->initProcessing (samplesPerBlock); - else if (overSamplingFactor == 8) - overSample8->initProcessing (samplesPerBlock); - else if (overSamplingFactor == 4) - overSample4->initProcessing (samplesPerBlock); - else - overSample2->initProcessing (samplesPerBlock); +float HysteresisProcessor::calcMakeup() +{ + return (1.0f + 0.6f * width[0].getTargetValue()) / (0.5f + 1.5f * (1.0f - sat[0].getTargetValue())); +} - currentSampleRate = (float) sampleRate * overSamplingFactor; - updateAngleDelta(); +void HysteresisProcessor::setDrive (float newDrive) +{ + for (int ch = 0; ch < 2; ++ch) + { + drive[ch].setTargetValue (newDrive); + } +} - fadeIn = true; +void HysteresisProcessor::setWidth (float newWidth) +{ + for (int ch = 0; ch < 2; ++ch) + { + width[ch].setTargetValue (newWidth); + makeup[ch].setTargetValue (calcMakeup()); + } +} + +void HysteresisProcessor::setSaturation (float newSaturation) +{ + for (int ch = 0; ch < 2; ++ch) + { + sat[ch].setTargetValue (newSaturation); + makeup[ch].setTargetValue (calcMakeup()); + } +} + +void HysteresisProcessor::toggleOnOff (bool shouldBeOn) +{ + if (shouldBeOn == isOn) + return; + + isChanging = true; +} + +void HysteresisProcessor::prepareToPlay (double sampleRate, int samplesPerBlock) +{ + fs = (float) sampleRate; + overSamplingFactor = (int) powf(2.0f, *osParam + 1); + + for (int ch = 0; ch < 2; ++ch) + { + drive[ch].skip (numSteps); + width[ch].skip (numSteps); + sat[ch].skip (numSteps); + makeup[ch].skip (numSteps); + + hProcs[ch].setSampleRate ((float) (sampleRate * overSamplingFactor)); + hProcs[ch].cook (drive[ch].getCurrentValue(), width[ch].getCurrentValue(), sat[ch].getCurrentValue()); + hProcs[ch].reset(); + } + + overSample[0]->initProcessing (samplesPerBlock); + overSample[1]->initProcessing (samplesPerBlock); + overSample[2]->initProcessing (samplesPerBlock); + prevOS = (int) *osParam; + + dcBlocker[0].reset (sampleRate); + dcBlocker[0].calcCoefs (dcFreq, 0.707f); + dcBlocker[1].reset (sampleRate); + dcBlocker[1].calcCoefs (dcFreq, 0.707f); + + // dcLower[0].reset (sampleRate); + // dcLower[0].calcCoefs (dcShelfFreq, 0.707f, Decibels::decibelsToGain (-12.0f)); + // dcLower[1].reset (sampleRate); + // dcLower[1].calcCoefs (dcShelfFreq, 0.707f, Decibels::decibelsToGain (-12.0f)); + + fadeBuffer.setSize (2, samplesPerBlock); + resetCount = 0; + + toggleOnOff (true); } void HysteresisProcessor::releaseResources() { - overSample2->reset(); - overSample4->reset(); - overSample8->reset(); - overSample16->reset(); + overSample[0]->reset(); + overSample[1]->reset(); + overSample[2]->reset(); } void HysteresisProcessor::processBlock (AudioBuffer<float>& buffer, MidiBuffer& /*midi*/) { - if (fadeIn) + setDrive (*driveParam); + setSaturation (*satParam); + setWidth (1.0f - *widthParam); + + if ((int) *osParam != prevOS) { - buffer.applyGainRamp (0, buffer.getNumSamples(), 0.0f, 1.0f); - fadeIn = false; + overSamplingFactor = (int) powf(2.0f, *osParam + 1); + prevOS = (int) *osParam; + + for (int ch = 0; ch < 2; ++ch) + { + hProcs[ch].setSampleRate (fs * overSamplingFactor); + hProcs[ch].cook (drive[ch].getCurrentValue(), width[ch].getCurrentValue(), sat[ch].getCurrentValue()); + hProcs[ch].reset(); + } + } + + if (! isOn && ! isChanging) //bypass + return; + + if (isChanging || resetCount + buffer.getNumSamples() >= reset) + { + for (int ch = 0; ch < buffer.getNumChannels(); ++ch) + fadeBuffer.copyFrom (ch, 0, buffer, ch, 0, buffer.getNumSamples()); } dsp::AudioBlock<float> block (buffer); dsp::AudioBlock<float> osBlock; - if (overSamplingFactor == 16) - osBlock = overSample16->processSamplesUp (block); - else if (overSamplingFactor == 8) - osBlock = overSample8->processSamplesUp (block); - else if (overSamplingFactor == 4) - osBlock = overSample4->processSamplesUp (block); - else - osBlock = overSample2->processSamplesUp (block); + osBlock = overSample[(int) *osParam]->processSamplesUp (block); float* ptrArray[] = { osBlock.getChannelPointer(0), osBlock.getChannelPointer(1) }; AudioBuffer<float> osBuffer (ptrArray, 2, static_cast<int> (osBlock.getNumSamples())); @@ -66,74 +160,75 @@ void HysteresisProcessor::processBlock (AudioBuffer<float>& buffer, MidiBuffer& auto* x = osBuffer.getWritePointer (channel); for (int samp = 0; samp < osBuffer.getNumSamples(); samp++) { - //x[samp] = hProcs[channel].process ((float) 5e4 * (x[samp])); - x[samp] = hProcs[channel].process ((float) 5e3 * (x[samp] + biasGain * sinf (currentAngle[channel]))); - //x[samp] = biasGain * sinf (currentAngle[channel]); + if (drive[channel].isSmoothing() || width[channel].isSmoothing() || sat[channel].isSmoothing()) + hProcs[channel].cook (drive[channel].getNextValue(), width[channel].getNextValue(), sat[channel].getNextValue()); - currentAngle[channel] += angleDelta; - if (currentAngle[channel] > MathConstants<float>::twoPi) - currentAngle[channel] -= MathConstants<float>::twoPi; + x[samp] = hProcs[channel].process (x[samp]) * makeup[channel].getNextValue(); } } - if (overSamplingFactor == 16) - overSample16->processSamplesDown (block); - else if (overSamplingFactor == 8) - overSample8->processSamplesDown (block); - else if (overSamplingFactor == 4) - overSample4->processSamplesDown (block); - else - overSample2->processSamplesDown (block); + overSample[(int) *osParam]->processSamplesDown (block); - buffer.applyGain (10.0f); -} + resetCount += buffer.getNumSamples(); + if (resetCount >= reset && ! isChanging) + { + resetCount = 0; -void HysteresisProcessor::updateAngleDelta() -{ - auto cyclesPerSample = biasFreq / currentSampleRate; - angleDelta = cyclesPerSample * 2.0f * MathConstants<float>::pi; -} + dsp::AudioBlock<float> fadeBlock (fadeBuffer); + dsp::AudioBlock<float> osFadeBlock; -void HysteresisProcessor::setOverSamplingFactor (String osFactor) -{ - //osFactor = "8x"; - auto factor = overSamplingFactor; - if (osFactor == "2x") - factor = 2; // set overSample factor 2 = 2^1 - else if (osFactor == "4x") - factor = 4; // set overSample factor 4 = 2^2 - else if (osFactor == "8x") - factor = 8; // set overSample factor 8 = 2^3 - else if (osFactor == "16x") - factor = 16; //set overSample factor 16 = 2^4 - - if (factor != overSamplingFactor) + osFadeBlock = overSample[(int) *osParam]->processSamplesUp (fadeBlock); + + float* fadePtrArray[] = { osFadeBlock.getChannelPointer(0), osFadeBlock.getChannelPointer(1) }; + AudioBuffer<float> osFadeBuffer (fadePtrArray, 2, static_cast<int> (osFadeBlock.getNumSamples())); + + for (int channel = 0; channel < osFadeBuffer.getNumChannels(); ++channel) + { + hProcs[channel].reset(); + + auto* x = osFadeBuffer.getWritePointer (channel); + for (int samp = 0; samp < osFadeBuffer.getNumSamples(); samp++) + { + x[samp] = hProcs[channel].process (x[samp]) * makeup[channel].getCurrentValue(); + } + } + + overSample[(int) *osParam]->processSamplesDown (fadeBlock); + + buffer.applyGainRamp (0, buffer.getNumSamples(), 1.0f, 0.0f); + + for (int ch = 0; ch < buffer.getNumChannels(); ++ch) + buffer.addFromWithRamp (ch, 0, fadeBuffer.getReadPointer (ch), buffer.getNumSamples(), 0.0f, 1.0f); + } + + for (int channel = 0; channel < buffer.getNumChannels(); ++channel) { - overSamplingFactor = factor; - releaseResources(); - prepareToPlay (getSampleRate(), getBlockSize()); + // auto* x = buffer.getWritePointer (channel); + // for (int samp = 0; samp < buffer.getNumSamples(); samp++) + // { + // // x[samp] = dcBlocker[channel].processSample (x[samp]); + // // x[samp] = dcLower[channel].processSample (x[samp]); + // } } -} -void HysteresisProcessor::setBiasFreq (float newFreqKHz) -{ - biasFreq = newFreqKHz * 1000.0f; - updateAngleDelta(); -} + if (isChanging) + { + if (! isOn) //fading in + { + buffer.applyGainRamp (0, buffer.getNumSamples(), 0.0f, 1.0f); -void HysteresisProcessor::setBiasGain (float newGainDB) -{ - biasGain = newGainDB; //Decibels::decibelsToGain (newGainDB); -} + for (int ch = 0; ch < buffer.getNumChannels(); ++ch) + buffer.addFromWithRamp (ch, 0, fadeBuffer.getReadPointer (ch), buffer.getNumSamples(), 1.0f, 0.0f); + } + else if (isOn) //fading out + { + buffer.applyGainRamp (0, buffer.getNumSamples(), 1.0f, 0.0f); -double HysteresisProcessor::getTailLengthSeconds() const -{ - if (overSamplingFactor == 16) - return overSample16->getLatencyInSamples() * getSampleRate(); - else if (overSamplingFactor == 8) - return overSample8->getLatencyInSamples() * getSampleRate(); - else if (overSamplingFactor == 4) - return overSample4->getLatencyInSamples() * getSampleRate(); - else - return overSample2->getLatencyInSamples() * getSampleRate(); + for (int ch = 0; ch < buffer.getNumChannels(); ++ch) + buffer.addFromWithRamp (ch, 0, fadeBuffer.getReadPointer (ch), buffer.getNumSamples(), 0.0f, 1.0f); + } + + isChanging = false; + isOn = ! isOn; + } } diff --git a/Plugin/Source/Processors/Hysteresis/HysteresisProcessor.h b/Plugin/Source/Processors/Hysteresis/HysteresisProcessor.h @@ -1,46 +1,166 @@ #ifndef HYSTERESISPROCESSOR_H_INCLUDED #define HYSTERESISPROCESSOR_H_INCLUDED -#include "../ProcessorBase.h" #include "HysteresisProcessing.h" -class HysteresisProcessor : public ProcessorBase +/* High-pass filter to compensate for low frequency noise of transformer */ +class TransformerHPF { public: - HysteresisProcessor(); + TransformerHPF() {} - void prepareToPlay (double sampleRate, int samplesPerBlock) override; - void releaseResources() override; - void processBlock (AudioBuffer<float>& buffer, MidiBuffer& midiBuffer) override; + void reset (double sampleRate) + { + for (int n = 0; n < 3; ++n) + z[n] = 0.0f; - double getTailLengthSeconds() const override; + fs = (float) sampleRate; + } - void setOverSamplingFactor (String osFactor); - void setBiasFreq (float newFreqKHz); - void setBiasGain (float newGainDB); + void calcCoefs (float fc, float Q) + { + float wc = MathConstants<float>::twoPi * fc / fs; + float c = 1.0f / dsp::FastMathApproximations::tan (wc / 2.0f); + float phi = c * c; + float K = c / Q; + float a0 = phi + K + 1.0f; + + b[0] = phi / a0; + b[1] = -2.0f * b[0]; + b[2] = b[0]; + a[1] = 2.0f * (1.0f - phi) / a0; + a[2] = (phi - K + 1.0f) / a0; + } + + inline float processSample (float x) + { + // direct form II transposed + float y = z[1] + x * b[0]; + + z[1] = z[2] + x*b[1] - y*a[1]; + z[2] = x*b[2] - y*a[2]; + + return y; + } private: - HysteresisProcessing hProcs[2]; - std::unique_ptr<dsp::Oversampling<float>> overSample2; - std::unique_ptr<dsp::Oversampling<float>> overSample4; - std::unique_ptr<dsp::Oversampling<float>> overSample8; - std::unique_ptr<dsp::Oversampling<float>> overSample16; + float a[3] = { 0.0f, 0.0f, 0.0f }; + float b[3] = { 1.0f, 0.0f, 0.0f }; + + float z[3] = { 0.0f, 0.0f, 0.0f }; + + float fs = 44100.0f; + + JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (TransformerHPF) +}; + +/* Low-shelf filter to compensate for low frequency noise of transformer +class TransformerShelf +{ +public: + TransformerShelf() {} + + void reset (double sampleRate) + { + for (int n = 0; n < 3; ++n) + z[n] = 0.0f; + + fs = (float) sampleRate; + } + + void calcCoefs (float fc, float Q, float gain) + { + float A = sqrtf (gain); + float wc = MathConstants<float>::twoPi * fc / fs; + float wS = dsp::FastMathApproximations::sin (wc); + float wC = dsp::FastMathApproximations::cos (wc); + float beta = sqrtf (A) / Q; + + float a0 = ((A+1.0f) + ((A-1.0f) * wC) + (beta*wS)); + + b[0] = A*((A+1.0f) - ((A-1.0f)*wC) + (beta*wS)) / a0; + b[1] = 2.0f*A * ((A-1.0f) - ((A+1.0f)*wC)) / a0; + b[2] = A*((A+1.0f) - ((A-1.0f)*wC) - (beta*wS)) / a0; + + a[1] = -2.0f * ((A-1.0f) + ((A+1.0f)*wC)) / a0; + a[2] = ((A+1.0f) + ((A-1.0f)*wC)-(beta*wS)) / a0; + } + + inline float processSample (float x) + { + // direct form II transposed + float y = z[1] + x * b[0]; - int overSamplingFactor = 8; + z[1] = z[2] + x*b[1] - y*a[1]; + z[2] = x*b[2] - y*a[2]; + return y; + } - ////////////////////////////// - float biasFreq = 0.0f; - float biasGain = 5.0f; - float currentSampleRate = 0.0f; - float angleDelta = 0.0f; - float currentAngle[2] = { 0.0f, 0.0f }; +private: + float a[3] = { 0.0f, 0.0f, 0.0f }; + float b[3] = { 1.0f, 0.0f, 0.0f }; + + float z[3] = { 0.0f, 0.0f, 0.0f }; + + float fs = 44100.0f; + + JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (TransformerShelf) +}; +*/ + +//============================================================= +/* Hysteresis Processor for transformer. */ +class HysteresisProcessor +{ +public: + HysteresisProcessor (AudioProcessorValueTreeState& vts); + + /* Reset fade buffers, filters, and processors. Prepare oversampling */ + void prepareToPlay (double sampleRate, int samplesPerBlock); + + /* Reset oversampling */ + void releaseResources(); + + /* Proceess a buffer. */ + void processBlock (AudioBuffer<float>& buffer, MidiBuffer& midiBuffer); + + void toggleOnOff (bool shouldBeOn); + + static void createParameterLayout (std::vector<std::unique_ptr<RangedAudioParameter>>& params); + + void setDrive (float newDrive); + void setSaturation (float newSat); + void setWidth (float newWidth); + float calcMakeup(); + +private: + float* driveParam = nullptr; + float* satParam = nullptr; + float* widthParam = nullptr; + float* osParam = nullptr; + + SmoothedValue<float, ValueSmoothingTypes::Linear> drive[2]; + SmoothedValue<float, ValueSmoothingTypes::Linear> width[2]; + SmoothedValue<float, ValueSmoothingTypes::Linear> sat[2]; + SmoothedValue<float, ValueSmoothingTypes::Multiplicative> makeup[2]; + + float fs = 44100.0f; + int prevOS = 0; + HysteresisProcessing hProcs[2]; + std::unique_ptr<dsp::Oversampling<float>> overSample[3]; // needs oversampling to avoid aliasing + TransformerHPF dcBlocker[2]; + // TransformerShelf dcLower[2]; - void updateAngleDelta(); + AudioBuffer<float> fadeBuffer; + bool isOn = false; + bool isChanging = true; - //int n[2] = { 0, 0 }; + int overSamplingFactor = 2; + const float dcFreq = 10.0f; + const float dcShelfFreq = 60.0f; - bool fadeIn = false; + int resetCount = 0; JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (HysteresisProcessor) }; diff --git a/Plugin/Source/Processors/Loss Effects/LossEffects.cpp b/Plugin/Source/Processors/Loss Effects/LossEffects.cpp @@ -1,53 +0,0 @@ -#include "LossEffects.h" - -LossEffects::LossEffects() -{} - -void LossEffects::init (float sampleRate, float speed, float spacing, float thickness, float gap) -{ - const auto numBins = sampleRate / (float) order; - const auto binWidth = sampleRate / numBins; - const auto speedMetric = speed * inchesToMeters; - - // Set freq domain multipliers - float H[order]; - for (int k = 0; k < order / 2; k++) - { - const auto freq = ((float) k * binWidth) + (binWidth / 2.0f); - const auto waveNumber = MathConstants<float>::twoPi * freq / speedMetric; - - const auto spacingLoss = expf (-1.0f * waveNumber * spacing); - const auto gapLoss = sinf (waveNumber * gap / 2.0f) / (waveNumber * gap / 2.0f); - const auto thicknessLoss = (1.0f - expf (-waveNumber * thickness)) / (waveNumber * thickness); - - H[k] = spacingLoss * gapLoss * thicknessLoss; - H[order - k - 1] = H[k]; - } - - // Create time domain filter signals - for (int n = 0; n < order; n++) - { - h[n] = 0; - for (int k = 0; k < order; k++) - h[n] += H[k] * cosf (MathConstants<float>::twoPi * (float) k * (float) n / (float) order); - - h[n] /= (float) order; - } - - // Clear xs - xPtr = 0; - for (int n = 0; n < order; n++) - x[n] = 0; -} - -float LossEffects::process (float in) -{ - float y = 0.0f; - x[xPtr]= in; - for (int n = 0; n < order; n++) - y += h[n] * x[negativeAwareModulo<int> (xPtr - n, order)]; - - xPtr = (xPtr + 1) % order; - - return y; -} diff --git a/Plugin/Source/Processors/Loss Effects/LossEffects.h b/Plugin/Source/Processors/Loss Effects/LossEffects.h @@ -1,32 +0,0 @@ -#ifndef LOSSEFFECTS_H_INCLUDED -#define LOSSEFFECTS_H_INCLUDED - -#include "JuceHeader.h" - -namespace -{ - constexpr float inchesToMeters = 0.0254f; - - enum - { - order = 100, - }; -} - -class LossEffects -{ -public: - LossEffects(); - - void init (float sampleRate, float speed, float spacing, float thickness, float gap); - float process (float in); - -private: - float h[order]; - float x[order]; - int xPtr = 0; - - JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (LossEffects) -}; - -#endif //LOSSEFFECTS_H_INCLUDED diff --git a/Plugin/Source/Processors/Loss Effects/LossEffectsFilter.cpp b/Plugin/Source/Processors/Loss Effects/LossEffectsFilter.cpp @@ -1,55 +0,0 @@ -#include "LossEffectsFilter.h" - -void LossEffectsFilter::prepareToPlay (double sampleRate, int samplesPerBlock) -{ - setRateAndBufferSizeDetails (sampleRate, samplesPerBlock); - - lossProcessors[0].init ((float) sampleRate, speed, spacing, thickness, gap); - lossProcessors[1].init ((float) sampleRate, speed, spacing, thickness, gap); -} - -void LossEffectsFilter::releaseResources() -{ -} - -void LossEffectsFilter::processBlock (AudioBuffer<float>& buffer, MidiBuffer& /*midiBuffer*/) -{ - for (int channel = 0; channel < buffer.getNumChannels(); ++channel) - { - auto* x = buffer.getWritePointer (channel); - for (int n = 0; n < buffer.getNumSamples(); n++) - { - x[n] = lossProcessors[channel].process (x[n]); - } - } -} - -void LossEffectsFilter::setSpeed (String newSpeed) -{ - if (newSpeed == "3.75 ips") - speed = 3.75f; - else if (newSpeed == "7.5 ips") - speed = 7.5f; - else if (newSpeed == "15 ips") - speed = 15.0f; - - prepareToPlay (getSampleRate(), getBlockSize()); -} - -void LossEffectsFilter::setSpacing (float newSpacingMicrons) -{ - spacing = newSpacingMicrons * (float) 1e-6; - prepareToPlay (getSampleRate(), getBlockSize()); -} - -void LossEffectsFilter::setThickness (float newThicknessMicrons) -{ - thickness = newThicknessMicrons * (float) 1e-6; - prepareToPlay (getSampleRate(), getBlockSize()); -} - -void LossEffectsFilter::setGap (float newGapMicrons) -{ - gap = newGapMicrons * (float) 1e-6; - prepareToPlay (getSampleRate(), getBlockSize()); -} diff --git a/Plugin/Source/Processors/Loss Effects/LossEffectsFilter.h b/Plugin/Source/Processors/Loss Effects/LossEffectsFilter.h @@ -1,32 +0,0 @@ -#ifndef LOSSEFFECTSFILTER_H_INCLUDED -#define LOSSEFFECTSFILTER_H_INCLUDED - -#include "LossEffects.h" -#include "../ProcessorBase.h" - -class LossEffectsFilter : public ProcessorBase -{ -public: - LossEffectsFilter() : ProcessorBase ("Loss Effects Filter") {} - - void prepareToPlay (double sampleRate, int samplesPerBlock) override; - void releaseResources() override; - void processBlock (AudioBuffer<float>& buffer, MidiBuffer& /*midiBuffer*/) override; - - void setSpeed (String newSpeed); - void setSpacing (float newSpacingMicrons); - void setThickness (float newThicknessMicrons); - void setGap (float newGapMicrons); - -private: - LossEffects lossProcessors[2]; - - float speed = 15.0f; - float spacing = (float) 1e-9; - float gap = (float) 3e-6; - float thickness = (float) 10e-6; - - JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (LossEffectsFilter) -}; - -#endif //LOSSEFFECTSFILTER_H_INCLUDED diff --git a/Plugin/Source/Processors/Loss_Effects/FIRFilter.h b/Plugin/Source/Processors/Loss_Effects/FIRFilter.h @@ -0,0 +1,66 @@ +#ifndef FIRFILTER_H_INCLUDED +#define FIRFILTER_H_INCLUDED + +#include "JuceHeader.h" + +class FIRFilter +{ +public: + FIRFilter (int order) : + order (order) + { + h.reset (new float[order]); + z.reset (new float[order]); + } + ~FIRFilter() {} + + void reset() + { + zPtr = 0; + for (int n = 0; n < order; ++n) + z[n] = 0.0f; + } + + void setCoefs (float* coefs) + { + for (int n = 0; n < order; ++n) + h[n] = coefs[n]; + } + + inline void process (float* buffer, int numSamples) + { + float y = 0.0f; + for (int n = 0; n < numSamples; ++n) + { + y = 0.0f; + z[zPtr] = buffer[n]; + + for (int m = 0; m < order; ++m) + y += h[m] * z[negativeAwareModulo<int> (zPtr - m, order)]; + + buffer[n] = y; + zPtr = (zPtr + 1) % order; + } + } + + inline void processBypassed (float* buffer, int numSamples) + { + for (int n = 0; n < numSamples; ++n) + { + z[zPtr] = buffer[n]; + zPtr = (zPtr + 1) % order; + } + } + +protected: + std::unique_ptr<float[]> h; + const int order; + +private: + std::unique_ptr<float[]> z; + int zPtr = 0; + + JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (FIRFilter) +}; + +#endif //FIRFILTER_H_INCLUDED diff --git a/Plugin/Source/Processors/Loss_Effects/LossFilter.h b/Plugin/Source/Processors/Loss_Effects/LossFilter.h @@ -0,0 +1,167 @@ +#ifndef LOSSFILTER_H_INCLUDED +#define LOSSFILTER_H_INCLUDED + +#include "FIRFilter.h" + +class LossFilter +{ +public: + LossFilter (AudioProcessorValueTreeState& vts) + { + speed = vts.getRawParameterValue ("speed"); + spacing = vts.getRawParameterValue ("spacing"); + thickness = vts.getRawParameterValue ("thick"); + gap = vts.getRawParameterValue ("gap"); + + filters.add (new FIRFilter (order)); + filters.add (new FIRFilter (order)); + + currentCoefs.resize (order); + } + ~LossFilter() {} + + static void createParameterLayout (std::vector<std::unique_ptr<RangedAudioParameter>>& params) + { + NormalisableRange<float> speedRange (1.0f, 100.0f); // meters per second + speedRange.setSkewForCentre (15.0f); + + NormalisableRange<float> spaceRange ((float) 1.0e-9, 0.01f); + spaceRange.setSkewForCentre ((float) 1.0e-6); + + NormalisableRange<float> thickRange ((float) 1.0e-9, 0.0001f); + thickRange.setSkewForCentre (0.0000001f); + + NormalisableRange<float> gapRange ((float) 1.0e-9, 0.01f); + gapRange.setSkewForCentre ((float) 1.0e-6); + + params.push_back (std::make_unique<AudioParameterFloat> ("speed", "Speed", speedRange, 15.0f)); + params.push_back (std::make_unique<AudioParameterFloat> ("spacing", "Spacing", spaceRange, (float) 1.0e-9)); + params.push_back (std::make_unique<AudioParameterFloat> ("thick", "Thickness", thickRange, (float) 1.0e-9)); + params.push_back (std::make_unique<AudioParameterFloat> ("gap", "Gap", gapRange, (float) 1.0e-9)); + } + + void prepare (float sampleRate, int samplesPerBlock) + { + fs = sampleRate; + binWidth = fs / (float) order; + fadeBuffer.resize (samplesPerBlock); + + fsFactor = (int) (fs / 44100.0f); + const int curOrder = order * fsFactor; + filters.clear(); + filters.add (new FIRFilter (curOrder)); + filters.add (new FIRFilter (curOrder)); + currentCoefs.resize (curOrder); + + filters[0]->reset(); + filters[1]->reset(); + + calcCoefs(); + filters[0]->setCoefs (currentCoefs.getRawDataPointer()); + filters[1]->setCoefs (currentCoefs.getRawDataPointer()); + + prevSpeed = *speed; + prevSpacing = *spacing; + prevThickness = *thickness; + prevGap = *gap; + } + + void calcCoefs() + { + // Set freq domain multipliers + int curOrder = order * fsFactor; + std::unique_ptr<float[]> H (new float[curOrder]); + for (int k = 0; k < curOrder / 2; k++) + { + const auto freq = ((float) k * binWidth); // + (binWidth / 2.0f); + const auto waveNumber = MathConstants<float>::twoPi * jmax (freq, 20.0f) / (*speed * 0.0254f); + const auto thickTimesK = waveNumber * *thickness; + const auto kGapOverTwo = waveNumber * *gap / 2.0f; + + H[k] = expf (-1.0f * waveNumber * *spacing); // Spacing loss formula + H[k] *= (1.0f - expf (-thickTimesK)) / thickTimesK; + H[k] *= sinf (kGapOverTwo) / kGapOverTwo; + H[curOrder - k - 1] = H[k]; + } + + // Create time domain filter signals + auto h = currentCoefs.getRawDataPointer(); + for (int n = 0; n < curOrder; n++) + { + for (int k = 0; k < curOrder; k++) + h[n] += H[k] * cosf (MathConstants<float>::twoPi * (float) k * (float) n / (float) curOrder); + + h[n] /= (float) curOrder; + } + } + + inline void processBlock (float* buffer, int numSamples) + { + if ((*speed != prevSpeed || *spacing != prevSpacing || + *thickness != prevThickness || *gap != prevGap) && fadeCount == 0) + { + calcCoefs(); + filters[! activeFilter]->setCoefs (currentCoefs.getRawDataPointer()); + + fadeCount = fadeLength; + prevSpeed = *speed; + prevSpacing = *spacing; + prevThickness = *thickness; + prevGap = *gap; + } + + if (fadeCount > 0) + for (int n = 0; n < numSamples; ++n) + fadeBuffer.setUnchecked (n, buffer[n]); + else + filters[! activeFilter]->processBypassed (buffer, numSamples); + + filters[activeFilter]->process (buffer, numSamples); + + if (fadeCount > 0) + { + filters[! activeFilter]->process (fadeBuffer.getRawDataPointer(), numSamples); + + for (int n = 0; n < numSamples; ++n) + { + float mult = (float) fadeCount / (float) fadeLength; + buffer[n] = buffer[n] * mult + fadeBuffer[n] * (1.0f - mult); + + fadeCount--; + if (fadeCount == 0) + break; + } + + if (fadeCount == 0) + activeFilter = ! activeFilter; + } + } + +private: + OwnedArray<FIRFilter> filters; + int activeFilter = 0; + int fadeCount = 0; + const int fadeLength = 512; + Array<float> fadeBuffer; + + float* speed = nullptr; + float* spacing = nullptr; + float* thickness = nullptr; + float* gap = nullptr; + + float prevSpeed; + float prevSpacing; + float prevThickness; + float prevGap; + + float fs = 44100.0f; + int fsFactor = (int) (fs / 44100.0f); + float binWidth = fs / (float) order; + + const int order = 100; + Array<float> currentCoefs; + + JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (LossFilter) +}; + +#endif //LOSSFILTER_H_INCLUDED diff --git a/Plugin/Source/Processors/ProcessorBase.h b/Plugin/Source/Processors/ProcessorBase.h @@ -1,37 +0,0 @@ -#ifndef PROCESSORBASE_H_INCLUDED -#define PROCESSORBASE_H_INCLUDED - -#include "JuceHeader.h" - -class ProcessorBase : public AudioProcessor -{ -public: - ProcessorBase (String name) : name (name) {} - - const String getName() const override { return name; } - - double getTailLengthSeconds() const override { return 0.0; } - - bool acceptsMidi() const override { return false; } - bool producesMidi() const override { return false; } - - AudioProcessorEditor* createEditor() override { return nullptr; }; - bool hasEditor() const override { return false; } - - int getNumPrograms() override { return 0; } - void setCurrentProgram (int /*index*/) override {} - int getCurrentProgram() override { return 0; } - - const String getProgramName (int /*index*/) override { return {}; } - void changeProgramName (int /*index*/, const String& /*newName*/) override {} - - void getStateInformation (MemoryBlock& /*destData*/) override {} - void setStateInformation (const void* /*data*/, int /*sizeInBytes*/) override {} - -private: - String name; - - JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (ProcessorBase) -}; - -#endif //PROCESSORBASE_H_INCLUDED diff --git a/Plugin/Source/Processors/Timing Effects/TimingEffect.cpp b/Plugin/Source/Processors/Timing Effects/TimingEffect.cpp @@ -1,146 +0,0 @@ -#include "TimingEffect.h" - -void TimingEffect::DelayChannel::setReadPtr (int maxLength) -{ - readPtr++; - if (readPtr >= maxLength) //wrap - readPtr = 0; -} - -TimingEffect::TimingEffect() : ProcessorBase (String ("Timing Processor")) -{ - delayBuffer.setSize (2, maxDelaySamples); -} - -double TimingEffect::getTailLengthSeconds() const -{ - int maxLengthSamples = 0; - for (auto& ch : dChannels) - maxLengthSamples = jmax ((int) ch.length.getTargetValue(), maxLengthSamples); - - return (double) maxLengthSamples / getSampleRate(); -} - -void TimingEffect::setLength (int channel, int lengthSamples) -{ - auto corrLength = roundToInt ((double) lengthSamples * getSampleRate() / fs_calc); - int newLength = jmax (0, jmin (corrLength, (int) maxDelaySamples)); - dChannels[channel].length.setValue ((float) newLength); -} - -int TimingEffect::getLength (int channel) const -{ - return roundToInt (dChannels[channel].length.getTargetValue()); -} - -void TimingEffect::setDepth (float newDepth) -{ - setLength (0, roundToInt (newDepth * dChannels[0].length.getTargetValue() / depth)); - setLength (1, roundToInt (newDepth * dChannels[1].length.getTargetValue() / depth)); - - depth = newDepth; -} - -void TimingEffect::prepareToPlay (double sampleRate, int maximumExpectedSamplesPerBlock) -{ - setRateAndBufferSizeDetails (sampleRate, maximumExpectedSamplesPerBlock); - - delayBuffer.clear(); - - resetDelay(); -} - -void TimingEffect::releaseResources() -{ - delayBuffer.clear(); - - for (auto& ch : dChannels) - ch.resetPtrs(); -} - -void TimingEffect::processBlock (AudioBuffer<float>& buffer, MidiBuffer& /*midiBuffer*/) -{ - countFreq += buffer.getNumSamples(); - - if (n >= periodLen) - n = 0; - - if (countFreq > numSamplesFreq) - { - countFreq = 0; - n++; - - updateDelay(); - } - - for (int channel = 0; channel < buffer.getNumChannels(); channel++) - { - float* x = buffer.getWritePointer(channel); - for (int samp = 0; samp < buffer.getNumSamples(); samp++) - x[samp] = delay (channel, x[samp]); - } -} - -float TimingEffect::delay (int channel, float x) -{ - auto& ch = dChannels[channel]; - - // Erase head - const int erasePtr = negativeAwareModulo(ch.readPtr - 1, (int) maxDelaySamples); - delayBuffer.setSample (channel, erasePtr, 0.0f); - - // Write head - const float len = ch.length.getNextValue(); - const float fractionSample = len - (int) len; - const int writePtr = (ch.readPtr + (int) floorf (len)) % (int) maxDelaySamples; - - delayBuffer.addSample (channel, writePtr, x * (1.0f - fractionSample)); - delayBuffer.addSample (channel, (writePtr + 1) % (int) maxDelaySamples, x * fractionSample); - - // Read head - float y = delayBuffer.getSample (channel, ch.readPtr); - - //update pointers - ch.setReadPtr ((int) maxDelaySamples); - - return y; -} - -void TimingEffect::updateDelay() -{ - float t = 0; - - for (int i = 0; i < polyOrder+1; i++) - t += coefs[i] * (float) pow (n, i); - - setLength (0, roundToInt (depth * t)); - setLength (1, roundToInt (depth * t)); -} - -void TimingEffect::resetDelay() -{ - n = 0; - countFreq = 0; - numSamplesFreq = roundToInt<double> ((1.0f / freq) * getSampleRate()); - - for (auto& ch : dChannels) - ch.length.reset (getSampleRate(), 1.0 / freq); //(1 / freq) seconds to interpolate -} - -void TimingEffect::setTapeSpeed (String tapeSpeed) -{ - auto newFreq = freq; - - if (tapeSpeed == "3.75 ips") - newFreq = 5.0f; - else if (tapeSpeed == "7.5 ips") - newFreq = 10.0f; - else if (tapeSpeed == "15 ips") - newFreq = 20.0f; - - if (newFreq != freq) - { - freq = newFreq; - resetDelay(); - } -} diff --git a/Plugin/Source/Processors/Timing Effects/TimingEffect.h b/Plugin/Source/Processors/Timing Effects/TimingEffect.h @@ -1,63 +0,0 @@ -#ifndef TIMINGEFFECT_H_INCLUDED -#define TIMINGEFFECT_H_INCLUDED - -#include "../ProcessorBase.h" - -class TimingEffect : public ProcessorBase -{ -public: - TimingEffect(); - - void prepareToPlay (double sampleRate, int maxExpectedBlockSize) override; - void releaseResources() override; - void processBlock (AudioBuffer<float>& buffer, MidiBuffer& /*midiBuffer*/) override; - - void setLength (int channel, int length); - int getLength (int channel) const; - - void setDepth (float newDepth); - void setTapeSpeed (String tapeSpeed); - - double getTailLengthSeconds() const override; - -private: - enum - { - maxDelaySamples = 600 *25, - polyOrder = 10, - periodLen = 490, - }; - static constexpr double fs_calc = 44100.0; //Flutter calculations originally done at this sample rate - - struct DelayChannel - { - LinearSmoothedValue<float> length = 0.0f; - int readPtr = 0; - - void resetPtrs() { readPtr = 0; } - void setReadPtr (int maxLength); - }; - - DelayChannel dChannels[2]; - - AudioBuffer<float> delayBuffer; - - float delay (int channel, float x); - void updateDelay(); - void resetDelay(); - - float coefs[polyOrder+1] = { 5.06485313f, 2.45320771f, (float) 5.90645745e-2, - (float) -3.05750415e-4, (float) -1.16380361e-5, (float) 1.95163044e-7, - (float) -1.36154581e-9, (float) 5.11840498e-12, (float) -1.08010214e-14, - (float) 1.20503554e-17, (float) -5.53696539e-21 }; - int n = 0; - int numSamplesFreq = 0; - int countFreq = 0; - - float freq = 10.0f; - float depth = 1.0f; - - JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (TimingEffect) -}; - -#endif //TIMINGEFFECT_H_INCLUDED diff --git a/Plugin/Source/Processors/Timing_Effects/DelayProcessor.cpp b/Plugin/Source/Processors/Timing_Effects/DelayProcessor.cpp @@ -0,0 +1,83 @@ +#include "DelayProcessor.h" + +void DelayProcessor::setReadPtr (int maxLength) +{ + readPtr++; + if (readPtr >= maxLength) //wrap + readPtr = 0; +} + +void DelayProcessor::setLengthMs (double lengthMs, bool force) +{ + int newLength = (int) (lengthMs * sampleRate / 1000.0); + newLength = jmin (newLength, bufferSize); + + if (! force) + length.setTargetValue ((float) newLength); + else + length.setCurrentAndTargetValue ((float) newLength); +} + +float DelayProcessor::getLengthMS () const +{ + return (length.getTargetValue() / sampleRate) * 1000.0f; +} + +void DelayProcessor::setFeedback (float newFeedback) +{ + feedback = jlimit<float> (0.0f, 0.9f, newFeedback); +} + +float DelayProcessor::getFeedback() const +{ + return feedback; +} + +void DelayProcessor::prepareToPlay (double sr, int /*maximumExpectedSamplesPerBlock*/) +{ + sampleRate = (float) sr; + bufferSize = (int) sampleRate * maxDelaySeconds; + delayBuffer.setSize (1, bufferSize); + delayBuffer.clear(); + + length.reset (sampleRate, 5e-3); + resetPtrs(); +} + +void DelayProcessor::process (float* buffer, int numSamples) +{ + for (int n = 0; n < numSamples; n++) + buffer[n] += delay (buffer[n]); +} + +float DelayProcessor::delay (float x) +{ + // Erase head + const int erasePtr = negativeAwareModulo (readPtr - 1, bufferSize); + delayBuffer.setSample (0, erasePtr, 0.0f); + + // Write head + const float len = length.getNextValue(); + + float y = x; + if (len > 0) + { + // Read head + y = delayBuffer.getSample (0, readPtr); + + // write ptr + const float fractionSample = len - (int) len; + const int writePtr = (readPtr + (int) floorf (len)) % bufferSize; + + // feedback + float writeSample = x + y * feedback; + + delayBuffer.addSample (0, writePtr, writeSample * (1.0f - fractionSample)); + delayBuffer.addSample (0, (writePtr + 1) % bufferSize, writeSample * fractionSample); + } + + //update pointers + setReadPtr (bufferSize); + + return y; +} diff --git a/Plugin/Source/Processors/Timing_Effects/DelayProcessor.h b/Plugin/Source/Processors/Timing_Effects/DelayProcessor.h @@ -0,0 +1,43 @@ +#ifndef DELAYPROCESSOR_H_INCLUDED +#define DELAYPROCESSOR_H_INCLUDED + +#include "JuceHeader.h" + +/** Mono Delay */ +class DelayProcessor +{ +public: + DelayProcessor() {} + + void prepareToPlay (double sampleRate, int maxExpectedBlockSize); + void process (float* buffer, int numSamples); + float delay (float x); + + void setLengthMs (double lengthMs, bool force = false); + float getLengthMS() const; + + void setFeedback (float newFeedback); + float getFeedback() const; + +private: + enum + { + maxDelaySeconds = 1, + }; + + SmoothedValue<float, ValueSmoothingTypes::Linear> length = 0.0f; + float feedback = 0.0f; + int readPtr = 0; + + void resetPtrs() { readPtr = 0; } + void setReadPtr (int maxLength); + + float sampleRate = 48000.0f; + + AudioBuffer<float> delayBuffer; + int bufferSize = 0; + + JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (DelayProcessor) +}; + +#endif //DELAYPROCESSOR_H_INCLUDED diff --git a/Plugin/Source/Processors/Timing_Effects/Flutter.cpp b/Plugin/Source/Processors/Timing_Effects/Flutter.cpp @@ -0,0 +1,82 @@ +#include "Flutter.h" + +Flutter::Flutter (AudioProcessorValueTreeState& vts) +{ + rate = vts.getRawParameterValue ("rate"); + depth = vts.getRawParameterValue ("depth"); + + depthSlew[0].setCurrentAndTargetValue (*depth); + depthSlew[1].setCurrentAndTargetValue (*depth); +} + +void Flutter::createParameterLayout (std::vector<std::unique_ptr<RangedAudioParameter>>& params) +{ + params.push_back (std::make_unique<AudioParameterFloat> ("rate", "Rate", 0.0f, 1.0f, 0.3f)); + params.push_back (std::make_unique<AudioParameterFloat> ("depth", "Depth", 0.0f, 1.0f, 0.0f)); +} + +void Flutter::prepareToPlay (double sampleRate, int samplesPerBlock) +{ + fs = (float) sampleRate; + + delay[0].prepareToPlay (sampleRate, samplesPerBlock); + delay[1].prepareToPlay (sampleRate, samplesPerBlock); + + delay[0].setLengthMs (0.f); + delay[1].setLengthMs (0.f); + + depthSlew[0].reset (100); + depthSlew[1].reset (100); + + phase1[0] = 0.0f; + phase1[1] = 0.0f; + phase2[0] = 0.0f; + phase2[1] = 0.0f; + phase3[0] = 0.0f; + phase3[1] = 0.0f; + + amp1 = -230.0f * 1000.0f / (float) sampleRate; + amp2 = -80.0f * 1000.0f / (float) sampleRate; + amp3 = -99.0f * 1000.0f / (float) sampleRate; + dcOffset = 350.0f * 1000.0f / (float) sampleRate; +} + +void Flutter::processBlock (AudioBuffer<float>& buffer, MidiBuffer& /*midiMessages*/) +{ + ScopedNoDenormals noDenormals; + + auto curDepth = powf (*depth * 81.0f / 625.0f, 0.5f); + depthSlew[0].setTargetValue (jmax ((float) 1.0e-12, curDepth)); + depthSlew[1].setTargetValue (jmax ((float) 1.0e-12, curDepth)); + + auto freq = 0.1f * powf (1000.0f, *rate); + auto angleDelta1 = MathConstants<float>::twoPi * 1.0f * freq / fs; + auto angleDelta2 = MathConstants<float>::twoPi * 2.0f * freq / fs; + auto angleDelta3 = MathConstants<float>::twoPi * 3.0f * freq / fs; + + for (int ch = 0; ch < buffer.getNumChannels(); ++ch) + { + auto* x = buffer.getWritePointer (ch); + for (int n = 0; n < buffer.getNumSamples(); ++n) + { + phase1[ch] += angleDelta1; + phase2[ch] += angleDelta2; + phase3[ch] += angleDelta3; + + auto lfo = amp1 * cosf (phase1[ch] + phaseOff1) + + amp2 * cosf (phase2[ch] + phaseOff2) + + amp3 * cosf (phase3[ch] + phaseOff3) + + dcOffset; + + delay[ch].setLengthMs (depthSlew[ch].getNextValue() * lfo); + x[n] = delay[ch].delay (x[n]); + } + + while (phase1[ch] >= MathConstants<float>::twoPi) + phase1[ch] -= MathConstants<float>::twoPi; + while (phase2[ch] >= MathConstants<float>::twoPi) + phase2[ch] -= MathConstants<float>::twoPi; + while (phase2[ch] >= MathConstants<float>::twoPi) + phase2[ch] -= MathConstants<float>::twoPi; + } +} diff --git a/Plugin/Source/Processors/Timing_Effects/Flutter.h b/Plugin/Source/Processors/Timing_Effects/Flutter.h @@ -0,0 +1,43 @@ +#ifndef FLUTTER_H_INCLUDED +#define FLUTTER_H_INCLUDED + +#include "../JuceLibraryCode/JuceHeader.h" +#include "DelayProcessor.h" + +class Flutter +{ +public: + Flutter (AudioProcessorValueTreeState& vts); + + static void createParameterLayout (std::vector<std::unique_ptr<RangedAudioParameter>>& params); + + void prepareToPlay (double sampleRate, int samplesPerBlock); + void processBlock (AudioBuffer<float>&, MidiBuffer&); + +private: + float* rate = nullptr; + float* depth = nullptr; + + float phase1[2] = { 0.0f, 0.0f }; + float phase2[2] = { 0.0f, 0.0f }; + float phase3[2] = { 0.0f, 0.0f }; + + float amp1 = 0.0f; + float amp2 = 0.0f; + float amp3 = 0.0f; + float fs = 48000.0f; + + float dcOffset = 0.0f; + const float phaseOff1 = 0.0f; + const float phaseOff2 = 13.0f * MathConstants<float>::pi / 4.0f; + const float phaseOff3 = -MathConstants<float>::pi / 10.0f; + + SmoothedValue<float, ValueSmoothingTypes::Multiplicative> depthSlew[2]; + + DelayProcessor delay[2]; + + //============================================================================== + JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (Flutter) +}; + +#endif //FLUTTER_H_INCLUDED