1
0
Fork 0
mirror of https://github.com/juce-framework/JUCE.git synced 2026-01-17 00:44:19 +00:00
JUCE/examples/BLOCKS/BlocksSynth/Source/Oscillators.h

250 lines
8 KiB
C++

/*
==============================================================================
This file is part of the JUCE library.
Copyright (c) 2017 - ROLI Ltd.
JUCE is an open source library subject to commercial or open-source
licensing.
By using JUCE, you agree to the terms of both the JUCE 5 End-User License
Agreement and JUCE 5 Privacy Policy (both updated and effective as of the
27th April 2017).
End User License Agreement: www.juce.com/juce-5-licence
Privacy Policy: www.juce.com/juce-5-privacy-policy
Or: You may also use this code under the terms of the GPL v3 (see
www.gnu.org/licenses).
JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
DISCLAIMED.
==============================================================================
*/
#pragma once
#include "../JuceLibraryCode/JuceHeader.h"
/**
Base class for oscillators
*/
class Oscillator : public SynthesiserVoice
{
public:
Oscillator()
{
amplitude.reset (getSampleRate(), 0.1);
phaseIncrement.reset (getSampleRate(), 0.1);
}
void startNote (int midiNoteNumber, float velocity, SynthesiserSound*, int) override
{
frequency = MidiMessage::getMidiNoteInHertz (midiNoteNumber);
phaseIncrement.setValue (((2.0 * double_Pi) * frequency) / sampleRate);
amplitude.setValue (velocity);
// Store the initial note and work out the maximum frequency deviations for pitch bend
initialNote = midiNoteNumber;
maxFreq = MidiMessage::getMidiNoteInHertz (initialNote + 4) - frequency;
minFreq = frequency - MidiMessage::getMidiNoteInHertz (initialNote - 4);
}
void stopNote (float, bool) override
{
clearCurrentNote();
amplitude.setValue (0.0);
}
void pitchWheelMoved (int newValue) override
{
// Change the phase increment based on pitch bend amount
auto frequencyOffset = ((newValue > 0 ? maxFreq : minFreq) * (newValue / 127.0));
phaseIncrement.setValue (((2.0 * double_Pi) * (frequency + frequencyOffset)) / sampleRate);
}
void controllerMoved (int, int) override
{
}
void channelPressureChanged (int newChannelPressureValue) override
{
// Set the amplitude based on pressure value
amplitude.setValue (newChannelPressureValue / 127.0);
}
void renderNextBlock (AudioSampleBuffer& outputBuffer, int startSample, int numSamples) override
{
while (--numSamples >= 0)
{
auto output = getSample() * amplitude.getNextValue();
for (auto i = outputBuffer.getNumChannels(); --i >= 0;)
outputBuffer.addSample (i, startSample, static_cast<float> (output));
++startSample;
}
}
/** Returns the next sample */
double getSample()
{
auto output = renderWaveShape (phasePos);
phasePos += phaseIncrement.getNextValue();
if (phasePos > (2.0 * double_Pi))
phasePos -= (2.0 * double_Pi);
return output;
}
/** Subclasses should override this to say whether they can play the given sound */
virtual bool canPlaySound (SynthesiserSound*) override = 0;
/** Subclasses should override this to render a waveshape */
virtual double renderWaveShape (const double currentPhase) = 0;
private:
LinearSmoothedValue<double> amplitude, phaseIncrement;
double frequency = 0;
double phasePos = 0.0f;
double sampleRate = 44100.0;
int initialNote = 0;
double maxFreq = 0, minFreq = 0;
//==============================================================================
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (Oscillator)
};
//==============================================================================
/**
Sine sound struct - applies to MIDI channel 1
*/
struct SineSound : public SynthesiserSound
{
SineSound () {}
bool appliesToNote (int) override { return true; }
bool appliesToChannel (int midiChannel) override { return (midiChannel == 1); }
//==============================================================================
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (SineSound)
};
/**
Sine voice struct that renders a sin waveshape
*/
struct SineVoice : public Oscillator
{
SineVoice() {};
bool canPlaySound (SynthesiserSound* sound) override { return dynamic_cast<SineSound*> (sound) != nullptr; }
double renderWaveShape (const double currentPhase) override { return sin (currentPhase); }
//==============================================================================
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (SineVoice)
};
//==============================================================================
/**
Square sound struct - applies to MIDI channel 2
*/
struct SquareSound : public SynthesiserSound
{
SquareSound() {}
bool appliesToNote (int) override { return true; }
bool appliesToChannel (int midiChannel) override { return (midiChannel == 2); }
//==============================================================================
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (SquareSound)
};
/**
Square voice struct that renders a square waveshape
*/
struct SquareVoice : public Oscillator
{
SquareVoice() {};
bool canPlaySound (SynthesiserSound* sound) override { return dynamic_cast<SquareSound*> (sound) != nullptr; }
double renderWaveShape (const double currentPhase) override { return (currentPhase < double_Pi ? 0.0 : 1.0); }
//==============================================================================
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (SquareVoice)
};
//==============================================================================
/**
Sawtooth sound - applies to MIDI channel 3
*/
struct SawSound : public SynthesiserSound
{
SawSound() {}
bool appliesToNote (int) override { return true; }
bool appliesToChannel (int midiChannel) override { return (midiChannel == 3); }
//==============================================================================
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (SawSound)
};
/**
Sawtooth voice that renders a sawtooth waveshape
*/
struct SawVoice : public Oscillator
{
SawVoice() {}
bool canPlaySound (SynthesiserSound* sound) override { return dynamic_cast<SawSound*> (sound) != nullptr; }
double renderWaveShape (const double currentPhase) override { return (1.0 / double_Pi) * currentPhase - 1.0; }
//==============================================================================
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (SawVoice)
};
//==============================================================================
/**
Triangle sound - applies to MIDI channel 4
*/
struct TriangleSound : public SynthesiserSound
{
TriangleSound() {}
bool appliesToNote (int) override { return true; }
bool appliesToChannel (int midiChannel) override { return (midiChannel == 4); }
//==============================================================================
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (TriangleSound)
};
/**
Triangle voice that renders a triangle waveshape
*/
struct TriangleVoice : public Oscillator
{
TriangleVoice() {}
bool canPlaySound (SynthesiserSound* sound) override { return dynamic_cast<TriangleSound*> (sound) != nullptr; }
double renderWaveShape (const double currentPhase) override
{
return (currentPhase < double_Pi ? -1.0 + (2.0 / double_Pi) * currentPhase
: 3.0 - (2.0 / double_Pi) * currentPhase);
}
//==============================================================================
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (TriangleVoice)
};