kopia lustrzana https://github.com/jameshball/osci-render
Add polygon bitcrush, optimized but readable bulge
rodzic
c052decbe9
commit
d842ee2096
|
@ -12,6 +12,7 @@
|
|||
#include "audio/BitCrushEffect.h"
|
||||
#include "audio/BulgeEffect.h"
|
||||
#include "audio/TwistEffect.h"
|
||||
#include "audio/PolygonBitCrushEffect.h"
|
||||
#include "audio/DistortEffect.h"
|
||||
#include "audio/KaleidoscopeEffect.h"
|
||||
#include "audio/MultiplexEffect.h"
|
||||
|
@ -54,6 +55,7 @@ OscirenderAudioProcessor::OscirenderAudioProcessor() : CommonAudioProcessor(Buse
|
|||
toggleableEffects.push_back(KaleidoscopeEffect(*this).build());
|
||||
toggleableEffects.push_back(BounceEffect().build());
|
||||
toggleableEffects.push_back(TwistEffect().build());
|
||||
toggleableEffects.push_back(PolygonBitCrushEffect().build());
|
||||
#endif
|
||||
|
||||
auto scaleEffect = ScaleEffectApp().build();
|
||||
|
|
|
@ -7,11 +7,10 @@ public:
|
|||
double value = values[0];
|
||||
double translatedBulge = -value + 1;
|
||||
|
||||
double r = sqrt(pow(input.x, 2) + pow(input.y, 2));
|
||||
double theta = atan2(input.y, input.x);
|
||||
double rn = pow(r, translatedBulge);
|
||||
|
||||
return osci::Point(rn * cos(theta), rn * sin(theta), input.z);
|
||||
double r = std::hypot(input.x, input.y);
|
||||
double rn = std::pow(r, translatedBulge);
|
||||
double scale = rn / r;
|
||||
return osci::Point(scale * input.x, scale * input.y, input.z);
|
||||
}
|
||||
|
||||
std::shared_ptr<osci::Effect> build() const override {
|
||||
|
|
|
@ -0,0 +1,52 @@
|
|||
#pragma once
|
||||
#include <JuceHeader.h>
|
||||
#include "../MathUtil.h"
|
||||
|
||||
// Inspired by xenontesla122
|
||||
class PolygonBitCrushEffect : public osci::EffectApplication {
|
||||
public:
|
||||
osci::Point apply(int index, osci::Point input, const std::vector<std::atomic<double>> &values, double sampleRate) override {
|
||||
const double pi = juce::MathConstants<double>::pi;
|
||||
const double twoPi = juce::MathConstants<double>::twoPi;
|
||||
double effectScale = juce::jlimit(0.0, 1.0, values[0].load());
|
||||
double nSides = juce::jmax(2.0, values[1].load());
|
||||
double bandSize = juce::jmax(1e-4, values[2].load());
|
||||
double thetaOffset = values[3] * twoPi;
|
||||
double rOffset = values[4];
|
||||
|
||||
osci::Point output(0);
|
||||
if (input.x != 0 || input.y != 0) {
|
||||
// Note 90 degree rotation: Theta is treated relative to -Y rather than +X
|
||||
double r = std::hypot(input.x, input.y);
|
||||
double theta = std::atan2(-input.x, input.y) - thetaOffset;
|
||||
theta = MathUtil::wrapAngle(theta + pi) - pi; // Move branch cut to +/-pi after thetaOffset is applied
|
||||
double regionTheta = std::round(theta * nSides / twoPi) / nSides * twoPi;
|
||||
double localTheta = theta - regionTheta;
|
||||
double dist = r * std::cos(localTheta);
|
||||
double newDist = juce::jmax(0.0, (std::round(dist / bandSize - rOffset) + rOffset) * bandSize);
|
||||
double scale = newDist / dist;
|
||||
output.x = scale * input.x;
|
||||
output.y = scale * input.y;
|
||||
}
|
||||
// Apply the same stripe quantization to abs(z)
|
||||
if (input.z != 0) {
|
||||
double signZ = input.z > 0 ? 1 : -1;
|
||||
output.z = signZ * juce::jmax(0.0, (std::round(std::abs(input.z) / bandSize - rOffset) + rOffset) * bandSize);
|
||||
}
|
||||
return (1 - effectScale) * input + effectScale * output;
|
||||
}
|
||||
|
||||
std::shared_ptr<osci::Effect> build() const override {
|
||||
auto eff = std::make_shared<osci::Effect>(
|
||||
std::make_shared<PolygonBitCrushEffect>(),
|
||||
std::vector<osci::EffectParameter*>{
|
||||
new osci::EffectParameter("Polygon Bit Crush", "Constrains points to a polygon pattern.", "polygonBitCrushEnable", VERSION_HINT, 1.0, 0.0, 1.0),
|
||||
new osci::EffectParameter("Sides", "Controls the number of sides of the polygon pattern.", "polygonSides", VERSION_HINT, 5.0, 3.0, 8.0),
|
||||
new osci::EffectParameter("Stripe Size", "Controls the spacing between the stripes of the polygon pattern.", "polygonBandSize", VERSION_HINT, 0.15, 0.0, 0.5),
|
||||
new osci::EffectParameter("Angle Offset", "Rotates the polygon pattern.", "polygonAngleOffset", VERSION_HINT, 0.0, 0.0, 1.0, 0.0001, osci::LfoType::Sawtooth, 0.1),
|
||||
new osci::EffectParameter("Radial Offset", "Offsets the stripes of the polygon pattern.", "polygonROffset", VERSION_HINT, 0.0, 0.0, 1.0, 0.0001, osci::LfoType::Sawtooth, 2.0)
|
||||
});
|
||||
eff->setIcon(BinaryData::diamond_svg);
|
||||
return eff;
|
||||
}
|
||||
};
|
|
@ -165,6 +165,8 @@
|
|||
</GROUP>
|
||||
<GROUP id="{75439074-E50C-362F-1EDF-8B4BE9011259}" name="Source">
|
||||
<GROUP id="{85A33213-D880-BD92-70D8-1901DA6D23F0}" name="audio">
|
||||
<FILE id="GJoA2i" name="PolygonBitCrushEffect.h" compile="0" resource="0"
|
||||
file="Source/audio/PolygonBitCrushEffect.h"/>
|
||||
<FILE id="HE3dFE" name="AudioRecorder.h" compile="0" resource="0" file="Source/audio/AudioRecorder.h"/>
|
||||
<FILE id="Bc8UeW" name="BitCrushEffect.h" compile="0" resource="0"
|
||||
file="Source/audio/BitCrushEffect.h"/>
|
||||
|
|
Ładowanie…
Reference in New Issue