-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathWaveShaper.tsx
71 lines (63 loc) · 2.44 KB
/
WaveShaper.tsx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
import React, { useCallback, useContext, useMemo, useState } from "react";
import { NodeProps } from "react-flow-renderer";
import { AudioContextContext } from "context/AudioContextContext";
import Node from "components/Node";
import useWaveShaperNode from "hooks/nodes/useWaveShaperNode";
// See: https://developer.mozilla.org/en-US/docs/Web/API/WaveShaperNode#Example
const distortion = `const k = 50;
const samples = curve.length;
const deg = Math.PI / 180;
for (let i = 0; i < samples; i++) {
const x = (i * 2) / samples - 1;
curve[i] = ((3 + k) * x * 20 * deg) / (Math.PI + k * Math.abs(x));
}
return curve;`;
function WaveShaper({ data, id, selected, type }: NodeProps) {
const { onChange, oversample } = data;
const [curveFn, setCurveFn] = useState(data.curveFn ?? distortion);
const [lastValidCurveFn, setLastValidCurveFn] = useState(curveFn);
const context = useContext(AudioContextContext);
const curve = useMemo(() => {
const curve = new Float32Array(context.sampleRate);
// eslint-disable-next-line no-new-func
return new Function("curve", lastValidCurveFn)(curve);
}, [context.sampleRate, lastValidCurveFn]);
// AudioNode
useWaveShaperNode(id, { curve, oversample });
const updateCurve = useCallback(() => {
try {
// eslint-disable-next-line no-new-func
new Function("curve", curveFn)(new Float32Array(context.sampleRate));
setLastValidCurveFn(curveFn);
onChange({ curveFn });
} catch (e) {
console.error(e);
}
}, [context.sampleRate, curveFn, onChange]);
return (
<Node id={id} inputs={["input"]} outputs={["output"]} type={type}>
{selected && (
<div className="customNode_editor nodrag">
<div className="customNode_item" style={{ flexWrap: "wrap" }}>
<textarea
onChange={e => setCurveFn(e.target.value)}
rows={8}
style={{ width: "100%" }}
title="Curve function"
value={curveFn}
/>
<button onClick={updateCurve}>save</button>
</div>
<div className="customNode_item">
<select onChange={e => onChange({ oversample: e.target.value })} title="Oversample" value={oversample}>
<option value="none">none</option>
<option value="2x">2x</option>
<option value="4x">4x</option>
</select>
</div>
</div>
)}
</Node>
);
}
export default React.memo(WaveShaper);