Files
InstaDrums/Source/DrumPad.h
hariel1985 ec9a8b4e23 Add FX chain: per-pad filter, master compressor/EQ/distortion/reverb, VU meter
- Per-pad low-pass filter (cutoff + resonance) applied in DrumPad render
- Master bus FX chain in PluginProcessor::applyMasterFx():
  - Distortion (tanh waveshaper with drive/mix)
  - 3-band EQ (low shelf 200Hz, peak 1kHz, high shelf 5kHz)
  - Compressor (juce::dsp::Compressor with threshold/ratio)
  - Reverb (juce::dsp::Reverb with size/decay)
- Master volume + pan with constant-power pan law
- VU meter: RMS level measurement with exponential smoothing
- All FX panel and master panel knobs now connected to audio processing
- SampleEditorPanel cutoff/reso knobs linked to DrumPad filter params

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-22 11:21:10 +01:00

115 sor
3.2 KiB
C++

#pragma once
#include <JuceHeader.h>
class DrumPad
{
public:
// A single sample with its audio data and source sample rate
struct Sample
{
juce::AudioBuffer<float> buffer;
double sampleRate = 44100.0;
juce::File file;
};
// A velocity layer: velocity range + multiple round-robin samples
struct VelocityLayer
{
float velocityLow = 0.0f; // 0.0 - 1.0
float velocityHigh = 1.0f;
juce::OwnedArray<Sample> samples; // round-robin variations
int nextRoundRobin = 0;
Sample* getNextSample()
{
if (samples.isEmpty()) return nullptr;
auto* s = samples[nextRoundRobin % samples.size()];
nextRoundRobin = (nextRoundRobin + 1) % samples.size();
return s;
}
};
DrumPad();
~DrumPad();
void prepareToPlay (double sampleRate, int samplesPerBlock);
void releaseResources();
// Single sample loading (backwards compatible)
void loadSample (const juce::File& file, juce::AudioFormatManager& formatManager);
// Velocity layer loading from a folder
void loadLayersFromFolder (const juce::File& folder, juce::AudioFormatManager& formatManager);
bool hasSample() const;
void trigger (float velocity = 1.0f);
void stop();
void renderNextBlock (juce::AudioBuffer<float>& outputBuffer, int startSample, int numSamples);
// Pad properties
juce::String name;
int midiNote = 36;
float volume = 1.0f;
float pan = 0.0f;
float pitch = 0.0f;
bool oneShot = true;
int chokeGroup = -1;
juce::Colour colour { 0xff00ff88 };
// ADSR
float attack = 0.001f;
float decay = 0.1f;
float sustain = 1.0f;
float release = 0.05f;
// Per-pad filter
float filterCutoff = 20000.0f; // Hz
float filterReso = 0.707f; // Q
// State
bool isPlaying() const { return playing; }
juce::String getLoadedFileName() const { return loadedFileName; }
juce::File getLoadedFile() const { return loadedFile; }
int getNumLayers() const { return layers.size(); }
const juce::AudioBuffer<float>& getSampleBuffer() const;
private:
// Velocity layers (sorted by velocity range)
juce::OwnedArray<VelocityLayer> layers;
// Currently playing sample reference
Sample* activeSample = nullptr;
// Fallback empty buffer for getSampleBuffer when nothing loaded
juce::AudioBuffer<float> emptyBuffer;
double sampleRate = 44100.0;
double readPosition = 0.0;
bool playing = false;
float currentVelocity = 1.0f;
// Per-pad low-pass filter (stereo)
juce::dsp::IIR::Filter<float> filterL, filterR;
float lastCutoff = 20000.0f;
// ADSR state
enum class EnvelopeStage { Idle, Attack, Decay, Sustain, Release };
EnvelopeStage envStage = EnvelopeStage::Idle;
float envLevel = 0.0f;
juce::String loadedFileName;
juce::File loadedFile;
void advanceEnvelope();
VelocityLayer* findLayerForVelocity (float velocity);
// Parse velocity tag from filename (e.g. "snare_OH_FF_1" -> FF)
static float velocityTagToLow (const juce::String& tag);
static float velocityTagToHigh (const juce::String& tag);
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (DrumPad)
};