paulxstretch/deps/juce/examples/GUI/BouncingBallWavetableDemo.h

275 lines
8.6 KiB
C
Raw Normal View History

/*
==============================================================================
This file is part of the JUCE examples.
Copyright (c) 2020 - Raw Material Software Limited
The code included in this file is provided under the terms of the ISC license
http://www.isc.org/downloads/software-support-policy/isc-license. Permission
To use, copy, modify, and/or distribute this software for any purpose with or
without fee is hereby granted provided that the above copyright notice and
this permission notice appear in all copies.
THE SOFTWARE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES,
WHETHER EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR
PURPOSE, ARE DISCLAIMED.
==============================================================================
*/
/*******************************************************************************
The block below describes the properties of this PIP. A PIP is a short snippet
of code that can be read by the Projucer and used to generate a JUCE project.
BEGIN_JUCE_PIP_METADATA
name: BouncingBallWavetableDemo
version: 1.0.0
vendor: JUCE
website: http://juce.com
description: Wavetable synthesis with a bouncing ball.
dependencies: juce_audio_basics, juce_audio_devices, juce_audio_formats,
juce_audio_processors, juce_audio_utils, juce_core,
juce_data_structures, juce_events, juce_graphics,
juce_gui_basics, juce_gui_extra
exporters: xcode_mac, vs2019, linux_make
moduleFlags: JUCE_STRICT_REFCOUNTEDPOINTER=1
type: Component
mainClass: BouncingBallWavetableDemo
useLocalCopy: 1
END_JUCE_PIP_METADATA
*******************************************************************************/
#pragma once
//==============================================================================
class BouncingBallWavetableDemo : public AudioAppComponent,
private Timer
{
public:
//==============================================================================
BouncingBallWavetableDemo()
#ifdef JUCE_DEMO_RUNNER
: AudioAppComponent (getSharedAudioDeviceManager (0, 2))
#endif
{
setSize (600, 600);
for (auto i = 0; i < numElementsInArray (waveValues); ++i)
zeromem (waveValues[i], sizeof (waveValues[i]));
// specify the number of input and output channels that we want to open
setAudioChannels (2, 2);
startTimerHz (60);
}
~BouncingBallWavetableDemo() override
{
shutdownAudio();
}
//==============================================================================
void prepareToPlay (int samplesPerBlockExpected, double newSampleRate) override
{
sampleRate = newSampleRate;
expectedSamplesPerBlock = samplesPerBlockExpected;
}
/* This method generates the actual audio samples.
In this example the buffer is filled with a sine wave whose frequency and
amplitude are controlled by the mouse position.
*/
void getNextAudioBlock (const AudioSourceChannelInfo& bufferToFill) override
{
bufferToFill.clearActiveBufferRegion();
for (auto chan = 0; chan < bufferToFill.buffer->getNumChannels(); ++chan)
{
auto ind = waveTableIndex;
auto* channelData = bufferToFill.buffer->getWritePointer (chan, bufferToFill.startSample);
for (auto i = 0; i < bufferToFill.numSamples; ++i)
{
if (isPositiveAndBelow (chan, numElementsInArray (waveValues)))
{
channelData[i] = waveValues[chan][ind % wavetableSize];
++ind;
}
}
}
waveTableIndex = (int) (waveTableIndex + bufferToFill.numSamples) % wavetableSize;
}
void releaseResources() override
{
// This gets automatically called when audio device parameters change
// or device is restarted.
stopTimer();
}
//==============================================================================
void paint (Graphics& g) override
{
// (Our component is opaque, so we must completely fill the background with a solid colour)
g.fillAll (getLookAndFeel().findColour (ResizableWindow::backgroundColourId));
auto nextPos = pos + delta;
if (nextPos.x < 10 || nextPos.x + 10 > (float) getWidth())
{
delta.x = -delta.x;
nextPos.x = pos.x + delta.x;
}
if (nextPos.y < 50 || nextPos.y + 10 > (float) getHeight())
{
delta.y = -delta.y;
nextPos.y = pos.y + delta.y;
}
if (! dragging)
{
writeInterpolatedValue (pos, nextPos);
pos = nextPos;
}
else
{
pos = lastMousePosition;
}
// draw a circle
g.setColour (getLookAndFeel().findColour (Slider::thumbColourId));
g.fillEllipse (pos.x, pos.y, 20, 20);
drawWaveform (g, 20.0f, 0);
drawWaveform (g, 40.0f, 1);
}
void drawWaveform (Graphics& g, float y, int channel) const
{
auto pathWidth = 2000;
Path wavePath;
wavePath.startNewSubPath (0.0f, y);
for (auto i = 1; i < pathWidth; ++i)
wavePath.lineTo ((float) i, (1.0f + waveValues[channel][i * numElementsInArray (waveValues[0]) / pathWidth]) * 10.0f);
g.strokePath (wavePath, PathStrokeType (1.0f),
wavePath.getTransformToScaleToFit (Rectangle<float> (0.0f, y, (float) getWidth(), 20.0f), false));
}
// Mouse handling..
void mouseDown (const MouseEvent& e) override
{
lastMousePosition = e.position;
mouseDrag (e);
dragging = true;
}
void mouseDrag (const MouseEvent& e) override
{
dragging = true;
if (e.position != lastMousePosition)
{
// calculate movement vector
delta = e.position - lastMousePosition;
waveValues[0][bufferIndex % wavetableSize] = xToAmplitude (e.position.x);
waveValues[1][bufferIndex % wavetableSize] = yToAmplitude (e.position.y);
++bufferIndex;
lastMousePosition = e.position;
}
}
void mouseUp (const MouseEvent&) override
{
dragging = false;
}
void writeInterpolatedValue (Point<float> lastPosition,
Point<float> currentPosition)
{
Point<float> start, finish;
if (lastPosition.getX() > currentPosition.getX())
{
finish = lastPosition;
start = currentPosition;
}
else
{
start = lastPosition;
finish = currentPosition;
}
for (auto i = 0; i < steps; ++i)
{
auto p = start + ((finish - start) * i) / (int) steps;
auto index = (bufferIndex + i) % wavetableSize;
waveValues[1][index] = yToAmplitude (p.y);
waveValues[0][index] = xToAmplitude (p.x);
}
bufferIndex = (bufferIndex + steps) % wavetableSize;
}
float indexToX (int indexValue) const noexcept
{
return (float) indexValue;
}
float amplitudeToY (float amp) const noexcept
{
return (float) getHeight() - (amp + 1.0f) * (float) getHeight() / 2.0f;
}
float xToAmplitude (float x) const noexcept
{
return jlimit (-1.0f, 1.0f, 2.0f * ((float) getWidth() - x) / (float) getWidth() - 1.0f);
}
float yToAmplitude (float y) const noexcept
{
return jlimit (-1.0f, 1.0f, 2.0f * ((float) getHeight() - y) / (float) getHeight() - 1.0f);
}
void timerCallback() override
{
repaint();
}
private:
//==============================================================================
enum
{
wavetableSize = 36000,
steps = 10
};
Point<float> pos = { 299.0f, 299.0f };
Point<float> delta = { 0.0f, 0.0f };
int waveTableIndex = 0;
int bufferIndex = 0;
double sampleRate = 0.0;
int expectedSamplesPerBlock = 0;
Point<float> lastMousePosition;
float waveValues[2][wavetableSize];
bool dragging = false;
JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (BouncingBallWavetableDemo)
};