2018-01-08 18:25:11 +00:00
|
|
|
var Class = require('../../utils/Class');
|
|
|
|
var BaseSoundManager = require('../BaseSoundManager');
|
2018-01-08 18:26:32 +00:00
|
|
|
var HTML5AudioSound = require('./HTML5AudioSound');
|
2018-01-26 13:31:10 +00:00
|
|
|
/*!
|
|
|
|
* @author Pavle Goloskokovic <pgoloskokovic@gmail.com> (http://prunegames.com)
|
|
|
|
*/
|
2018-01-08 18:25:11 +00:00
|
|
|
var HTML5AudioSoundManager = new Class({
|
|
|
|
Extends: BaseSoundManager,
|
2018-01-26 13:31:10 +00:00
|
|
|
/**
|
|
|
|
* HTML5 Audio implementation of the sound manager.
|
|
|
|
*
|
|
|
|
* @class Phaser.Sound.HTML5AudioSoundManager
|
|
|
|
* @constructor
|
|
|
|
* @param {Phaser.Game} game - Reference to the current game instance.
|
|
|
|
*/
|
2018-01-08 18:25:11 +00:00
|
|
|
initialize: function HTML5AudioSoundManager(game) {
|
2018-01-11 15:50:50 +00:00
|
|
|
/**
|
2018-01-11 16:32:49 +00:00
|
|
|
* Flag indicating whether if there are no idle instances of HTML5 Audio tag,
|
2018-01-26 13:31:48 +00:00
|
|
|
* for any particular sound, if one of the used tags should be hijacked and used
|
2018-01-11 16:32:49 +00:00
|
|
|
* for succeeding playback or if succeeding Phaser.Sound.HTML5AudioSound#play
|
|
|
|
* call should be ignored.
|
2018-01-11 15:50:50 +00:00
|
|
|
*
|
|
|
|
* @property {boolean} override
|
|
|
|
* @default true
|
|
|
|
*/
|
|
|
|
this.override = true;
|
2018-01-14 14:29:27 +00:00
|
|
|
/**
|
2018-01-26 13:32:11 +00:00
|
|
|
* Value representing time difference, in seconds, between calling
|
2018-01-14 14:29:27 +00:00
|
|
|
* play method on an audio tag and when it actually starts playing.
|
|
|
|
* It is used to achieve more accurate delayed sound playback.
|
|
|
|
*
|
|
|
|
* You might need to tweak this value to get the desired results
|
|
|
|
* since audio play delay varies depending on the browser/platform.
|
|
|
|
*
|
|
|
|
* @property {number} audioPlayDelay
|
|
|
|
* @default 0.1
|
|
|
|
*/
|
|
|
|
this.audioPlayDelay = 0.1;
|
2018-01-14 14:31:24 +00:00
|
|
|
/**
|
2018-01-26 13:32:38 +00:00
|
|
|
* A value by which we should offset the loop end marker of the
|
|
|
|
* looping sound to compensate for lag, caused by changing audio
|
|
|
|
* tag playback position, in order to achieve gapless looping.
|
2018-01-14 14:31:24 +00:00
|
|
|
*
|
|
|
|
* You might need to tweak this value to get the desired results
|
|
|
|
* since loop lag varies depending on the browser/platform.
|
|
|
|
*
|
|
|
|
* @property {number} loopEndOffset
|
|
|
|
* @default 0.05
|
|
|
|
*/
|
|
|
|
this.loopEndOffset = 0.05;
|
2018-01-08 18:34:23 +00:00
|
|
|
/**
|
|
|
|
* An array for keeping track of all the sounds
|
|
|
|
* that were paused when game lost focus.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @property {Phaser.Sound.HTML5AudioSound[]} onBlurPausedSounds
|
|
|
|
* @default []
|
|
|
|
*/
|
2018-01-08 18:27:44 +00:00
|
|
|
this.onBlurPausedSounds = [];
|
2018-01-17 18:03:16 +00:00
|
|
|
this.locked = 'ontouchstart' in window;
|
2018-01-17 18:02:10 +00:00
|
|
|
/**
|
|
|
|
* A queue of all actions performed on sound objects while audio was locked.
|
|
|
|
* Once the audio gets unlocked, after an explicit user interaction,
|
|
|
|
* all actions will be performed in chronological order.
|
|
|
|
*
|
|
|
|
* @private
|
2018-01-26 13:33:15 +00:00
|
|
|
* @property {{ sound: Phaser.Sound.HTML5AudioSound, name: string, value?: * }[]} lockedActionsQueue
|
2018-01-17 18:02:10 +00:00
|
|
|
*/
|
2018-01-17 18:03:16 +00:00
|
|
|
this.lockedActionsQueue = this.locked ? [] : null;
|
2018-01-08 18:34:41 +00:00
|
|
|
/**
|
|
|
|
* Property that actually holds the value of global mute
|
|
|
|
* for HTML5 Audio sound manager implementation.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @property {boolean} _mute
|
|
|
|
* @default false
|
|
|
|
*/
|
2018-01-08 18:30:42 +00:00
|
|
|
this._mute = false;
|
2018-01-08 18:34:55 +00:00
|
|
|
/**
|
|
|
|
* Property that actually holds the value of global volume
|
|
|
|
* for HTML5 Audio sound manager implementation.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @property {boolean} _volume
|
|
|
|
* @default 1
|
|
|
|
*/
|
2018-01-08 18:32:41 +00:00
|
|
|
this._volume = 1;
|
2018-01-08 18:25:11 +00:00
|
|
|
BaseSoundManager.call(this, game);
|
2018-01-08 18:26:32 +00:00
|
|
|
},
|
2018-01-26 13:33:54 +00:00
|
|
|
/**
|
|
|
|
* Adds a new sound into the sound manager.
|
|
|
|
*
|
|
|
|
* @method Phaser.Sound.HTML5AudioSoundManager#add
|
|
|
|
* @param {string} key - Asset key for the sound.
|
|
|
|
* @param {ISoundConfig} [config] - An optional config object containing default sound settings.
|
|
|
|
* @returns {Phaser.Sound.HTML5AudioSound} The new sound instance.
|
|
|
|
*/
|
2018-01-08 18:26:32 +00:00
|
|
|
add: function (key, config) {
|
|
|
|
var sound = new HTML5AudioSound(this, key, config);
|
|
|
|
this.sounds.push(sound);
|
|
|
|
return sound;
|
2018-01-08 18:28:21 +00:00
|
|
|
},
|
2018-01-26 13:34:32 +00:00
|
|
|
/**
|
|
|
|
* Unlocks HTML5 Audio loading and playback on mobile
|
|
|
|
* devices on the initial explicit user interaction.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @method Phaser.Sound.HTML5AudioSoundManager#unlock
|
|
|
|
*/
|
2018-01-17 13:12:04 +00:00
|
|
|
unlock: function () {
|
|
|
|
var _this = this;
|
2018-01-18 13:10:03 +00:00
|
|
|
var moved = false;
|
|
|
|
var detectMove = function () {
|
|
|
|
moved = true;
|
|
|
|
};
|
2018-01-17 17:19:26 +00:00
|
|
|
var unlock = function () {
|
2018-01-18 13:11:49 +00:00
|
|
|
if (moved) {
|
|
|
|
moved = false;
|
|
|
|
return;
|
|
|
|
}
|
2018-01-18 13:12:32 +00:00
|
|
|
document.body.removeEventListener('touchmove', detectMove);
|
2018-01-17 17:19:26 +00:00
|
|
|
document.body.removeEventListener('touchend', unlock);
|
|
|
|
var allTags = [];
|
|
|
|
_this.game.cache.audio.entries.each(function (key, tags) {
|
|
|
|
for (var i = 0; i < tags.length; i++) {
|
|
|
|
allTags.push(tags[i]);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
});
|
|
|
|
var lastTag = allTags[allTags.length - 1];
|
|
|
|
lastTag.oncanplaythrough = function () {
|
|
|
|
lastTag.oncanplaythrough = null;
|
2018-01-17 17:19:51 +00:00
|
|
|
_this.unlocked = true;
|
2018-01-17 13:12:04 +00:00
|
|
|
};
|
2018-01-17 17:19:26 +00:00
|
|
|
allTags.forEach(function (tag) {
|
|
|
|
tag.load();
|
|
|
|
});
|
|
|
|
};
|
2018-01-17 17:24:03 +00:00
|
|
|
this.once('unlocked', function () {
|
2018-01-18 13:08:33 +00:00
|
|
|
_this.forEachActiveSound(function (sound) {
|
|
|
|
sound.duration = sound.tags[0].duration;
|
2018-01-18 13:08:49 +00:00
|
|
|
sound.totalDuration = sound.tags[0].duration;
|
2018-01-18 13:08:33 +00:00
|
|
|
});
|
2018-01-17 18:07:52 +00:00
|
|
|
_this.lockedActionsQueue.forEach(function (lockedAction) {
|
2018-01-17 18:16:21 +00:00
|
|
|
if (lockedAction.sound[lockedAction.prop].apply) {
|
|
|
|
lockedAction.sound[lockedAction.prop].apply(lockedAction.sound, lockedAction.value || []);
|
2018-01-17 18:07:52 +00:00
|
|
|
}
|
|
|
|
else {
|
2018-01-17 18:16:21 +00:00
|
|
|
lockedAction.sound[lockedAction.prop] = lockedAction.value;
|
2018-01-17 16:35:08 +00:00
|
|
|
}
|
|
|
|
});
|
2018-01-17 18:09:36 +00:00
|
|
|
_this.lockedActionsQueue.length = 0;
|
|
|
|
_this.lockedActionsQueue = null;
|
2018-01-17 17:24:03 +00:00
|
|
|
});
|
2018-01-18 13:10:51 +00:00
|
|
|
document.body.addEventListener('touchmove', detectMove, false);
|
2018-01-17 17:24:03 +00:00
|
|
|
document.body.addEventListener('touchend', unlock, false);
|
|
|
|
},
|
2018-01-26 13:35:34 +00:00
|
|
|
/**
|
|
|
|
* Method used internally for pausing sound manager if
|
|
|
|
* Phaser.Sound.HTML5AudioSoundManager#pauseOnBlur is set to true.
|
|
|
|
*
|
|
|
|
* @protected
|
|
|
|
* @method Phaser.Sound.HTML5AudioSoundManager#onBlur
|
|
|
|
*/
|
2018-01-17 17:24:03 +00:00
|
|
|
onBlur: function () {
|
|
|
|
this.forEachActiveSound(function (sound) {
|
|
|
|
if (sound.isPlaying) {
|
|
|
|
this.onBlurPausedSounds.push(sound);
|
|
|
|
sound.onBlur();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
2018-01-26 13:36:01 +00:00
|
|
|
/**
|
|
|
|
* Method used internally for resuming sound manager if
|
|
|
|
* Phaser.Sound.HTML5AudioSoundManager#pauseOnBlur is set to true.
|
|
|
|
*
|
|
|
|
* @protected
|
|
|
|
* @method Phaser.Sound.HTML5AudioSoundManager#onFocus
|
|
|
|
*/
|
2018-01-17 17:24:03 +00:00
|
|
|
onFocus: function () {
|
|
|
|
this.onBlurPausedSounds.forEach(function (sound) {
|
|
|
|
sound.onFocus();
|
|
|
|
});
|
|
|
|
this.onBlurPausedSounds.length = 0;
|
2018-01-17 16:32:41 +00:00
|
|
|
},
|
2018-01-08 18:29:16 +00:00
|
|
|
destroy: function () {
|
|
|
|
BaseSoundManager.prototype.destroy.call(this);
|
|
|
|
this.onBlurPausedSounds.length = 0;
|
|
|
|
this.onBlurPausedSounds = null;
|
2018-01-17 18:10:24 +00:00
|
|
|
},
|
2018-01-17 18:16:21 +00:00
|
|
|
isLocked: function (sound, prop, value) {
|
2018-01-17 18:10:24 +00:00
|
|
|
if (this.locked) {
|
|
|
|
this.lockedActionsQueue.push({
|
|
|
|
sound: sound,
|
2018-01-17 18:16:21 +00:00
|
|
|
prop: prop,
|
2018-01-17 18:10:24 +00:00
|
|
|
value: value
|
|
|
|
});
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2018-01-08 18:25:11 +00:00
|
|
|
}
|
|
|
|
});
|
2018-01-08 18:35:13 +00:00
|
|
|
/**
|
|
|
|
* Global mute setting.
|
|
|
|
*
|
|
|
|
* @name Phaser.Sound.HTML5AudioSoundManager#mute
|
|
|
|
* @property {boolean} mute
|
|
|
|
*/
|
2018-01-08 18:31:26 +00:00
|
|
|
Object.defineProperty(HTML5AudioSoundManager.prototype, 'mute', {
|
|
|
|
get: function () {
|
2018-01-08 18:32:09 +00:00
|
|
|
return this._mute;
|
2018-01-08 18:31:26 +00:00
|
|
|
},
|
|
|
|
set: function (value) {
|
2018-01-08 18:32:20 +00:00
|
|
|
this._mute = value;
|
|
|
|
this.forEachActiveSound(function (sound) {
|
|
|
|
sound.setMute();
|
|
|
|
});
|
2018-01-14 15:53:48 +00:00
|
|
|
this.emit('mute', this, value);
|
2018-01-08 18:31:26 +00:00
|
|
|
}
|
|
|
|
});
|
2018-01-08 18:35:29 +00:00
|
|
|
/**
|
|
|
|
* Global volume setting.
|
|
|
|
*
|
|
|
|
* @name Phaser.Sound.HTML5AudioSoundManager#volume
|
|
|
|
* @property {number} volume
|
|
|
|
*/
|
2018-01-08 18:33:10 +00:00
|
|
|
Object.defineProperty(HTML5AudioSoundManager.prototype, 'volume', {
|
|
|
|
get: function () {
|
2018-01-08 18:33:23 +00:00
|
|
|
return this._volume;
|
2018-01-08 18:33:10 +00:00
|
|
|
},
|
|
|
|
set: function (value) {
|
2018-01-08 18:33:43 +00:00
|
|
|
this._volume = value;
|
|
|
|
this.forEachActiveSound(function (sound) {
|
|
|
|
sound.setVolume();
|
|
|
|
});
|
2018-01-14 15:54:11 +00:00
|
|
|
this.emit('volume', this, value);
|
2018-01-08 18:33:10 +00:00
|
|
|
}
|
|
|
|
});
|
2018-01-08 18:25:11 +00:00
|
|
|
module.exports = HTML5AudioSoundManager;
|