@logue/reverb 1.4.5 → 1.4.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/LICENSE CHANGED
@@ -1,6 +1,6 @@
1
1
  MIT License
2
2
 
3
- Copyright (c) 2019-2025 Masashi Yoshikawa <https://logue.dev>
3
+ Copyright (c) 2019-2026 Masashi Yoshikawa <https://logue.dev>
4
4
 
5
5
  Permission is hereby granted, free of charge, to any person obtaining a copy
6
6
  of this software and associated documentation files (the "Software"), to deal
package/README.md CHANGED
@@ -163,4 +163,4 @@ The dependent libraries [@thi.ng/colored-noise](https://www.jsdelivr.com/package
163
163
 
164
164
  ## License
165
165
 
166
- ©2019-2025 by Logue. Licensed under the [MIT License](LICENSE).
166
+ ©2019-2026 by Logue. Licensed under the [MIT License](LICENSE).
package/dist/Meta.d.ts ADDED
@@ -0,0 +1,3 @@
1
+ import { default as MetaInterface } from './interfaces/MetaInterface';
2
+ declare const meta: MetaInterface;
3
+ export default meta;
@@ -0,0 +1,4 @@
1
+ import { ColoredNoiseOpts } from '@thi.ng/colored-noise';
2
+ /** Noise Type */
3
+ export type NoiseType = 'blue' | 'brown' | 'green' | 'pink' | 'red' | 'violet' | 'white';
4
+ export declare const NoiseType: Record<NoiseType, (opts?: Partial<ColoredNoiseOpts>) => Generator<number, void, unknown>>;
@@ -0,0 +1,12 @@
1
+ /**
2
+ * @logue/reverb
3
+ *
4
+ * @description JavaScript Reverb effect class
5
+ * @author Logue <logue@hotmail.co.jp>
6
+ * @copyright 2019-2026 By Masashi Yoshikawa All rights reserved.
7
+ * @license MIT
8
+ * @version 1.4.6
9
+ * @see {@link https://github.com/logue/Reverb.js}
10
+ */
11
+
12
+ "use strict";Object.defineProperty(exports,"__esModule",{value:!0});var s=require("@thi.ng/colored-noise"),m=require("@thi.ng/transducers"),b=require("@thi.ng/random");const p={version:"1.4.6",date:"2026-03-06T12:11:28.691Z"},N={blue:s.blue,brown:s.red,green:s.green,pink:s.pink,red:s.red,violet:s.violet,white:s.white},w={noise:"white",scale:1,peaks:2,randomAlgorithm:b.SYSTEM,decay:2,delay:0,reverse:!1,time:2,filterType:"allpass",filterFreq:2200,filterQ:1,mix:.5,once:!1};class o{static version=p.version;static build=p.date;ctx;wetGainNode;dryGainNode;filterNode;convolverNode;outputNode;options;isConnected;noise=s.white;noiseMap=N;constructor(e,t){this.ctx=e,this.options=Object.assign({},w,t),this.wetGainNode=this.ctx.createGain(),this.dryGainNode=this.ctx.createGain(),this.filterNode=this.ctx.createBiquadFilter(),this.convolverNode=this.ctx.createConvolver(),this.outputNode=this.ctx.createGain(),this.isConnected=!1,this.filterType(this.options.filterType),this.setNoise(this.options.noise),this.buildImpulse(),this.mix(this.options.mix)}connect(e){return this.isConnected&&this.options.once?(this.isConnected=!1,this.outputNode):(this.convolverNode.connect(this.filterNode),this.filterNode.connect(this.wetGainNode),e.connect(this.convolverNode),e.connect(this.dryGainNode),e.connect(this.wetGainNode),this.dryGainNode.connect(this.outputNode),this.wetGainNode.connect(this.outputNode),this.isConnected=!0,this.outputNode)}disconnect(e){return this.isConnected&&(this.convolverNode.disconnect(this.filterNode),this.filterNode.disconnect(this.wetGainNode)),this.isConnected=!1,e}mix(e){if(!o.inRange(e,0,1))throw new RangeError("[Reverb.js] Dry/Wet ratio must be between 0 to 1.");this.options.mix=e,this.dryGainNode.gain.value=1-e,this.wetGainNode.gain.value=e}time(e){if(!o.inRange(e,1,50))throw new RangeError("[Reverb.js] Time length of impulse response must be less than 50sec.");this.options.time=e,this.buildImpulse()}decay(e){if(!o.inRange(e,0,100))throw new RangeError("[Reverb.js] Impulse Response decay level must be less than 100.");this.options.decay=e,this.buildImpulse()}delay(e){if(!o.inRange(e,0,100))throw new RangeError("[Reverb.js] Impulse Response delay time must be less than 100.");this.options.delay=e,this.buildImpulse()}reverse(e){this.options.reverse=e,this.buildImpulse()}filterType(e="allpass"){this.filterNode.type=this.options.filterType=e}filterFreq(e){if(!o.inRange(e,20,2e4))throw new RangeError("[Reverb.js] Filter frequrncy must be between 20 and 20000.");this.options.filterFreq=e,this.filterNode.frequency.value=this.options.filterFreq}filterQ(e){if(!o.inRange(e,0,10))throw new RangeError("[Reverb.js] Filter Q value must be between 0 and 10.");this.options.filterQ=e,this.filterNode.Q.value=this.options.filterQ}peaks(e){this.options.peaks=e,this.buildImpulse()}scale(e){this.options.scale=e,this.buildImpulse()}getNoise(e){return[...m.take(e,this.noise({bins:this.options.peaks,scale:this.options.scale,rnd:this.options.randomAlgorithm}))]}setNoise(e){switch(this.options.noise=e,e){case"blue":this.noise=this.noiseMap.blue;break;case"brown":this.noise=this.noiseMap.brown;break;case"green":this.noise=this.noiseMap.green;break;case"pink":this.noise=this.noiseMap.pink;break;case"red":this.noise=this.noiseMap.red;break;case"violet":this.noise=this.noiseMap.violet;break;case"white":this.noise=this.noiseMap.white;break;default:this.noise=s.white;break}this.buildImpulse()}setRandomAlgorithm(e){this.options.randomAlgorithm=e,this.buildImpulse()}static inRange(e,t,n){return e>=t&&e<=n}buildImpulse(){const e=this.ctx.sampleRate,t=Math.max(e*this.options.time,1),n=e*this.options.delay,l=this.ctx.createBuffer(2,t,e),c=new Float32Array(t),d=new Float32Array(t),r=new Float32Array(1),a=new Float32Array(1),u=this.getNoise(t),f=this.getNoise(t);for(let i=0;i<t;i++){let h;i<n?(r[0]=0,a[0]=0,c.set(r,i),d.set(a,i),h=this.options.reverse??!1?t-(i-n):i-n):h=this.options.reverse??!1?t-i:i,r[0]=(u.at(i)??0)*(1-h/t)**this.options.decay,a[0]=(f.at(i)??0)*(1-h/t)**this.options.decay,c.set(r,i),d.set(a,i)}l.getChannelData(0).set(c),l.getChannelData(1).set(d),this.convolverNode.buffer=l}}exports.default=o;
@@ -0,0 +1,141 @@
1
+ import { default as OptionInterface } from './interfaces/OptionInterface';
2
+ import { INorm } from '@thi.ng/random';
3
+ import { NoiseType } from './NoiseType';
4
+ /**
5
+ * Reverb effect class
6
+ */
7
+ export default class Reverb {
8
+ /** Version strings */
9
+ static readonly version: string;
10
+ /** Build date */
11
+ static readonly build: string;
12
+ /** AudioContext */
13
+ private readonly ctx;
14
+ /** Wet Level (Reverberated node) */
15
+ private readonly wetGainNode;
16
+ /** Dry Level (Original sound node) */
17
+ private readonly dryGainNode;
18
+ /** Impulse response filter */
19
+ private readonly filterNode;
20
+ /** Convolution node for applying impulse response */
21
+ private readonly convolverNode;
22
+ /** Output gain node */
23
+ private readonly outputNode;
24
+ /** Option */
25
+ private readonly options;
26
+ /** Connected flag */
27
+ private isConnected;
28
+ /** Noise Generator */
29
+ private noise;
30
+ /**
31
+ * Map of noise types to their respective generator functions.
32
+ */
33
+ private readonly noiseMap;
34
+ /**
35
+ * Constructor
36
+ *
37
+ * @param ctx - Root AudioContext
38
+ * @param options - Configure
39
+ */
40
+ constructor(ctx: AudioContext, options: Partial<OptionInterface>);
41
+ /**
42
+ * Connect the node for the reverb effect to the original sound node.
43
+ *
44
+ * @param sourceNode - Input source node
45
+ */
46
+ connect(sourceNode: AudioNode): AudioNode;
47
+ /**
48
+ * Disconnect the reverb node
49
+ *
50
+ * @param sourceNode - Input source node
51
+ */
52
+ disconnect(sourceNode?: AudioNode): AudioNode | undefined;
53
+ /**
54
+ * Dry/Wet ratio
55
+ *
56
+ * @param mix - Ratio (0~1)
57
+ */
58
+ mix(mix: number): void;
59
+ /**
60
+ * Set Impulse Response time length (second)
61
+ *
62
+ * @param value - IR length
63
+ */
64
+ time(value: number): void;
65
+ /**
66
+ * Impulse response decay rate.
67
+ *
68
+ * @param value - Decay value
69
+ */
70
+ decay(value: number): void;
71
+ /**
72
+ * Delay before reverberation starts
73
+ *
74
+ * @param value - Time[ms]
75
+ */
76
+ delay(value: number): void;
77
+ /**
78
+ * Reverse the impulse response.
79
+ *
80
+ * @param reverse - Reverse IR
81
+ */
82
+ reverse(reverse: boolean): void;
83
+ /**
84
+ * Filter for impulse response
85
+ *
86
+ * @param type - Filiter Type
87
+ */
88
+ filterType(type?: BiquadFilterType): void;
89
+ /**
90
+ * Filter frequency applied to impulse response
91
+ *
92
+ * @param freq - Frequency
93
+ */
94
+ filterFreq(freq: number): void;
95
+ /**
96
+ * Filter quality.
97
+ *
98
+ * @param q - Quality
99
+ */
100
+ filterQ(q: number): void;
101
+ /**
102
+ * set IR source noise peaks
103
+ *
104
+ * @param p - Peaks
105
+ */
106
+ peaks(p: number): void;
107
+ /**
108
+ * set IR source noise scale.
109
+ *
110
+ * @param s - Scale
111
+ */
112
+ scale(s: number): void;
113
+ /**
114
+ * Noise source
115
+ *
116
+ * @param duration - length of IR.
117
+ */
118
+ private getNoise;
119
+ /**
120
+ * Inpulse Response Noise algorithm.
121
+ *
122
+ * @param type - IR noise algorithm type.
123
+ */
124
+ setNoise(type: NoiseType): void;
125
+ /**
126
+ * Set Random Algorythm
127
+ *
128
+ * @param algorithm - Algorythm
129
+ */
130
+ setRandomAlgorithm(algorithm: INorm): void;
131
+ /**
132
+ * Return true if in range, otherwise false
133
+ *
134
+ * @param x - Target value
135
+ * @param min - Minimum value
136
+ * @param max - Maximum value
137
+ */
138
+ private static inRange;
139
+ /** Utility function for building an impulse response from the module parameters. */
140
+ private buildImpulse;
141
+ }
@@ -0,0 +1,276 @@
1
+ /**
2
+ * @logue/reverb
3
+ *
4
+ * @description JavaScript Reverb effect class
5
+ * @author Logue <logue@hotmail.co.jp>
6
+ * @copyright 2019-2026 By Masashi Yoshikawa All rights reserved.
7
+ * @license MIT
8
+ * @version 1.4.6
9
+ * @see {@link https://github.com/logue/Reverb.js}
10
+ */
11
+
12
+ import { white as d, violet as b, red as p, pink as N, green as w, blue as y } from "@thi.ng/colored-noise";
13
+ import { take as g } from "@thi.ng/transducers";
14
+ import { SYSTEM as v } from "@thi.ng/random";
15
+ const u = {
16
+ version: "1.4.6",
17
+ date: "2026-03-06T12:11:28.691Z"
18
+ }, R = {
19
+ blue: y,
20
+ brown: p,
21
+ green: w,
22
+ pink: N,
23
+ red: p,
24
+ violet: b,
25
+ white: d
26
+ }, k = {
27
+ noise: "white",
28
+ scale: 1,
29
+ peaks: 2,
30
+ randomAlgorithm: v,
31
+ decay: 2,
32
+ delay: 0,
33
+ reverse: !1,
34
+ time: 2,
35
+ filterType: "allpass",
36
+ filterFreq: 2200,
37
+ filterQ: 1,
38
+ mix: 0.5,
39
+ once: !1
40
+ };
41
+ class s {
42
+ /** Version strings */
43
+ static version = u.version;
44
+ /** Build date */
45
+ static build = u.date;
46
+ /** AudioContext */
47
+ ctx;
48
+ /** Wet Level (Reverberated node) */
49
+ wetGainNode;
50
+ /** Dry Level (Original sound node) */
51
+ dryGainNode;
52
+ /** Impulse response filter */
53
+ filterNode;
54
+ /** Convolution node for applying impulse response */
55
+ convolverNode;
56
+ /** Output gain node */
57
+ outputNode;
58
+ /** Option */
59
+ options;
60
+ /** Connected flag */
61
+ isConnected;
62
+ /** Noise Generator */
63
+ noise = d;
64
+ /**
65
+ * Map of noise types to their respective generator functions.
66
+ */
67
+ noiseMap = R;
68
+ /**
69
+ * Constructor
70
+ *
71
+ * @param ctx - Root AudioContext
72
+ * @param options - Configure
73
+ */
74
+ constructor(e, t) {
75
+ this.ctx = e, this.options = Object.assign({}, k, t), this.wetGainNode = this.ctx.createGain(), this.dryGainNode = this.ctx.createGain(), this.filterNode = this.ctx.createBiquadFilter(), this.convolverNode = this.ctx.createConvolver(), this.outputNode = this.ctx.createGain(), this.isConnected = !1, this.filterType(this.options.filterType), this.setNoise(this.options.noise), this.buildImpulse(), this.mix(this.options.mix);
76
+ }
77
+ /**
78
+ * Connect the node for the reverb effect to the original sound node.
79
+ *
80
+ * @param sourceNode - Input source node
81
+ */
82
+ connect(e) {
83
+ return this.isConnected && this.options.once ? (this.isConnected = !1, this.outputNode) : (this.convolverNode.connect(this.filterNode), this.filterNode.connect(this.wetGainNode), e.connect(this.convolverNode), e.connect(this.dryGainNode), e.connect(this.wetGainNode), this.dryGainNode.connect(this.outputNode), this.wetGainNode.connect(this.outputNode), this.isConnected = !0, this.outputNode);
84
+ }
85
+ /**
86
+ * Disconnect the reverb node
87
+ *
88
+ * @param sourceNode - Input source node
89
+ */
90
+ disconnect(e) {
91
+ return this.isConnected && (this.convolverNode.disconnect(this.filterNode), this.filterNode.disconnect(this.wetGainNode)), this.isConnected = !1, e;
92
+ }
93
+ /**
94
+ * Dry/Wet ratio
95
+ *
96
+ * @param mix - Ratio (0~1)
97
+ */
98
+ mix(e) {
99
+ if (!s.inRange(e, 0, 1))
100
+ throw new RangeError("[Reverb.js] Dry/Wet ratio must be between 0 to 1.");
101
+ this.options.mix = e, this.dryGainNode.gain.value = 1 - e, this.wetGainNode.gain.value = e;
102
+ }
103
+ /**
104
+ * Set Impulse Response time length (second)
105
+ *
106
+ * @param value - IR length
107
+ */
108
+ time(e) {
109
+ if (!s.inRange(e, 1, 50))
110
+ throw new RangeError(
111
+ "[Reverb.js] Time length of impulse response must be less than 50sec."
112
+ );
113
+ this.options.time = e, this.buildImpulse();
114
+ }
115
+ /**
116
+ * Impulse response decay rate.
117
+ *
118
+ * @param value - Decay value
119
+ */
120
+ decay(e) {
121
+ if (!s.inRange(e, 0, 100))
122
+ throw new RangeError(
123
+ "[Reverb.js] Impulse Response decay level must be less than 100."
124
+ );
125
+ this.options.decay = e, this.buildImpulse();
126
+ }
127
+ /**
128
+ * Delay before reverberation starts
129
+ *
130
+ * @param value - Time[ms]
131
+ */
132
+ delay(e) {
133
+ if (!s.inRange(e, 0, 100))
134
+ throw new RangeError(
135
+ "[Reverb.js] Impulse Response delay time must be less than 100."
136
+ );
137
+ this.options.delay = e, this.buildImpulse();
138
+ }
139
+ /**
140
+ * Reverse the impulse response.
141
+ *
142
+ * @param reverse - Reverse IR
143
+ */
144
+ reverse(e) {
145
+ this.options.reverse = e, this.buildImpulse();
146
+ }
147
+ /**
148
+ * Filter for impulse response
149
+ *
150
+ * @param type - Filiter Type
151
+ */
152
+ filterType(e = "allpass") {
153
+ this.filterNode.type = this.options.filterType = e;
154
+ }
155
+ /**
156
+ * Filter frequency applied to impulse response
157
+ *
158
+ * @param freq - Frequency
159
+ */
160
+ filterFreq(e) {
161
+ if (!s.inRange(e, 20, 2e4))
162
+ throw new RangeError(
163
+ "[Reverb.js] Filter frequrncy must be between 20 and 20000."
164
+ );
165
+ this.options.filterFreq = e, this.filterNode.frequency.value = this.options.filterFreq;
166
+ }
167
+ /**
168
+ * Filter quality.
169
+ *
170
+ * @param q - Quality
171
+ */
172
+ filterQ(e) {
173
+ if (!s.inRange(e, 0, 10))
174
+ throw new RangeError(
175
+ "[Reverb.js] Filter Q value must be between 0 and 10."
176
+ );
177
+ this.options.filterQ = e, this.filterNode.Q.value = this.options.filterQ;
178
+ }
179
+ /**
180
+ * set IR source noise peaks
181
+ *
182
+ * @param p - Peaks
183
+ */
184
+ peaks(e) {
185
+ this.options.peaks = e, this.buildImpulse();
186
+ }
187
+ /**
188
+ * set IR source noise scale.
189
+ *
190
+ * @param s - Scale
191
+ */
192
+ scale(e) {
193
+ this.options.scale = e, this.buildImpulse();
194
+ }
195
+ /**
196
+ * Noise source
197
+ *
198
+ * @param duration - length of IR.
199
+ */
200
+ getNoise(e) {
201
+ return [
202
+ ...g(
203
+ e,
204
+ this.noise({
205
+ bins: this.options.peaks,
206
+ scale: this.options.scale,
207
+ rnd: this.options.randomAlgorithm
208
+ })
209
+ )
210
+ ];
211
+ }
212
+ /**
213
+ * Inpulse Response Noise algorithm.
214
+ *
215
+ * @param type - IR noise algorithm type.
216
+ */
217
+ setNoise(e) {
218
+ switch (this.options.noise = e, e) {
219
+ case "blue":
220
+ this.noise = this.noiseMap.blue;
221
+ break;
222
+ case "brown":
223
+ this.noise = this.noiseMap.brown;
224
+ break;
225
+ case "green":
226
+ this.noise = this.noiseMap.green;
227
+ break;
228
+ case "pink":
229
+ this.noise = this.noiseMap.pink;
230
+ break;
231
+ case "red":
232
+ this.noise = this.noiseMap.red;
233
+ break;
234
+ case "violet":
235
+ this.noise = this.noiseMap.violet;
236
+ break;
237
+ case "white":
238
+ this.noise = this.noiseMap.white;
239
+ break;
240
+ default:
241
+ this.noise = d;
242
+ break;
243
+ }
244
+ this.buildImpulse();
245
+ }
246
+ /**
247
+ * Set Random Algorythm
248
+ *
249
+ * @param algorithm - Algorythm
250
+ */
251
+ setRandomAlgorithm(e) {
252
+ this.options.randomAlgorithm = e, this.buildImpulse();
253
+ }
254
+ /**
255
+ * Return true if in range, otherwise false
256
+ *
257
+ * @param x - Target value
258
+ * @param min - Minimum value
259
+ * @param max - Maximum value
260
+ */
261
+ static inRange(e, t, o) {
262
+ return e >= t && e <= o;
263
+ }
264
+ /** Utility function for building an impulse response from the module parameters. */
265
+ buildImpulse() {
266
+ const e = this.ctx.sampleRate, t = Math.max(e * this.options.time, 1), o = e * this.options.delay, h = this.ctx.createBuffer(2, t, e), l = new Float32Array(t), c = new Float32Array(t), n = new Float32Array(1), r = new Float32Array(1), f = this.getNoise(t), m = this.getNoise(t);
267
+ for (let i = 0; i < t; i++) {
268
+ let a;
269
+ i < o ? (n[0] = 0, r[0] = 0, l.set(n, i), c.set(r, i), a = this.options.reverse ?? !1 ? t - (i - o) : i - o) : a = this.options.reverse ?? !1 ? t - i : i, n[0] = (f.at(i) ?? 0) * (1 - a / t) ** this.options.decay, r[0] = (m.at(i) ?? 0) * (1 - a / t) ** this.options.decay, l.set(n, i), c.set(r, i);
270
+ }
271
+ h.getChannelData(0).set(l), h.getChannelData(1).set(c), this.convolverNode.buffer = h;
272
+ }
273
+ }
274
+ export {
275
+ s as default
276
+ };
@@ -0,0 +1,12 @@
1
+ /**
2
+ * @logue/reverb
3
+ *
4
+ * @description JavaScript Reverb effect class
5
+ * @author Logue <logue@hotmail.co.jp>
6
+ * @copyright 2019-2026 By Masashi Yoshikawa All rights reserved.
7
+ * @license MIT
8
+ * @version 1.4.6
9
+ * @see {@link https://github.com/logue/Reverb.js}
10
+ */
11
+
12
+ var Reverb=(function(l,s,f,m){"use strict";const u={version:"1.4.6",date:"2026-03-06T12:11:28.691Z"},b={blue:s.blue,brown:s.red,green:s.green,pink:s.pink,red:s.red,violet:s.violet,white:s.white},w={noise:"white",scale:1,peaks:2,randomAlgorithm:m.SYSTEM,decay:2,delay:0,reverse:!1,time:2,filterType:"allpass",filterFreq:2200,filterQ:1,mix:.5,once:!1};class n{static version=u.version;static build=u.date;ctx;wetGainNode;dryGainNode;filterNode;convolverNode;outputNode;options;isConnected;noise=s.white;noiseMap=b;constructor(e,t){this.ctx=e,this.options=Object.assign({},w,t),this.wetGainNode=this.ctx.createGain(),this.dryGainNode=this.ctx.createGain(),this.filterNode=this.ctx.createBiquadFilter(),this.convolverNode=this.ctx.createConvolver(),this.outputNode=this.ctx.createGain(),this.isConnected=!1,this.filterType(this.options.filterType),this.setNoise(this.options.noise),this.buildImpulse(),this.mix(this.options.mix)}connect(e){return this.isConnected&&this.options.once?(this.isConnected=!1,this.outputNode):(this.convolverNode.connect(this.filterNode),this.filterNode.connect(this.wetGainNode),e.connect(this.convolverNode),e.connect(this.dryGainNode),e.connect(this.wetGainNode),this.dryGainNode.connect(this.outputNode),this.wetGainNode.connect(this.outputNode),this.isConnected=!0,this.outputNode)}disconnect(e){return this.isConnected&&(this.convolverNode.disconnect(this.filterNode),this.filterNode.disconnect(this.wetGainNode)),this.isConnected=!1,e}mix(e){if(!n.inRange(e,0,1))throw new RangeError("[Reverb.js] Dry/Wet ratio must be between 0 to 1.");this.options.mix=e,this.dryGainNode.gain.value=1-e,this.wetGainNode.gain.value=e}time(e){if(!n.inRange(e,1,50))throw new RangeError("[Reverb.js] Time length of impulse response must be less than 50sec.");this.options.time=e,this.buildImpulse()}decay(e){if(!n.inRange(e,0,100))throw new RangeError("[Reverb.js] Impulse Response decay level must be less than 100.");this.options.decay=e,this.buildImpulse()}delay(e){if(!n.inRange(e,0,100))throw new RangeError("[Reverb.js] Impulse Response delay time must be less than 100.");this.options.delay=e,this.buildImpulse()}reverse(e){this.options.reverse=e,this.buildImpulse()}filterType(e="allpass"){this.filterNode.type=this.options.filterType=e}filterFreq(e){if(!n.inRange(e,20,2e4))throw new RangeError("[Reverb.js] Filter frequrncy must be between 20 and 20000.");this.options.filterFreq=e,this.filterNode.frequency.value=this.options.filterFreq}filterQ(e){if(!n.inRange(e,0,10))throw new RangeError("[Reverb.js] Filter Q value must be between 0 and 10.");this.options.filterQ=e,this.filterNode.Q.value=this.options.filterQ}peaks(e){this.options.peaks=e,this.buildImpulse()}scale(e){this.options.scale=e,this.buildImpulse()}getNoise(e){return[...f.take(e,this.noise({bins:this.options.peaks,scale:this.options.scale,rnd:this.options.randomAlgorithm}))]}setNoise(e){switch(this.options.noise=e,e){case"blue":this.noise=this.noiseMap.blue;break;case"brown":this.noise=this.noiseMap.brown;break;case"green":this.noise=this.noiseMap.green;break;case"pink":this.noise=this.noiseMap.pink;break;case"red":this.noise=this.noiseMap.red;break;case"violet":this.noise=this.noiseMap.violet;break;case"white":this.noise=this.noiseMap.white;break;default:this.noise=s.white;break}this.buildImpulse()}setRandomAlgorithm(e){this.options.randomAlgorithm=e,this.buildImpulse()}static inRange(e,t,o){return e>=t&&e<=o}buildImpulse(){const e=this.ctx.sampleRate,t=Math.max(e*this.options.time,1),o=e*this.options.delay,c=this.ctx.createBuffer(2,t,e),d=new Float32Array(t),p=new Float32Array(t),a=new Float32Array(1),r=new Float32Array(1),v=this.getNoise(t),y=this.getNoise(t);for(let i=0;i<t;i++){let h;i<o?(a[0]=0,r[0]=0,d.set(a,i),p.set(r,i),h=this.options.reverse??!1?t-(i-o):i-o):h=this.options.reverse??!1?t-i:i,a[0]=(v.at(i)??0)*(1-h/t)**this.options.decay,r[0]=(y.at(i)??0)*(1-h/t)**this.options.decay,d.set(a,i),p.set(r,i)}c.getChannelData(0).set(d),c.getChannelData(1).set(p),this.convolverNode.buffer=c}}return l.default=n,Object.defineProperty(l,"__esModule",{value:!0}),l})({},coloredNoise,transducers,random);
@@ -0,0 +1,12 @@
1
+ /**
2
+ * @logue/reverb
3
+ *
4
+ * @description JavaScript Reverb effect class
5
+ * @author Logue <logue@hotmail.co.jp>
6
+ * @copyright 2019-2026 By Masashi Yoshikawa All rights reserved.
7
+ * @license MIT
8
+ * @version 1.4.6
9
+ * @see {@link https://github.com/logue/Reverb.js}
10
+ */
11
+
12
+ (function(n,i){typeof exports=="object"&&typeof module<"u"?i(exports,require("@thi.ng/colored-noise"),require("@thi.ng/transducers"),require("@thi.ng/random")):typeof define=="function"&&define.amd?define(["exports","@thi.ng/colored-noise","@thi.ng/transducers","@thi.ng/random"],i):(n=typeof globalThis<"u"?globalThis:n||self,i(n.Reverb={},n.coloredNoise,n.transducers,n.random))})(this,(function(n,i,f,m){"use strict";const u={version:"1.4.6",date:"2026-03-06T12:11:28.691Z"},b={blue:i.blue,brown:i.red,green:i.green,pink:i.pink,red:i.red,violet:i.violet,white:i.white},w={noise:"white",scale:1,peaks:2,randomAlgorithm:m.SYSTEM,decay:2,delay:0,reverse:!1,time:2,filterType:"allpass",filterFreq:2200,filterQ:1,mix:.5,once:!1};class o{static version=u.version;static build=u.date;ctx;wetGainNode;dryGainNode;filterNode;convolverNode;outputNode;options;isConnected;noise=i.white;noiseMap=b;constructor(e,t){this.ctx=e,this.options=Object.assign({},w,t),this.wetGainNode=this.ctx.createGain(),this.dryGainNode=this.ctx.createGain(),this.filterNode=this.ctx.createBiquadFilter(),this.convolverNode=this.ctx.createConvolver(),this.outputNode=this.ctx.createGain(),this.isConnected=!1,this.filterType(this.options.filterType),this.setNoise(this.options.noise),this.buildImpulse(),this.mix(this.options.mix)}connect(e){return this.isConnected&&this.options.once?(this.isConnected=!1,this.outputNode):(this.convolverNode.connect(this.filterNode),this.filterNode.connect(this.wetGainNode),e.connect(this.convolverNode),e.connect(this.dryGainNode),e.connect(this.wetGainNode),this.dryGainNode.connect(this.outputNode),this.wetGainNode.connect(this.outputNode),this.isConnected=!0,this.outputNode)}disconnect(e){return this.isConnected&&(this.convolverNode.disconnect(this.filterNode),this.filterNode.disconnect(this.wetGainNode)),this.isConnected=!1,e}mix(e){if(!o.inRange(e,0,1))throw new RangeError("[Reverb.js] Dry/Wet ratio must be between 0 to 1.");this.options.mix=e,this.dryGainNode.gain.value=1-e,this.wetGainNode.gain.value=e}time(e){if(!o.inRange(e,1,50))throw new RangeError("[Reverb.js] Time length of impulse response must be less than 50sec.");this.options.time=e,this.buildImpulse()}decay(e){if(!o.inRange(e,0,100))throw new RangeError("[Reverb.js] Impulse Response decay level must be less than 100.");this.options.decay=e,this.buildImpulse()}delay(e){if(!o.inRange(e,0,100))throw new RangeError("[Reverb.js] Impulse Response delay time must be less than 100.");this.options.delay=e,this.buildImpulse()}reverse(e){this.options.reverse=e,this.buildImpulse()}filterType(e="allpass"){this.filterNode.type=this.options.filterType=e}filterFreq(e){if(!o.inRange(e,20,2e4))throw new RangeError("[Reverb.js] Filter frequrncy must be between 20 and 20000.");this.options.filterFreq=e,this.filterNode.frequency.value=this.options.filterFreq}filterQ(e){if(!o.inRange(e,0,10))throw new RangeError("[Reverb.js] Filter Q value must be between 0 and 10.");this.options.filterQ=e,this.filterNode.Q.value=this.options.filterQ}peaks(e){this.options.peaks=e,this.buildImpulse()}scale(e){this.options.scale=e,this.buildImpulse()}getNoise(e){return[...f.take(e,this.noise({bins:this.options.peaks,scale:this.options.scale,rnd:this.options.randomAlgorithm}))]}setNoise(e){switch(this.options.noise=e,e){case"blue":this.noise=this.noiseMap.blue;break;case"brown":this.noise=this.noiseMap.brown;break;case"green":this.noise=this.noiseMap.green;break;case"pink":this.noise=this.noiseMap.pink;break;case"red":this.noise=this.noiseMap.red;break;case"violet":this.noise=this.noiseMap.violet;break;case"white":this.noise=this.noiseMap.white;break;default:this.noise=i.white;break}this.buildImpulse()}setRandomAlgorithm(e){this.options.randomAlgorithm=e,this.buildImpulse()}static inRange(e,t,r){return e>=t&&e<=r}buildImpulse(){const e=this.ctx.sampleRate,t=Math.max(e*this.options.time,1),r=e*this.options.delay,d=this.ctx.createBuffer(2,t,e),c=new Float32Array(t),p=new Float32Array(t),a=new Float32Array(1),h=new Float32Array(1),g=this.getNoise(t),y=this.getNoise(t);for(let s=0;s<t;s++){let l;s<r?(a[0]=0,h[0]=0,c.set(a,s),p.set(h,s),l=this.options.reverse??!1?t-(s-r):s-r):l=this.options.reverse??!1?t-s:s,a[0]=(g.at(s)??0)*(1-l/t)**this.options.decay,h[0]=(y.at(s)??0)*(1-l/t)**this.options.decay,c.set(a,s),p.set(h,s)}d.getChannelData(0).set(c),d.getChannelData(1).set(p),this.convolverNode.buffer=d}}n.default=o,Object.defineProperty(n,"__esModule",{value:!0})}));
package/dist/demo.flac ADDED
Binary file
@@ -0,0 +1,7 @@
1
+ /** Meta Information Interface */
2
+ export default interface MetaInterface {
3
+ /** Version */
4
+ version: string;
5
+ /** Build Date */
6
+ date: string;
7
+ }
@@ -0,0 +1,39 @@
1
+ import { NoiseType } from '../NoiseType';
2
+ import { INorm } from '@thi.ng/random';
3
+ /** Reverb Option */
4
+ export default interface OptionInterface {
5
+ /**
6
+ * IR (Inpulse Response) colord noise algorithm (BLUE, GREEN, PINK, RED, VIOLET, WHITE)
7
+ * @see {@link https://github.com/thi-ng/umbrella/tree/develop/packages/colored-noise}
8
+ */
9
+ noise: NoiseType;
10
+ /** IR source noise scale */
11
+ scale: number;
12
+ /** Number of IR source noise peaks */
13
+ peaks: number;
14
+ /**
15
+ * Randam noise algorythm
16
+ * @see {@link https://github.com/thi-ng/umbrella/tree/develop/packages/random}
17
+ */
18
+ randomAlgorithm: INorm;
19
+ /** Decay */
20
+ decay: number;
21
+ /** Delay until impulse response is generated */
22
+ delay: number;
23
+ /** Filter frequency applied to impulse response[Hz] */
24
+ filterFreq: number;
25
+ /** Filter quality for impulse response */
26
+ filterQ: number;
27
+ /** Filter type for impulse response */
28
+ filterType: BiquadFilterType;
29
+ /** Dry/Wet ratio */
30
+ mix: number;
31
+ /** Invert the impulse response */
32
+ reverse?: boolean;
33
+ /** Impulse response length */
34
+ time: number;
35
+ /** Prevents multiple effectors from being connected. */
36
+ once: boolean;
37
+ }
38
+ /** Default Value */
39
+ export declare const defaults: OptionInterface;
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "$schema": "https://json.schemastore.org/package.json",
3
3
  "name": "@logue/reverb",
4
- "version": "1.4.5",
4
+ "version": "1.4.6",
5
5
  "description": "JavaScript Reverb effect class",
6
6
  "keywords": [
7
7
  "webaudio",
@@ -50,7 +50,7 @@
50
50
  "node": "^20.19.0 || >=22.12.0",
51
51
  "pnpm": ">=10.2.0"
52
52
  },
53
- "packageManager": "pnpm@10.29.3",
53
+ "packageManager": "pnpm@10.30.3",
54
54
  "sideEffects": false,
55
55
  "scripts": {
56
56
  "dev": "vite",
@@ -69,7 +69,8 @@
69
69
  "prepare": "husky",
70
70
  "test": "vitest",
71
71
  "test:run": "vitest run",
72
- "test:coverage": "vitest run --coverage"
72
+ "test:coverage": "vitest run --coverage",
73
+ "version": "auto-changelog -p && git add CHANGELOG.md"
73
74
  },
74
75
  "peerDependencies": {
75
76
  "@thi.ng/colored-noise": "^1.0.103",
@@ -77,30 +78,31 @@
77
78
  "@thi.ng/transducers": "^9.5.1"
78
79
  },
79
80
  "devDependencies": {
80
- "@eslint/js": "^9.17.0",
81
+ "@eslint/js": "^10.0.1",
81
82
  "@tsconfig/node-lts": "^24.0.0",
82
- "@types/node": "^25.2.3",
83
- "@typescript-eslint/eslint-plugin": "^8.56.0",
83
+ "@types/node": "^25.3.5",
84
+ "@typescript-eslint/eslint-plugin": "^8.56.1",
84
85
  "@vitest/coverage-v8": "^4.0.18",
85
86
  "bootstrap": "^5.3.8",
86
- "eslint": "^9.17.0",
87
+ "eslint": "^10.0.2",
87
88
  "eslint-config-prettier": "^10.1.8",
88
89
  "eslint-import-resolver-custom-alias": "^1.3.2",
89
90
  "eslint-import-resolver-typescript": "^4.4.4",
90
- "eslint-plugin-import": "^2.32.0",
91
- "eslint-plugin-oxlint": "^1.46.0",
92
- "globals": "^17.3.0",
93
- "happy-dom": "^20.6.1",
91
+ "eslint-plugin-import-x": "^4.16.1",
92
+ "eslint-plugin-oxlint": "^1.51.0",
93
+ "eslint-plugin-security": "^4.0.0",
94
+ "globals": "^17.4.0",
95
+ "happy-dom": "^20.8.3",
94
96
  "husky": "^9.1.7",
95
97
  "jiti": "^2.6.1",
96
- "lint-staged": "^16.2.7",
98
+ "lint-staged": "^16.3.2",
97
99
  "npm-run-all2": "^8.0.4",
98
- "oxlint": "^1.47.0",
100
+ "oxlint": "^1.51.0",
99
101
  "prettier": "^3.8.1",
100
102
  "rimraf": "^6.1.3",
101
- "rollup-plugin-visualizer": "^6.0.5",
103
+ "rollup-plugin-visualizer": "^7.0.1",
102
104
  "typescript": "^5.9.3",
103
- "typescript-eslint": "^8.56.0",
105
+ "typescript-eslint": "^8.56.1",
104
106
  "vite": "^7.3.1",
105
107
  "vite-plugin-banner": "^0.8.1",
106
108
  "vite-plugin-checker": "^0.12.0",
@@ -118,6 +120,7 @@
118
120
  },
119
121
  "resolutions": {
120
122
  "json5": ">=2.2.3",
121
- "yaml": ">=2.3.2"
123
+ "minimatch": ">=10.2.3",
124
+ "yaml": ">=2.6.0"
122
125
  }
123
- }
126
+ }