mirror of
https://github.com/Tonejs/Tone.js
synced 2025-01-23 17:15:01 +00:00
324 lines
8.3 KiB
JavaScript
324 lines
8.3 KiB
JavaScript
define(["Tone/core/Tone", "Tone/component/CrossFade", "Tone/component/Merge", "Tone/component/Split",
|
|
"Tone/signal/Signal", "Tone/signal/AudioToGain", "Tone/signal/Zero", "Tone/core/AudioNode"], function(Tone){
|
|
|
|
"use strict";
|
|
|
|
/**
|
|
* @class A spatialized panner node which supports equalpower or HRTF panning.
|
|
* Tries to normalize the API across various browsers. See Tone.Listener
|
|
*
|
|
* @constructor
|
|
* @extends {Tone.AudioNode}
|
|
* @param {Number} positionX The initial x position.
|
|
* @param {Number} positionY The initial y position.
|
|
* @param {Number} positionZ The initial z position.
|
|
*/
|
|
Tone.Panner3D = function(){
|
|
|
|
var options = Tone.defaults(arguments, ["positionX", "positionY", "positionZ"], Tone.Panner3D);
|
|
Tone.AudioNode.call(this);
|
|
|
|
/**
|
|
* The panner node
|
|
* @type {PannerNode}
|
|
* @private
|
|
*/
|
|
this._panner = this.input = this.output = this.context.createPanner();
|
|
//set some values
|
|
this._panner.panningModel = options.panningModel;
|
|
this._panner.maxDistance = options.maxDistance;
|
|
this._panner.distanceModel = options.distanceModel;
|
|
this._panner.coneOuterGain = options.coneOuterGain;
|
|
this._panner.coneOuterAngle = options.coneOuterAngle;
|
|
this._panner.coneInnerAngle = options.coneInnerAngle;
|
|
this._panner.refDistance = options.refDistance;
|
|
this._panner.rolloffFactor = options.rolloffFactor;
|
|
|
|
/**
|
|
* Holds the current orientation
|
|
* @type {Array}
|
|
* @private
|
|
*/
|
|
this._orientation = [options.orientationX, options.orientationY, options.orientationZ];
|
|
|
|
/**
|
|
* Holds the current position
|
|
* @type {Array}
|
|
* @private
|
|
*/
|
|
this._position = [options.positionX, options.positionY, options.positionZ];
|
|
|
|
// set the default position/orientation
|
|
this.orientationX = options.orientationX;
|
|
this.orientationY = options.orientationY;
|
|
this.orientationZ = options.orientationZ;
|
|
this.positionX = options.positionX;
|
|
this.positionY = options.positionY;
|
|
this.positionZ = options.positionZ;
|
|
};
|
|
|
|
Tone.extend(Tone.Panner3D, Tone.AudioNode);
|
|
|
|
/**
|
|
* Defaults according to the specification
|
|
* @static
|
|
* @const
|
|
* @type {Object}
|
|
*/
|
|
Tone.Panner3D.defaults = {
|
|
"positionX" : 0,
|
|
"positionY" : 0,
|
|
"positionZ" : 0,
|
|
"orientationX" : 0,
|
|
"orientationY" : 0,
|
|
"orientationZ" : 0,
|
|
"panningModel" : "equalpower",
|
|
"maxDistance" : 10000,
|
|
"distanceModel" : "inverse",
|
|
"coneOuterGain" : 0,
|
|
"coneOuterAngle" : 360,
|
|
"coneInnerAngle" : 360,
|
|
"refDistance" : 1,
|
|
"rolloffFactor" : 1
|
|
};
|
|
|
|
/**
|
|
* The ramp time which is applied to the setTargetAtTime
|
|
* @type {Number}
|
|
* @private
|
|
*/
|
|
Tone.Panner3D.prototype._rampTimeConstant = 0.01;
|
|
|
|
/**
|
|
* Sets the position of the source in 3d space.
|
|
* @param {Number} x
|
|
* @param {Number} y
|
|
* @param {Number} z
|
|
* @return {Tone.Panner3D} this
|
|
*/
|
|
Tone.Panner3D.prototype.setPosition = function(x, y, z){
|
|
if (this._panner.positionX){
|
|
var now = this.now();
|
|
this._panner.positionX.setTargetAtTime(x, now, this._rampTimeConstant);
|
|
this._panner.positionY.setTargetAtTime(y, now, this._rampTimeConstant);
|
|
this._panner.positionZ.setTargetAtTime(z, now, this._rampTimeConstant);
|
|
} else {
|
|
this._panner.setPosition(x, y, z);
|
|
}
|
|
this._position = Array.prototype.slice.call(arguments);
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Sets the orientation of the source in 3d space.
|
|
* @param {Number} x
|
|
* @param {Number} y
|
|
* @param {Number} z
|
|
* @return {Tone.Panner3D} this
|
|
*/
|
|
Tone.Panner3D.prototype.setOrientation = function(x, y, z){
|
|
if (this._panner.orientationX){
|
|
var now = this.now();
|
|
this._panner.orientationX.setTargetAtTime(x, now, this._rampTimeConstant);
|
|
this._panner.orientationY.setTargetAtTime(y, now, this._rampTimeConstant);
|
|
this._panner.orientationZ.setTargetAtTime(z, now, this._rampTimeConstant);
|
|
} else {
|
|
this._panner.setOrientation(x, y, z);
|
|
}
|
|
this._orientation = Array.prototype.slice.call(arguments);
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* The x position of the panner object.
|
|
* @type {Number}
|
|
* @memberOf Tone.Panner3D#
|
|
* @name positionX
|
|
*/
|
|
Object.defineProperty(Tone.Panner3D.prototype, "positionX", {
|
|
set : function(pos){
|
|
this._position[0] = pos;
|
|
this.setPosition.apply(this, this._position);
|
|
},
|
|
get : function(){
|
|
return this._position[0];
|
|
}
|
|
});
|
|
|
|
/**
|
|
* The y position of the panner object.
|
|
* @type {Number}
|
|
* @memberOf Tone.Panner3D#
|
|
* @name positionY
|
|
*/
|
|
Object.defineProperty(Tone.Panner3D.prototype, "positionY", {
|
|
set : function(pos){
|
|
this._position[1] = pos;
|
|
this.setPosition.apply(this, this._position);
|
|
},
|
|
get : function(){
|
|
return this._position[1];
|
|
}
|
|
});
|
|
|
|
/**
|
|
* The z position of the panner object.
|
|
* @type {Number}
|
|
* @memberOf Tone.Panner3D#
|
|
* @name positionZ
|
|
*/
|
|
Object.defineProperty(Tone.Panner3D.prototype, "positionZ", {
|
|
set : function(pos){
|
|
this._position[2] = pos;
|
|
this.setPosition.apply(this, this._position);
|
|
},
|
|
get : function(){
|
|
return this._position[2];
|
|
}
|
|
});
|
|
|
|
/**
|
|
* The x orientation of the panner object.
|
|
* @type {Number}
|
|
* @memberOf Tone.Panner3D#
|
|
* @name orientationX
|
|
*/
|
|
Object.defineProperty(Tone.Panner3D.prototype, "orientationX", {
|
|
set : function(pos){
|
|
this._orientation[0] = pos;
|
|
this.setOrientation.apply(this, this._orientation);
|
|
},
|
|
get : function(){
|
|
return this._orientation[0];
|
|
}
|
|
});
|
|
|
|
/**
|
|
* The y orientation of the panner object.
|
|
* @type {Number}
|
|
* @memberOf Tone.Panner3D#
|
|
* @name orientationY
|
|
*/
|
|
Object.defineProperty(Tone.Panner3D.prototype, "orientationY", {
|
|
set : function(pos){
|
|
this._orientation[1] = pos;
|
|
this.setOrientation.apply(this, this._orientation);
|
|
},
|
|
get : function(){
|
|
return this._orientation[1];
|
|
}
|
|
});
|
|
|
|
/**
|
|
* The z orientation of the panner object.
|
|
* @type {Number}
|
|
* @memberOf Tone.Panner3D#
|
|
* @name orientationZ
|
|
*/
|
|
Object.defineProperty(Tone.Panner3D.prototype, "orientationZ", {
|
|
set : function(pos){
|
|
this._orientation[2] = pos;
|
|
this.setOrientation.apply(this, this._orientation);
|
|
},
|
|
get : function(){
|
|
return this._orientation[2];
|
|
}
|
|
});
|
|
|
|
/**
|
|
* Proxy a property on the panner to an exposed public propery
|
|
* @param {String} prop
|
|
* @private
|
|
*/
|
|
Tone.Panner3D._aliasProperty = function(prop){
|
|
Object.defineProperty(Tone.Panner3D.prototype, prop, {
|
|
set : function(val){
|
|
this._panner[prop] = val;
|
|
},
|
|
get : function(){
|
|
return this._panner[prop];
|
|
}
|
|
});
|
|
};
|
|
|
|
/**
|
|
* The panning model. Either "equalpower" or "HRTF".
|
|
* @type {String}
|
|
* @memberOf Tone.Panner3D#
|
|
* @name panningModel
|
|
*/
|
|
Tone.Panner3D._aliasProperty("panningModel");
|
|
|
|
/**
|
|
* A reference distance for reducing volume as source move further from the listener
|
|
* @type {Number}
|
|
* @memberOf Tone.Panner3D#
|
|
* @name refDistance
|
|
*/
|
|
Tone.Panner3D._aliasProperty("refDistance");
|
|
|
|
/**
|
|
* Describes how quickly the volume is reduced as source moves away from listener.
|
|
* @type {Number}
|
|
* @memberOf Tone.Panner3D#
|
|
* @name rolloffFactor
|
|
*/
|
|
Tone.Panner3D._aliasProperty("rolloffFactor");
|
|
|
|
/**
|
|
* The distance model used by, "linear", "inverse", or "exponential".
|
|
* @type {String}
|
|
* @memberOf Tone.Panner3D#
|
|
* @name distanceModel
|
|
*/
|
|
Tone.Panner3D._aliasProperty("distanceModel");
|
|
|
|
/**
|
|
* The angle, in degrees, inside of which there will be no volume reduction
|
|
* @type {Degrees}
|
|
* @memberOf Tone.Panner3D#
|
|
* @name coneInnerAngle
|
|
*/
|
|
Tone.Panner3D._aliasProperty("coneInnerAngle");
|
|
|
|
/**
|
|
* The angle, in degrees, outside of which the volume will be reduced
|
|
* to a constant value of coneOuterGain
|
|
* @type {Degrees}
|
|
* @memberOf Tone.Panner3D#
|
|
* @name coneOuterAngle
|
|
*/
|
|
Tone.Panner3D._aliasProperty("coneOuterAngle");
|
|
|
|
/**
|
|
* The gain outside of the coneOuterAngle
|
|
* @type {Gain}
|
|
* @memberOf Tone.Panner3D#
|
|
* @name coneOuterGain
|
|
*/
|
|
Tone.Panner3D._aliasProperty("coneOuterGain");
|
|
|
|
/**
|
|
* The maximum distance between source and listener,
|
|
* after which the volume will not be reduced any further.
|
|
* @type {Positive}
|
|
* @memberOf Tone.Panner3D#
|
|
* @name maxDistance
|
|
*/
|
|
Tone.Panner3D._aliasProperty("maxDistance");
|
|
|
|
/**
|
|
* Clean up.
|
|
* @returns {Tone.Panner3D} this
|
|
*/
|
|
Tone.Panner3D.prototype.dispose = function(){
|
|
Tone.AudioNode.prototype.dispose.call(this);
|
|
this._panner.disconnect();
|
|
this._panner = null;
|
|
this._orientation = null;
|
|
this._position = null;
|
|
return this;
|
|
};
|
|
|
|
return Tone.Panner3D;
|
|
});
|