/* Copyright (C) 2006-2011 Nasca Octavian Paul Author: Nasca Octavian Paul Copyright (C) 2017 Xenakios This program is free software; you can redistribute it and/or modify it under the terms of version 2 of the GNU General Public License as published by the Free Software Foundation. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License (version 2) for more details. You should have received a copy of the GNU General Public License (version 2) along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #pragma once #include "../JuceLibraryCode/JuceHeader.h" #include "Input/AInputS.h" #include "ProcessedStretch.h" #include #include #include "../WDL/resample.h" class StretchAudioSource final : public PositionableAudioSource { public: StretchAudioSource() {} StretchAudioSource(int initialnumoutchans, AudioFormatManager* afm, std::array& enab_pars); ~StretchAudioSource(); // Inherited via PositionableAudioSource void prepareToPlay(int samplesPerBlockExpected, double sampleRate) override; void releaseResources() override; void getNextAudioBlock(const AudioSourceChannelInfo & bufferToFill) override; void setNextReadPosition(int64 newPosition) override; int64 getNextReadPosition() const override; int64 getTotalLength() const override; bool isLooping() const override; String setAudioFile(File file); File getAudioFile(); AudioBuffer* getSourceAudioBuffer(); void setNumOutChannels(int chans); int getNumOutChannels() { return m_num_outchans; } double getInfilePositionPercent(); double getInfilePositionSeconds(); double getInfileLengthSeconds(); void setRate(double rate); double getRate() { return m_playrate; } void setProcessParameters(ProcessParameters* pars); const ProcessParameters& getProcessParameters(); void setFFTSize(int size); int getFFTSize() { return m_process_fftsize; } double getFreezePos() const { return m_freeze_pos; } void setFreezing(bool b) { m_freezing = b; } bool isFreezing() { return m_freezing; } void setPaused(bool b); bool isPaused() const; void seekPercent(double pos); double getOutputDurationSecondsForRange(Range range, int fftsize); void setOnsetDetection(double x); void setPlayRange(Range playrange, bool isloop); Range getPlayRange() { return m_playrange; } bool isLoopEnabled(); bool hasReachedEnd(); bool isResampling(); std::vector getSpectrumProcessOrder(); void setSpectrumProcessOrder(std::vector order); void setFFTWindowingType(int windowtype); int getFFTWindowingType() { return m_fft_window_type; } std::pair,Range> getFileCachedRangesNormalized(); void setFreeFilterEnvelope(shared_envelope env); ValueTree getStateTree(); void setStateTree(ValueTree state); void setClippingEnabled(bool b) { m_clip_output = b; } bool isLoopingEnabled(); void setLoopingEnabled(bool b); void setMaxLoops(int64_t numloops) { m_maxloops = numloops; } void setAudioBufferAsInputSource(AudioBuffer* buf, int sr, int len); void setMainVolume(double decibels); double getMainVolume() const { return m_main_volume; } //void setSpectralModulesEnabled(const std::array& params); void setSpectralModuleEnabled(int index, bool b); void setLoopXFadeLength(double lenseconds); double getLoopXFadeLengtj() const { return m_loopxfadelen; } void setPreviewDry(bool b); bool isPreviewingDry() const; int m_param_change_count = 0; double getLastSeekPos() const { return m_seekpos; } CriticalSection* getMutex() { return &m_cs; } private: CircularBuffer m_stretchoutringbuf{ 1024 * 1024 }; AudioBuffer m_file_inbuf; LinearSmoothedValue m_vol_smoother; std::unique_ptr m_inputfile; std::vector> m_stretchers; std::function SourceEndedCallback; bool m_firstbuffer = false; bool m_output_has_begun = false; int m_num_outchans = 0; double m_outsr = 44100.0; int m_process_fftsize = 0; int m_fft_window_type = -1; double m_main_volume = 0.0; double m_loopxfadelen = 0.0; ProcessParameters m_ppar; double m_playrate = 1.0; double m_lastplayrate = 0.0; double m_onsetdetection = 0.0; double m_seekpos = 0.0; bool m_freezing = false; int m_pause_state = 0; Range m_playrange{ 0.0,1.0 }; int64_t m_rand_count = 0; bool m_stream_end_reached = false; int64_t m_output_silence_counter = 0; File m_curfile; int64_t m_maxloops = 0; std::unique_ptr m_resampler; std::vector m_resampler_outbuf; CriticalSection m_cs; std::vector m_specproc_order; bool m_stop_play_requested = false; double m_freeze_pos = 0.0; int64_t m_output_counter = 0; int64_t m_output_length = 0; bool m_clip_output = true; void initObjects(); shared_envelope m_free_filter_envelope; AudioFormatManager* m_afm = nullptr; struct { AudioBuffer buffer; int state = 0; // 0 not active, 1 fill xfade buffer, 2 play xfade buffer int xfade_len = 0; int counter = 0; int requested_fft_size = 0; File requested_file; } m_xfadetask; int m_pause_fade_counter = 0; bool m_preview_dry = false; AudioBuffer m_drypreviewbuf; void playDrySound(const AudioSourceChannelInfo & bufferToFill); };