paulxstretch/Source/PluginProcessor.h

363 lines
13 KiB
C++

// SPDX-License-Identifier: GPLv3-or-later WITH Appstore-exception
// Copyright (C) 2017 Xenakios
// Copyright (C) 2020 Jesse Chappell
#pragma once
#include "PS_Source/PaulStretchControl.h"
#include "../JuceLibraryCode/JuceHeader.h"
#include "jcdp_envelope.h"
#include <array>
class MyThumbCache;
class AudioFilePreviewComponent;
const int cpi_main_volume = 0;
const int cpi_stretchamount = 1;
const int cpi_fftsize = 2;
const int cpi_pitchshift = 3;
const int cpi_frequencyshift = 4;
const int cpi_soundstart = 5;
const int cpi_soundend = 6;
const int cpi_freeze = 7;
const int cpi_spreadamount = 8;
const int cpi_compress = 9;
const int cpi_loopxfadelen = 10;
const int cpi_numharmonics = 11;
const int cpi_harmonicsfreq = 12;
const int cpi_harmonicsbw = 13;
const int cpi_harmonicsgauss = 14;
const int cpi_octavesm2 = 15;
const int cpi_octavesm1 = 16;
const int cpi_octaves0 = 17;
const int cpi_octaves1 = 18;
const int cpi_octaves15 = 19;
const int cpi_octaves2 = 20;
const int cpi_tonalvsnoisebw = 21;
const int cpi_tonalvsnoisepreserve = 22;
const int cpi_filter_low = 23;
const int cpi_filter_high = 24;
const int cpi_onsetdetection = 25;
const int cpi_capture_trigger = 26;
const int cpi_num_outchans = 27;
const int cpi_pause_enabled = 28;
const int cpi_max_capture_len = 29;
const int cpi_passthrough = 30;
const int cpi_markdirty = 31;
const int cpi_num_inchans = 32;
const int cpi_bypass_stretch = 33;
const int cpi_freefilter_shiftx = 34;
const int cpi_freefilter_shifty = 35;
const int cpi_freefilter_scaley = 36;
const int cpi_freefilter_tilty = 37;
const int cpi_freefilter_randomy_numbands = 38;
const int cpi_freefilter_randomy_rate = 39;
const int cpi_freefilter_randomy_amount = 40;
const int cpi_enable_spec_module0 = 41;
const int cpi_enable_spec_module1 = 42;
const int cpi_enable_spec_module2 = 43;
const int cpi_enable_spec_module3 = 44;
const int cpi_enable_spec_module4 = 45;
const int cpi_enable_spec_module5 = 46;
const int cpi_enable_spec_module6 = 47;
const int cpi_enable_spec_module7 = 48;
const int cpi_enable_spec_module8 = 49;
const int cpi_octaves_extra1 = 50;
const int cpi_octaves_extra2 = 51;
const int cpi_octaves_ratio0 = 52;
const int cpi_octaves_ratio1 = 53;
const int cpi_octaves_ratio2 = 54;
const int cpi_octaves_ratio3 = 55;
const int cpi_octaves_ratio4 = 56;
const int cpi_octaves_ratio5 = 57;
const int cpi_octaves_ratio6 = 58;
const int cpi_octaves_ratio7 = 59;
const int cpi_looping_enabled = 60;
const int cpi_rewind = 61;
const int cpi_dryplayrate = 62;
const int cpi_binauralbeats = 63;
const int cpi_binauralbeats_mono = 64;
const int cpi_binauralbeats_freq = 65;
const int cpi_binauralbeats_mode = 66;
class MyThreadPool : public ThreadPool
{
public:
MyThreadPool() : ThreadPool(2) {}
};
class MyPropertiesFile
{
public:
MyPropertiesFile()
{
PropertiesFile::Options poptions;
poptions.applicationName = "PaulXStretch3";
poptions.folderName = "PaulXStretch3";
poptions.commonToAllUsers = false;
poptions.doNotSave = false;
poptions.storageFormat = PropertiesFile::storeAsXML;
poptions.millisecondsBeforeSaving = 1000;
poptions.ignoreCaseOfKeyNames = false;
poptions.processLock = nullptr;
poptions.filenameSuffix = ".xml";
poptions.osxLibrarySubFolder = "Application Support";
m_props_file = std::make_unique<PropertiesFile>(poptions);
}
std::unique_ptr<PropertiesFile> m_props_file;
};
class PaulstretchpluginAudioProcessorEditor;
struct OfflineRenderParams
{
OfflineRenderParams(File ofile, double osr, int oformat, double omaxdur, int onumloops, CallOutBox* ocb=nullptr, std::function<void(bool,File file)> completion=nullptr) :
outputfile(ofile), outsr(osr), maxoutdur(omaxdur), numloops(onumloops), outputformat(oformat), cbox(ocb), completionHandler(completion)
{}
File outputfile;
double outsr = 44100.0;
double maxoutdur = 3600.0;
int numloops = 1;
int outputformat = 0; // 0=16 bit pcm, 1=24 bit pcm, 2=32 bit float, 3=32 bit float clipped
CallOutBox* cbox = nullptr;
std::function<void(bool,File file)> completionHandler;
};
class PaulstretchpluginAudioProcessor : public AudioProcessor,
public MultiTimer, public VSTCallbackHandler, public AudioProcessorParameter::Listener
{
public:
using EditorType = PaulstretchpluginAudioProcessorEditor;
PaulstretchpluginAudioProcessor(bool is_stand_alone_offline=false);
~PaulstretchpluginAudioProcessor();
//==============================================================================
void prepareToPlay (double sampleRate, int samplesPerBlock) override;
void releaseResources() override;
#ifndef JucePlugin_PreferredChannelConfigurations
bool isBusesLayoutSupported (const BusesLayout& layouts) const override;
#endif
void processBlock (AudioSampleBuffer&, MidiBuffer&) override;
//void processBlock (AudioBuffer<double>&, MidiBuffer&) override;
//bool supportsDoublePrecisionProcessing() const override { return true; }
//==============================================================================
AudioProcessorEditor* createEditor() override;
bool hasEditor() const override;
//==============================================================================
const String getName() const override;
bool acceptsMidi() const override;
bool producesMidi() const override;
bool isMidiEffect () const override;
double getTailLengthSeconds() const override;
//==============================================================================
int getNumPrograms() override;
int getCurrentProgram() override;
void setCurrentProgram (int index) override;
const String getProgramName (int index) override;
void changeProgramName (int index, const String& newName) override;
void parameterValueChanged(int parameterIndex, float newValue) override;
void parameterGestureChanged(int parameterIndex, bool gestureIsStarting) override;
//==============================================================================
void getStateInformation (MemoryBlock& destData) override;
void setStateInformation (const void* data, int sizeInBytes) override;
AudioParameterFloat* getFloatParameter(int index)
{
return dynamic_cast<AudioParameterFloat*>(getParameters()[index]);
}
AudioParameterInt* getIntParameter(int index)
{
return dynamic_cast<AudioParameterInt*>(getParameters()[index]);
}
AudioParameterBool* getBoolParameter(int index)
{
return dynamic_cast<AudioParameterBool*>(getParameters()[index]);
}
AudioParameterChoice* getChoiceParameter(int index)
{
return dynamic_cast<AudioParameterChoice*>(getParameters()[index]);
}
void setLastPluginBounds(juce::Rectangle<int> bounds) { mPluginWindowWidth = bounds.getWidth(); mPluginWindowHeight = bounds.getHeight();}
juce::Rectangle<int> getLastPluginBounds() const { return juce::Rectangle<int>(0,0,mPluginWindowWidth, mPluginWindowHeight); }
void setDirty();
void setInputRecordingEnabled(bool b);
bool isInputRecordingEnabled() { return m_is_recording_pending; }
double getInputRecordingPositionPercent();
enum RecordFileFormat {
FileFormatDefault = 0,
FileFormatAuto,
FileFormatFLAC,
FileFormatWAV,
FileFormatOGG
};
bool startRecordingToFile(File & file, RecordFileFormat fileformat=FileFormatDefault);
bool stopRecordingToFile();
bool isRecordingToFile();
double getElapsedRecordTime() const { return m_elapsedRecordSamples / getSampleRate(); }
String getLastErrorMessage() const { return m_lastError; }
void setDefaultRecordingDirectory(String recdir) {
m_defaultRecordDir = recdir;
}
String getDefaultRecordingDirectory() const { return m_defaultRecordDir; }
RecordFileFormat getDefaultRecordingFormat() const { return m_defaultRecordingFormat; }
void setDefaultRecordingFormat(RecordFileFormat fmt) { m_defaultRecordingFormat = fmt; }
int getDefaultRecordingBitsPerSample() const { return m_defaultRecordingBitsPerSample; }
void setDefaultRecordingBitsPerSample(int fmt) { m_defaultRecordingBitsPerSample = fmt; }
String setAudioFile(const URL& url);
URL getAudioFile() { return m_current_file; }
Range<double> getTimeSelection();
SharedResourcePointer<AudioFormatManager> m_afm;
SharedResourcePointer<MyPropertiesFile> m_propsfile;
StretchAudioSource* getStretchSource() { return m_stretch_source.get(); }
double getPreBufferingPercent();
void timerCallback(int id) override;
double getSampleRateChecked();
int m_abnormal_output_samples = 0;
AudioPlayHead::CurrentPositionInfo m_playposinfo;
bool m_play_when_host_plays = false;
bool m_capture_when_host_plays = false;
bool m_mute_while_capturing = false;
bool m_mute_processed_while_capturing = false;
bool m_use_backgroundbuffering = true;
bool m_restore_playstate = true;
bool m_lastpassthru = false;
bool m_standalone = false;
void resetParameters();
void setPreBufferAmount(int x);
int getPreBufferAmount();
bool m_load_file_with_state = true;
ValueTree getStateTree(bool ignoreoptions, bool ignorefile);
void setStateFromTree(ValueTree tree);
String offlineRender(OfflineRenderParams renderpars);
std::atomic<int> m_offline_render_state{ -1 };
std::atomic<bool> m_offline_render_cancel_requested{ false };
std::atomic<int> m_capture_save_state{ 0 };
bool m_state_dirty = false;
std::unique_ptr<AudioThumbnail> m_thumb;
bool m_show_technical_info = false;
Range<double> m_wave_view_range;
int m_prepare_count = 0;
shared_envelope m_free_filter_envelope;
bool m_import_dlg_open = false;
void setAudioPreview(AudioFilePreviewComponent* afpc);
CriticalSection& getCriticalSection() { return m_cs; }
pointer_sized_int handleVstPluginCanDo(int32 index,
pointer_sized_int value,
void* ptr,
float opt) override;
pointer_sized_int handleVstManufacturerSpecific(int32 index,
pointer_sized_int value,
void* ptr,
float opt) override;
int m_cur_tab_index = 0;
bool m_save_captured_audio = true;
String m_capture_location;
bool m_midinote_control = false;
#if JUCE_IOS
bool m_use_jumpsliders = false;
#else
bool m_use_jumpsliders = true;
#endif
bool m_auto_finish_record = true;
std::function<void(const FileChooser&)> m_filechoose_callback;
private:
static BusesProperties getDefaultLayout();
bool m_prebuffering_inited = false;
AudioBuffer<float> m_recbuffer;
double m_max_reclen = 10.0;
int64 m_rec_pos = 0;
int64 m_rec_count = 0;
int64 m_next_rec_count = 0;
Range<int64> m_recorded_range;
void commitRecording(int lenrecorded);
void finishRecording(int lenrecorded, bool nosave=false);
bool m_using_memory_buffer = true;
int m_cur_num_out_chans = 2;
CriticalSection m_cs;
URL m_current_file;
Time m_current_file_date;
bool m_is_recording = false;
volatile bool m_is_recording_pending = false;
volatile bool m_is_recording_finished = false;
TimeSliceThread m_bufferingthread;
std::unique_ptr<StretchAudioSource> m_stretch_source;
std::unique_ptr<MyBufferingAudioSource> m_buffering_source;
int m_prebuffer_amount = 1;
bool m_recreate_buffering_source = true;
double m_smoothed_prebuffer_ready = 0.0;
SignalSmoother m_prebufsmoother;
int m_fft_size_to_use = 1024;
float m_last_fftsizeparamval = -1.0f;
double m_last_outpos_pos = 0.0;
double m_last_in_pos = 0.0;
std::vector<int> m_bufamounts{ 4096,8192,16384,32768,65536,262144 };
ProcessParameters m_ppar;
BinauralBeatsParameters m_bbpar;
int mPluginWindowWidth = 810;
int mPluginWindowHeight = 745;
void setFFTSize(float size, bool force=false);
void startplay(Range<double> playrange, int numoutchans, int maxBlockSize, String& err);
SharedResourcePointer<MyThumbCache> m_thumbcache;
AudioParameterInt* m_outchansparam = nullptr;
AudioParameterInt* m_inchansparam = nullptr;
int m_curmaxblocksize = 0;
double m_cur_sr = 0.0;
bool m_last_host_playing = false;
AudioBuffer<float> m_input_buffer;
std::vector<float> m_reset_pars;
int m_cur_program = 0;
void setParameters(const std::vector<double>& pars);
float m_cur_playrangeoffset = 0.0;
void updateStretchParametersFromPluginParameters(ProcessParameters& pars,BinauralBeatsParameters & bbpar);
std::array<AudioParameterBool*, 9> m_sm_enab_pars;
bool m_lastrewind = false;
AudioFilePreviewComponent* m_previewcomponent = nullptr;
void saveCaptureBuffer();
SharedResourcePointer<MyThreadPool> m_threadpool;
int m_midinote_to_use = -1;
ADSR m_adsr;
bool m_is_stand_alone_offline = false;
// recording stuff
RecordFileFormat m_defaultRecordingFormat = FileFormatFLAC;
int m_defaultRecordingBitsPerSample = 24;
String m_defaultRecordDir;
String m_defaultCaptureDir;
String m_lastError;
std::atomic<bool> m_writingPossible = { false };
int m_totalRecordingChannels = 2;
int64 m_elapsedRecordSamples = 0;
CriticalSection m_writerLock;
std::unique_ptr<TimeSliceThread> m_recordingThread;
std::unique_ptr<AudioFormatWriter::ThreadedWriter> m_threadedMixWriter;
std::atomic<AudioFormatWriter::ThreadedWriter*> m_activeMixWriter { nullptr };
//==============================================================================
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (PaulstretchpluginAudioProcessor)
};