-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathdirect.js
108 lines (83 loc) · 2.02 KB
/
direct.js
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
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
/**
* @module audio-generator
*/
'use strict';
var extend = require('xtend/mutable');
var util = require('audio-buffer-utils');
var pcm = require('pcm-util');
module.exports = Generator;
/**
* Sync map constructor
* @constructor
*/
function Generator (fn, opts) {
if (fn instanceof Function) {
if (typeof opts === 'number') {
opts = {duration: opts};
}
else {
opts = opts || {};
}
opts.generate = fn;
}
else {
opts = fn || {};
}
//sort out arguments
opts = extend({
//total duration of a stream
duration: Infinity,
//time repeat period, in seconds, or 1/frequency
period: Infinity,
//inferred from period
//frequency: 0,
/**
* Generate sample value for a time.
* Returns [L, R, ...] or a number for each channel
*
* @param {number} time current time
*/
generate: Math.random
}, pcm.defaults, opts);
//align frequency/period
if (opts.frequency != null) {
opts.period = 1 / opts.frequency;
} else {
opts.frequency = 1 / opts.period;
}
let time = 0, count = 0;
return generate;
//return sync source/map
function generate (buffer) {
if (!buffer) buffer = util.create(opts.samplesPerFrame, opts.channels, opts.sampleRate);
//get audio buffer channels data in array
var data = util.data(buffer);
//enough?
if (time + buffer.length / opts.sampleRate > opts.duration) return null;
//generate [channeled] samples
for (var i = 0; i < buffer.length; i++) {
var moment = time + i / opts.sampleRate;
//rotate by period
if (opts.period !== Infinity) {
moment %= opts.period;
}
var gen = opts.generate(moment);
//treat null as end
if (gen === null) {
return gen;
}
//wrap number value
if (!Array.isArray(gen)) {
gen = [gen || 0];
}
//distribute generated data by channels
for (var channel = 0; channel < buffer.numberOfChannels; channel++) {
data[channel][i] = (gen[channel] == null ? gen[0] : gen[channel]);
}
}
//update counters
count += buffer.length;
time = count / opts.sampleRate;
return buffer;
}
}