git subrepo clone --branch=sono6good https://github.com/essej/JUCE.git deps/juce
subrepo: subdir: "deps/juce" merged: "b13f9084e" upstream: origin: "https://github.com/essej/JUCE.git" branch: "sono6good" commit: "b13f9084e" git-subrepo: version: "0.4.3" origin: "https://github.com/ingydotnet/git-subrepo.git" commit: "2f68596"
This commit is contained in:
274
deps/juce/examples/GUI/BouncingBallWavetableDemo.h
vendored
Normal file
274
deps/juce/examples/GUI/BouncingBallWavetableDemo.h
vendored
Normal file
@ -0,0 +1,274 @@
|
||||
/*
|
||||
==============================================================================
|
||||
|
||||
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)
|
||||
};
|
Reference in New Issue
Block a user