phaser/src/sound/BaseSound.js

604 lines
16 KiB
JavaScript
Raw Normal View History

2018-02-12 16:01:20 +00:00
/**
* @author Richard Davey <rich@photonstorm.com>
* @copyright 2018 Photon Storm Ltd.
* @license {@link https://github.com/photonstorm/phaser/blob/master/license.txt|MIT License}
*/
2017-11-14 15:00:24 +00:00
var Class = require('../utils/Class');
var EventEmitter = require('eventemitter3');
2018-02-12 12:25:30 +00:00
var Extend = require('../utils/object/Extend');
var NOOP = require('../utils/NOOP');
2018-01-26 14:34:18 +00:00
2018-02-12 12:25:30 +00:00
/**
* @classdesc
2018-02-18 14:50:32 +00:00
* Class containing all the shared state and behaviour of a sound object, independent of the implementation.
2018-02-12 12:25:30 +00:00
*
* @class BaseSound
2018-02-13 01:39:22 +00:00
* @extends EventEmitter
2018-02-12 12:25:30 +00:00
* @memberOf Phaser.Sound
* @constructor
2018-01-06 14:45:42 +00:00
* @author Pavle Goloskokovic <pgoloskokovic@gmail.com> (http://prunegames.com)
2018-02-12 12:25:30 +00:00
* @since 3.0.0
*
* @param {Phaser.Sound.BaseSoundManager} manager - Reference to the current sound manager instance.
* @param {string} key - Asset key for the sound.
2018-02-18 14:51:50 +00:00
* @param {SoundConfig} [config] - An optional config object containing default sound settings.
2018-01-06 14:45:42 +00:00
*/
2017-11-14 15:00:24 +00:00
var BaseSound = new Class({
2018-02-12 12:25:30 +00:00
Extends: EventEmitter,
2018-01-26 14:34:18 +00:00
2018-02-12 12:25:30 +00:00
initialize:
function BaseSound (manager, key, config)
2018-01-26 14:34:18 +00:00
{
EventEmitter.call(this);
2018-01-26 14:34:18 +00:00
2017-11-14 15:00:24 +00:00
/**
* Local reference to the sound manager.
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#manager
* @type {Phaser.Sound.BaseSoundManager}
* @private
2018-02-12 12:25:30 +00:00
* @since 3.0.0
2017-11-14 15:00:24 +00:00
*/
this.manager = manager;
2018-01-26 14:34:18 +00:00
/**
2018-01-06 14:47:58 +00:00
* Asset key for the sound.
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#key
* @type {string}
* @readOnly
* @since 3.0.0
*/
2017-11-14 15:00:24 +00:00
this.key = key;
2018-01-26 14:34:18 +00:00
/**
* Flag indicating if sound is currently playing.
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#isPlaying
* @type {boolean}
* @default false
2018-02-12 12:25:30 +00:00
* @readOnly
* @since 3.0.0
*/
this.isPlaying = false;
2018-01-26 14:34:18 +00:00
/**
* Flag indicating if sound is currently paused.
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#isPaused
* @type {boolean}
* @default false
2018-02-12 12:25:30 +00:00
* @readOnly
* @since 3.0.0
*/
this.isPaused = false;
2018-01-26 14:34:18 +00:00
/**
* A property that holds the value of sound's actual playback rate,
* after its rate and detune values has been combined with global
* rate and detune values.
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#totalRate
* @type {number}
* @default 1
2018-02-12 12:25:30 +00:00
* @readOnly
* @since 3.0.0
*/
this.totalRate = 1;
2018-01-26 14:34:18 +00:00
/**
* A value representing the duration, in seconds.
* It could be total sound duration or a marker duration.
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#duration
* @type {number}
* @readOnly
* @since 3.0.0
*/
this.duration = this.duration || 0;
2018-01-26 14:34:18 +00:00
/**
2018-01-06 14:47:58 +00:00
* The total duration of the sound in seconds.
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#totalDuration
* @type {number}
* @readOnly
* @since 3.0.0
*/
this.totalDuration = this.totalDuration || 0;
2018-01-26 14:34:18 +00:00
/**
2018-01-06 14:47:58 +00:00
* A config object used to store default sound settings' values.
* Default values will be set by properties' setters.
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#config
2018-02-18 14:52:39 +00:00
* @type {SoundConfig}
* @private
2018-02-12 12:25:30 +00:00
* @since 3.0.0
*/
this.config = {
/**
* Initializing delay config setting
*/
delay: 0
};
2018-01-26 14:34:18 +00:00
/**
* Reference to the currently used config.
* It could be default config or marker config.
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#currentConfig
2018-02-18 14:52:57 +00:00
* @type {SoundConfig}
* @private
2018-02-12 12:25:30 +00:00
* @since 3.0.0
*/
this.currentConfig = this.config;
2018-01-26 14:34:18 +00:00
/**
2017-12-07 19:57:05 +00:00
* Boolean indicating whether the sound is muted or not.
* Gets or sets the muted state of this sound.
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#mute
* @type {boolean}
* @default false
2018-02-12 12:25:30 +00:00
* @since 3.0.0
*/
2017-11-14 15:00:24 +00:00
this.mute = false;
2018-01-26 14:34:18 +00:00
/**
* Gets or sets the volume of this sound,
* a value between 0 (silence) and 1 (full volume).
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#volume
* @type {number}
* @default 1
2018-02-12 12:25:30 +00:00
* @since 3.0.0
*/
2017-11-14 15:00:24 +00:00
this.volume = 1;
2018-01-26 14:34:18 +00:00
/**
* Defines the speed at which the audio asset will be played.
* Value of 1.0 plays the audio at full speed, 0.5 plays the audio
* at half speed and 2.0 doubles the audio's playback speed.
* This value gets multiplied by global rate to have the final playback speed.
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#rate
* @type {number}
* @default 1
2018-02-12 12:25:30 +00:00
* @since 3.0.0
*/
2017-11-14 15:00:24 +00:00
this.rate = 1;
2018-01-26 14:34:18 +00:00
2017-11-16 15:41:48 +00:00
/**
* Represents detuning of sound in [cents](https://en.wikipedia.org/wiki/Cent_%28music%29).
* The range of the value is -1200 to 1200, but we recommend setting it to [50](https://en.wikipedia.org/wiki/50_Cent).
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#detune
* @type {number}
* @default 0
2018-02-12 12:25:30 +00:00
* @since 3.0.0
2017-11-16 15:41:48 +00:00
*/
this.detune = 0;
2018-01-26 14:34:18 +00:00
/**
2017-12-07 19:56:34 +00:00
* Property representing the position of playback for this sound, in seconds.
* Setting it to a specific value moves current playback to that position.
* The value given is clamped to the range 0 to current marker duration.
* Setting seek of a stopped sound has no effect.
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#seek
* @type {number}
* @default 0
2018-02-12 12:25:30 +00:00
* @since 3.0.0
*/
2017-11-14 15:00:24 +00:00
this.seek = 0;
2018-01-26 14:34:18 +00:00
/**
2018-01-06 14:47:58 +00:00
* Flag indicating whether or not the sound or current sound marker will loop.
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#loop
* @type {boolean}
* @default false
2018-02-12 12:25:30 +00:00
* @since 3.0.0
*/
2017-11-14 15:00:24 +00:00
this.loop = false;
this.config = Extend(this.config, config);
2018-01-26 14:34:18 +00:00
/**
2018-02-18 14:54:16 +00:00
* Object containing markers definitions.
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#markers
2018-02-18 14:54:16 +00:00
* @type {Object.<string, SoundMarker>}
* @default {}
2018-02-12 12:25:30 +00:00
* @readOnly
* @since 3.0.0
*/
this.markers = {};
2018-01-26 14:34:18 +00:00
/**
* Currently playing marker.
* 'null' if whole sound is playing.
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#currentMarker
2018-02-18 14:54:36 +00:00
* @type {SoundMarker}
* @default null
2018-02-12 12:25:30 +00:00
* @readOnly
* @since 3.0.0
*/
this.currentMarker = null;
2018-01-26 14:34:18 +00:00
/**
* Flag indicating if destroy method was called on this sound.
*
2018-02-12 12:25:30 +00:00
* @name Phaser.Sound.BaseSound#pendingRemove
* @type {boolean}
* @private
* @default false
2018-02-12 12:25:30 +00:00
* @since 3.0.0
*/
this.pendingRemove = false;
2017-11-14 15:00:24 +00:00
},
2018-01-26 14:34:18 +00:00
2018-01-06 14:50:14 +00:00
/**
* Adds a marker into the current sound. A marker is represented by name, start time, duration, and optionally config object.
* This allows you to bundle multiple sounds together into a single audio file and use markers to jump between them for playback.
*
* @method Phaser.Sound.BaseSound#addMarker
2018-02-12 12:25:30 +00:00
* @since 3.0.0
*
2018-02-18 14:55:08 +00:00
* @param {SoundMarker} marker - Marker object.
2018-02-12 12:25:30 +00:00
*
2018-02-18 14:55:08 +00:00
* @return {boolean} Whether the marker was added successfully.
2018-01-06 14:50:14 +00:00
*/
2018-01-26 14:34:18 +00:00
addMarker: function (marker)
{
if (!marker || !marker.name || typeof marker.name !== 'string')
2018-01-26 14:34:18 +00:00
{
return false;
2017-11-23 11:34:43 +00:00
}
2018-02-12 12:25:30 +00:00
2018-01-26 14:34:18 +00:00
if (this.markers[marker.name])
{
// eslint-disable-next-line no-console
console.error('addMarker - Marker with name \'' + marker.name + '\' already exists for sound \'' + this.key + '\'!');
return false;
2017-11-23 11:35:55 +00:00
}
2018-02-12 12:25:30 +00:00
marker = Extend(true, {
name: '',
start: 0,
duration: this.totalDuration,
config: {
mute: false,
volume: 1,
rate: 1,
detune: 0,
seek: 0,
loop: false,
delay: 0
}
}, marker);
2018-02-12 12:25:30 +00:00
this.markers[marker.name] = marker;
2018-02-12 12:25:30 +00:00
2017-11-23 11:35:55 +00:00
return true;
2017-11-14 15:00:24 +00:00
},
2018-01-26 14:34:18 +00:00
2018-01-06 14:50:40 +00:00
/**
* Updates previously added marker.
*
* @method Phaser.Sound.BaseSound#updateMarker
2018-02-12 12:25:30 +00:00
* @since 3.0.0
*
2018-01-06 14:50:40 +00:00
* @param {ISoundMarker} marker - Marker object with updated values.
2018-02-12 12:25:30 +00:00
*
* @return {boolean} Whether the marker was updated successfully.
2018-01-06 14:50:40 +00:00
*/
2018-01-26 14:34:18 +00:00
updateMarker: function (marker)
{
if (!marker || !marker.name || typeof marker.name !== 'string')
2018-01-26 14:34:18 +00:00
{
return false;
}
2018-02-12 12:25:30 +00:00
2018-01-26 14:34:18 +00:00
if (!this.markers[marker.name])
{
// eslint-disable-next-line no-console
console.error('updateMarker - Marker with name \'' + marker.name + '\' does not exist for sound \'' + this.key + '\'!');
return false;
}
2018-02-12 12:25:30 +00:00
this.markers[marker.name] = Extend(true, this.markers[marker.name], marker);
2018-02-12 12:25:30 +00:00
return true;
},
2018-01-26 14:34:18 +00:00
2018-01-06 14:51:12 +00:00
/**
* Removes a marker from the sound.
*
* @method Phaser.Sound.BaseSound#removeMarker
2018-02-12 12:25:30 +00:00
* @since 3.0.0
*
2018-01-06 14:51:12 +00:00
* @param {string} markerName - The name of the marker to remove.
2018-02-12 12:25:30 +00:00
*
* @return {ISoundMarker|null} Removed marker object or 'null' if there was no marker with provided name.
2018-01-06 14:51:12 +00:00
*/
2018-01-26 14:34:18 +00:00
removeMarker: function (markerName)
{
var marker = this.markers[markerName];
2018-02-12 12:25:30 +00:00
2018-01-26 14:34:18 +00:00
if (!marker)
{
return null;
}
2018-02-12 12:25:30 +00:00
this.markers[markerName] = null;
2018-02-12 12:25:30 +00:00
return marker;
2017-11-14 15:00:24 +00:00
},
2018-01-26 14:34:18 +00:00
2018-01-06 14:52:22 +00:00
/**
* Play this sound, or a marked section of it.
* It always plays the sound from the start. If you want to start playback from a specific time
* you can set 'seek' setting of the config object, provided to this call, to that value.
*
* @method Phaser.Sound.BaseSound#play
2018-02-12 12:25:30 +00:00
* @since 3.0.0
*
2018-01-06 14:52:22 +00:00
* @param {string} [markerName=''] - If you want to play a marker then provide the marker name here, otherwise omit it to play the full sound.
* @param {ISoundConfig} [config] - Optional sound config object to be applied to this marker or entire sound if no marker name is provided. It gets memorized for future plays of current section of the sound.
2018-02-12 12:25:30 +00:00
*
* @return {boolean} Whether the sound started playing successfully.
2018-01-06 14:52:22 +00:00
*/
2018-01-26 14:34:18 +00:00
play: function (markerName, config)
{
if (markerName === undefined) { markerName = ''; }
2018-02-12 12:25:30 +00:00
2018-01-26 14:34:18 +00:00
if (typeof markerName === 'object')
{
config = markerName;
markerName = '';
}
2018-02-12 12:25:30 +00:00
2018-01-26 14:34:18 +00:00
if (typeof markerName !== 'string')
{
// eslint-disable-next-line no-console
console.error('Sound marker name has to be a string!');
return false;
}
2018-02-12 12:25:30 +00:00
2018-01-26 14:34:18 +00:00
if (!markerName)
{
this.currentMarker = null;
this.currentConfig = this.config;
this.duration = this.totalDuration;
}
2018-01-26 14:34:18 +00:00
else
{
if (!this.markers[markerName])
{
// eslint-disable-next-line no-console
console.error('No marker with name \'' + markerName + '\' found for sound \'' + this.key + '\'!');
return false;
}
2018-02-12 12:25:30 +00:00
this.currentMarker = this.markers[markerName];
this.currentConfig = this.currentMarker.config;
this.duration = this.currentMarker.duration;
}
2018-02-12 12:25:30 +00:00
this.resetConfig();
this.currentConfig = Extend(this.currentConfig, config);
this.isPlaying = true;
this.isPaused = false;
2018-02-12 12:25:30 +00:00
return true;
2017-11-14 15:00:24 +00:00
},
2018-01-26 14:34:18 +00:00
2018-01-06 14:52:46 +00:00
/**
* Pauses the sound.
*
* @method Phaser.Sound.BaseSound#pause
2018-02-12 12:25:30 +00:00
* @since 3.0.0
2018-02-18 14:50:32 +00:00
*
2018-02-12 12:25:30 +00:00
* @return {boolean} Whether the sound was paused successfully.
2018-01-06 14:52:46 +00:00
*/
2018-01-26 14:34:18 +00:00
pause: function ()
{
if (this.isPaused || !this.isPlaying)
{
return false;
}
2018-02-12 12:25:30 +00:00
this.isPlaying = false;
this.isPaused = true;
2018-02-12 12:25:30 +00:00
return true;
2017-11-14 15:00:24 +00:00
},
2018-01-26 14:34:18 +00:00
2018-01-06 14:53:03 +00:00
/**
* Resumes the sound.
*
* @method Phaser.Sound.BaseSound#resume
2018-02-12 12:25:30 +00:00
* @since 3.0.0
2018-02-18 14:50:32 +00:00
*
2018-02-12 12:25:30 +00:00
* @return {boolean} Whether the sound was resumed successfully.
2018-01-06 14:53:03 +00:00
*/
2018-01-26 14:34:18 +00:00
resume: function ()
{
if (!this.isPaused || this.isPlaying)
{
return false;
}
2018-02-12 12:25:30 +00:00
this.isPlaying = true;
this.isPaused = false;
2018-02-12 12:25:30 +00:00
return true;
2017-11-14 15:00:24 +00:00
},
2018-01-26 14:34:18 +00:00
2018-01-06 14:53:23 +00:00
/**
* Stop playing this sound.
*
* @method Phaser.Sound.BaseSound#stop
2018-02-12 12:25:30 +00:00
* @since 3.0.0
2018-02-18 14:50:32 +00:00
*
2018-02-12 12:25:30 +00:00
* @return {boolean} Whether the sound was stopped successfully.
2018-01-06 14:53:23 +00:00
*/
2018-01-26 14:34:18 +00:00
stop: function ()
{
if (!this.isPaused && !this.isPlaying)
{
return false;
}
this.isPlaying = false;
this.isPaused = false;
2017-12-07 19:17:38 +00:00
this.resetConfig();
return true;
2017-11-14 15:00:24 +00:00
},
2018-01-26 14:34:18 +00:00
2017-11-26 15:19:56 +00:00
/**
2018-01-07 21:07:04 +00:00
* Method used internally for applying config values to some of the sound properties.
*
2018-01-06 16:34:22 +00:00
* @method Phaser.Sound.BaseSound#applyConfig
2018-02-12 12:25:30 +00:00
* @protected
* @since 3.0.0
2017-11-26 15:19:56 +00:00
*/
2018-01-26 14:34:18 +00:00
applyConfig: function ()
{
this.mute = this.currentConfig.mute;
this.volume = this.currentConfig.volume;
this.rate = this.currentConfig.rate;
this.detune = this.currentConfig.detune;
this.loop = this.currentConfig.loop;
},
2018-01-26 14:34:18 +00:00
/**
2018-01-07 21:07:20 +00:00
* Method used internally for resetting values of some of the config properties.
*
2018-01-06 16:34:22 +00:00
* @method Phaser.Sound.BaseSound#resetConfig
2018-02-12 12:25:30 +00:00
* @protected
* @since 3.0.0
*/
2018-01-26 14:34:18 +00:00
resetConfig: function ()
{
this.currentConfig.seek = 0;
this.currentConfig.delay = 0;
},
2018-01-26 14:34:18 +00:00
/**
2018-01-06 16:33:46 +00:00
* Update method called automatically by sound manager on every game step.
*
2018-02-12 12:25:30 +00:00
* @method Phaser.Sound.BaseSound#update
2018-01-06 16:33:46 +00:00
* @override
* @protected
2018-02-12 12:25:30 +00:00
* @since 3.0.0
2018-02-18 14:50:32 +00:00
*
2018-01-06 16:33:46 +00:00
* @param {number} time - The current timestamp as generated by the Request Animation Frame or SetTimeout.
* @param {number} delta - The delta time elapsed since the last frame.
*/
update: NOOP,
2018-01-26 14:34:18 +00:00
2018-01-06 14:53:50 +00:00
/**
* Destroys this sound and all associated events and marks it for removal from the sound manager.
*
* @method Phaser.Sound.BaseSound#destroy
2018-02-12 12:25:30 +00:00
* @since 3.0.0
2018-01-06 14:53:50 +00:00
*/
2018-01-26 14:34:18 +00:00
destroy: function ()
{
if (this.pendingRemove)
{
return;
}
2018-02-12 12:25:30 +00:00
this.pendingRemove = true;
this.manager = null;
this.key = '';
this.removeAllListeners();
this.isPlaying = false;
this.isPaused = false;
this.config = null;
this.currentConfig = null;
this.markers = null;
this.currentMarker = null;
},
2018-01-26 14:34:18 +00:00
/**
2018-01-07 21:07:35 +00:00
* Method used internally to calculate total playback rate of the sound.
*
2018-01-06 16:34:22 +00:00
* @method Phaser.Sound.BaseSound#setRate
2018-02-12 12:25:30 +00:00
* @protected
* @since 3.0.0
*/
2018-01-26 14:34:18 +00:00
setRate: function ()
{
var cent = 1.0005777895065548; // Math.pow(2, 1/1200);
2017-12-04 21:05:29 +00:00
var totalDetune = this.currentConfig.detune + this.manager.detune;
var detuneRate = Math.pow(cent, totalDetune);
2018-02-12 12:25:30 +00:00
this.totalRate = this.currentConfig.rate * this.manager.rate * detuneRate;
2017-11-14 15:00:24 +00:00
}
});
2018-01-26 14:34:18 +00:00
/**
* Playback rate.
*
* @name Phaser.Sound.BaseSound#rate
2018-02-12 12:25:30 +00:00
* @type {number}
* @since 3.0.0
*/
Object.defineProperty(BaseSound.prototype, 'rate', {
2018-02-12 12:25:30 +00:00
2018-01-26 14:34:18 +00:00
get: function ()
{
return this.currentConfig.rate;
},
2018-02-12 12:25:30 +00:00
2018-01-26 14:34:18 +00:00
set: function (value)
{
this.currentConfig.rate = value;
this.setRate();
2018-01-26 14:34:18 +00:00
2018-01-26 13:54:15 +00:00
/**
* @event Phaser.Sound.BaseSound#rate
* @param {Phaser.Sound.BaseSound} sound - Reference to the sound that emitted event.
* @param {number} value - An updated value of Phaser.Sound.BaseSound#rate property.
*/
this.emit('rate', this, value);
}
});
2018-01-26 14:34:18 +00:00
/**
* Detuning of sound.
*
* @name Phaser.Sound.BaseSound#detune
* @property {number} detune
2018-02-12 12:25:30 +00:00
* @since 3.0.0
*/
Object.defineProperty(BaseSound.prototype, 'detune', {
2018-02-12 12:25:30 +00:00
2018-01-26 14:34:18 +00:00
get: function ()
{
return this.currentConfig.detune;
},
2018-02-12 12:25:30 +00:00
2018-01-26 14:34:18 +00:00
set: function (value)
{
this.currentConfig.detune = value;
this.setRate();
2018-01-26 14:34:18 +00:00
2018-01-26 13:54:45 +00:00
/**
* @event Phaser.Sound.BaseSound#detune
* @param {Phaser.Sound.BaseSound} sound - Reference to the sound that emitted event.
* @param {number} value - An updated value of Phaser.Sound.BaseSound#detune property.
*/
this.emit('detune', this, value);
}
});
2018-02-12 12:25:30 +00:00
2017-11-14 15:00:24 +00:00
module.exports = BaseSound;