1218 lines
44 KiB
JavaScript
1218 lines
44 KiB
JavaScript
/* eslint-disable no-param-reassign */
|
|
/* eslint-disable eqeqeq */
|
|
/* eslint-disable no-plusplus */
|
|
import { isAndroid, isPc, webAudioNeedResume, isSupportBufferURL, isSupportPlayBackRate, isSupportInnerAudio, isIOS175, } from '../../check-version';
|
|
import { WEBAudio, unityAudioVolume } from './store';
|
|
import { TEMP_DIR_PATH } from './const';
|
|
import { createInnerAudio, destroyInnerAudio, printErrMsg, resumeWebAudio } from './utils';
|
|
|
|
const defaultSoundLength = 441000;
|
|
|
|
const needGetLength = false;
|
|
function jsAudioCreateUncompressedSoundClip(buffer, error, length) {
|
|
const soundClip = {
|
|
buffer,
|
|
error,
|
|
release() {
|
|
this.buffer = null;
|
|
WEBAudio.audioBufferLength -= length;
|
|
},
|
|
resetGain() { },
|
|
getLength() {
|
|
if (!this.buffer) {
|
|
|
|
return 0;
|
|
}
|
|
const sampleRateRatio = 44100 / this.buffer.sampleRate;
|
|
return this.buffer.length * sampleRateRatio;
|
|
},
|
|
getData(ptr, length) {
|
|
if (!this.buffer) {
|
|
console.log('Trying to get data of sound which is not loaded.');
|
|
return 0;
|
|
}
|
|
const startOutputBuffer = ptr >> 2;
|
|
const output = GameGlobal.unityNamespace.Module.HEAPF32.subarray(startOutputBuffer, startOutputBuffer + (length >> 2));
|
|
const numMaxSamples = Math.floor((length >> 2) / this.buffer.numberOfChannels);
|
|
const numReadSamples = Math.min(this.buffer.length, numMaxSamples);
|
|
for (let i = 0; i < this.buffer.numberOfChannels; i++) {
|
|
const channelData = this.buffer.getChannelData(i).subarray(0, numReadSamples);
|
|
output.set(channelData, i * numReadSamples);
|
|
}
|
|
return numReadSamples * this.buffer.numberOfChannels * 4;
|
|
},
|
|
getNumberOfChannels() {
|
|
if (!this.buffer) {
|
|
console.log('Trying to get metadata of sound which is not loaded.');
|
|
return 0;
|
|
}
|
|
return this.buffer.numberOfChannels;
|
|
},
|
|
getFrequency() {
|
|
if (!this.buffer) {
|
|
console.log('Trying to get metadata of sound which is not loaded.');
|
|
return 0;
|
|
}
|
|
return this.buffer.sampleRate;
|
|
},
|
|
};
|
|
return soundClip;
|
|
}
|
|
function jsAudioCreateUncompressedSoundClipFromCompressedAudio(audioData, ptr, length) {
|
|
const soundClip = jsAudioCreateUncompressedSoundClip(null, false, length);
|
|
WEBAudio.audioContext?.decodeAudioData(audioData, (buffer) => {
|
|
soundClip.buffer = buffer;
|
|
WEBAudio.audioBufferLength += length;
|
|
}, (error) => {
|
|
soundClip.error = true;
|
|
console.log(`Decode error: ${error}`);
|
|
});
|
|
return soundClip;
|
|
}
|
|
function jsAudioCreateCompressedSoundClip(audioData, ptr, length) {
|
|
const soundClip = {
|
|
error: false,
|
|
length: needGetLength ? 0 : defaultSoundLength,
|
|
url: undefined,
|
|
release() {
|
|
WEBAudio.audioBufferLength -= length;
|
|
if (isSupportBufferURL && this.url) {
|
|
wx.revokeBufferURL(this.url);
|
|
}
|
|
delete this.url;
|
|
},
|
|
resetGain() { },
|
|
getLength() {
|
|
return this.length || 0;
|
|
},
|
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
getData(ptr, length) {
|
|
console.warn('getData() is not supported for compressed sound.');
|
|
return 0;
|
|
},
|
|
getNumberOfChannels() {
|
|
console.warn('getNumberOfChannels() is not supported for compressed sound.');
|
|
return 0;
|
|
},
|
|
getFrequency() {
|
|
console.warn('getFrequency() is not supported for compressed sound.');
|
|
return 0;
|
|
},
|
|
};
|
|
if (isSupportBufferURL) {
|
|
const url = wx.createBufferURL(audioData);
|
|
soundClip.url = url;
|
|
WEBAudio.audioBufferLength += length;
|
|
}
|
|
else {
|
|
const tempFilePath = `${TEMP_DIR_PATH}/temp-audio${ptr + length}.mp3`;
|
|
if (GameGlobal.manager.getCachePath(tempFilePath)) {
|
|
soundClip.url = tempFilePath;
|
|
WEBAudio.audioBufferLength += length;
|
|
}
|
|
else {
|
|
GameGlobal.manager
|
|
.writeFile(tempFilePath, audioData)
|
|
.then(() => {
|
|
soundClip.url = tempFilePath;
|
|
WEBAudio.audioBufferLength += length;
|
|
})
|
|
.catch((res) => {
|
|
soundClip.error = true;
|
|
printErrMsg(res);
|
|
});
|
|
}
|
|
}
|
|
if (needGetLength && soundClip.url) {
|
|
const { audio: getAudio } = createInnerAudio();
|
|
getAudio.src = soundClip.url;
|
|
getAudio.onCanplay(() => {
|
|
soundClip.length = getAudio.duration * 44100;
|
|
setTimeout(() => {
|
|
soundClip.length = getAudio.duration * 44100;
|
|
getAudio.destroy();
|
|
}, 0);
|
|
});
|
|
}
|
|
return soundClip;
|
|
}
|
|
function jsAudioCreateUncompressedSoundClipFromPCM(channels, length, sampleRate, ptr) {
|
|
if (WEBAudio.audioContext) {
|
|
const buffer = WEBAudio.audioContext.createBuffer(channels, length, sampleRate);
|
|
for (let i = 0; i < channels; i++) {
|
|
const offs = (ptr >> 2) + length * i;
|
|
const copyToChannel = buffer.copyToChannel
|
|
|| function (source, channelNumber, startInChannel) {
|
|
const clipped = source.subarray(0, Math.min(source.length, buffer.length - (startInChannel | 0)));
|
|
buffer.getChannelData(channelNumber | 0).set(clipped, startInChannel | 0);
|
|
};
|
|
copyToChannel.apply(buffer, [GameGlobal.unityNamespace.Module.HEAPF32.subarray(offs, offs + length), i, 0]);
|
|
}
|
|
return jsAudioCreateUncompressedSoundClip(buffer, false, length);
|
|
}
|
|
return jsAudioCreateUncompressedSoundClip(null, false, length);
|
|
}
|
|
export class AudioChannelInstance {
|
|
threeD = false;
|
|
source;
|
|
gain;
|
|
callback = 0;
|
|
userData = 0;
|
|
loop = false;
|
|
loopStart = 0;
|
|
loopEnd = 0;
|
|
deleyTime = 0;
|
|
deleyOffset = 0;
|
|
constructor(callback, userData) {
|
|
if (WEBAudio.audioContext) {
|
|
this.gain = WEBAudio.audioContext.createGain();
|
|
this.gain?.connect(WEBAudio.audioContext.destination);
|
|
}
|
|
this.callback = callback;
|
|
this.userData = userData;
|
|
}
|
|
resetGain() {
|
|
if (WEBAudio.audioContext && this.gain) {
|
|
this.gain.disconnect();
|
|
this.gain = WEBAudio.audioContext.createGain();
|
|
this.gain?.connect(WEBAudio.audioContext.destination);
|
|
}
|
|
}
|
|
release() {
|
|
this.disconnectSource();
|
|
if (this.gain) {
|
|
this.gain.disconnect();
|
|
}
|
|
}
|
|
setLoop(loop) {
|
|
this.loop = loop;
|
|
if (!this.source || this.source.loop == loop) {
|
|
return;
|
|
}
|
|
this.source.loop = loop;
|
|
}
|
|
setLoopPoints(loopStart, loopEnd) {
|
|
this.loopStart = loopStart;
|
|
this.loopEnd = loopEnd;
|
|
if (!this.source) {
|
|
return;
|
|
}
|
|
if (this.source.loopStart !== loopStart) {
|
|
this.source.loopStart = loopStart;
|
|
}
|
|
if (this.source.loopEnd !== loopEnd) {
|
|
this.source.loopEnd = loopEnd;
|
|
}
|
|
}
|
|
playUrl(startTime, url, startOffset, volume, soundClip) {
|
|
try {
|
|
this.setup(url);
|
|
if (!this.source || !this.source.mediaElement) {
|
|
return;
|
|
}
|
|
if (typeof volume !== 'undefined') {
|
|
this.source.mediaElement.volume = volume;
|
|
}
|
|
if (WEBAudio.isMute) {
|
|
this.source.mediaElement.volume = 0;
|
|
}
|
|
this.source.mediaElement.onPlay(() => {
|
|
if (typeof this.source !== 'undefined') {
|
|
this.source.isPlaying = true;
|
|
if (!this.source.loop && this.source.mediaElement) {
|
|
const { duration } = this.source.mediaElement;
|
|
|
|
if (duration > 0) {
|
|
if (this.source.stopTicker) {
|
|
clearTimeout(this.source.stopTicker);
|
|
this.source.stopTicker = undefined;
|
|
}
|
|
const time = Math.floor(duration * 1000) + 1000;
|
|
this.source.stopTicker = setTimeout(() => {
|
|
if (this.source && this.source.mediaElement) {
|
|
this.source.mediaElement.stop();
|
|
}
|
|
}, time);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
this.source.mediaElement.onPause(() => {
|
|
if (typeof this.source !== 'undefined') {
|
|
this.source.isPlaying = false;
|
|
if (this.source.stopTicker) {
|
|
clearTimeout(this.source.stopTicker);
|
|
this.source.stopTicker = undefined;
|
|
}
|
|
}
|
|
});
|
|
this.source.mediaElement.onStop(() => {
|
|
if (typeof this.source !== 'undefined') {
|
|
if (this.source.playAfterStop) {
|
|
this.source._reset();
|
|
if (typeof this.source.mediaElement !== 'undefined') {
|
|
this.source.mediaElement.play();
|
|
}
|
|
return;
|
|
}
|
|
this.source._reset();
|
|
this.disconnectSource();
|
|
}
|
|
if (this.callback) {
|
|
GameGlobal.unityNamespace.Module.dynCall_vi(this.callback, [this.userData]);
|
|
}
|
|
});
|
|
this.source.mediaElement.onEnded(() => {
|
|
if (typeof this.source !== 'undefined') {
|
|
this.source._reset();
|
|
this.disconnectSource();
|
|
}
|
|
if (this.callback) {
|
|
GameGlobal.unityNamespace.Module.dynCall_vi(this.callback, [this.userData]);
|
|
}
|
|
});
|
|
this.source.mediaElement.onError((e) => {
|
|
printErrMsg(e);
|
|
const { errMsg } = e;
|
|
|
|
if (errMsg && errMsg.indexOf('play audio fail') < 0) {
|
|
return;
|
|
}
|
|
|
|
if (typeof this.source !== 'undefined' && this.source.mediaElement) {
|
|
this.source._reset();
|
|
this.source.mediaElement.stop();
|
|
}
|
|
});
|
|
const fn = () => {
|
|
if (typeof this.source !== 'undefined' && this.source.mediaElement) {
|
|
// @ts-ignore
|
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
const { duration } = this.source.mediaElement;
|
|
setTimeout(() => {
|
|
if (soundClip && this.source && this.source.mediaElement) {
|
|
soundClip.length = Math.round(Math.max(this.source.mediaElement.duration, 0) * 44100);
|
|
}
|
|
}, 0);
|
|
}
|
|
};
|
|
if (!this.source.canPlayFnList) {
|
|
this.source.canPlayFnList = [];
|
|
}
|
|
this.source.canPlayFnList.push(fn);
|
|
this.source.mediaElement.onCanplay(fn);
|
|
this.source.mediaElement.loop = this.loop;
|
|
this.deleyTime = startTime;
|
|
this.deleyOffset = startOffset;
|
|
this.source.start(startTime, startOffset);
|
|
this.source.playbackStartTime = startTime - startOffset / this.source.playbackRateValue;
|
|
}
|
|
catch (e) {
|
|
printErrMsg(`playUrl error. Exception: ${e}`);
|
|
}
|
|
}
|
|
playBuffer(startTime, buffer, startOffset, channel) {
|
|
try {
|
|
this.setup();
|
|
if (!this.source) {
|
|
return;
|
|
}
|
|
this.source.buffer = buffer;
|
|
this.source.onended = () => {
|
|
this.disconnectSource();
|
|
if (this.callback) {
|
|
GameGlobal.unityNamespace.Module.dynCall_vi(this.callback, [this.userData]);
|
|
}
|
|
};
|
|
if (this.gain && channel) {
|
|
let volume;
|
|
if (WEBAudio.isMute) {
|
|
unityAudioVolume.set(channel, this.gain.gain.value || 1);
|
|
volume = 0;
|
|
}
|
|
else {
|
|
volume = unityAudioVolume.get(channel);
|
|
}
|
|
if (this.gain.gain.value !== volume && typeof volume === 'number') {
|
|
this.gain.gain.value = volume;
|
|
}
|
|
}
|
|
this.source.loop = this.loop;
|
|
this.source.loopStart = this.loopStart;
|
|
this.source.loopEnd = this.loopEnd;
|
|
this.source.start(startTime, startOffset);
|
|
this.source.playbackStartTime = startTime - startOffset / this.source.playbackRateValue;
|
|
}
|
|
catch (e) {
|
|
printErrMsg(`playBuffer error. Exception: ${e}`);
|
|
}
|
|
}
|
|
disconnectSource() {
|
|
if (this.source) {
|
|
if (this.source.mediaElement) {
|
|
destroyInnerAudio(this.source.instanceId, false);
|
|
delete this.source.mediaElement;
|
|
delete this.source;
|
|
}
|
|
else if (!this.source.isPausedMockNode) {
|
|
this.source.onended = null;
|
|
if (this.source.disconnect) {
|
|
this.source.disconnect();
|
|
}
|
|
if (GameGlobal.isIOSHighPerformanceMode) {
|
|
this.source.buffer = null;
|
|
}
|
|
WEBAudio.bufferSourceNodeLength -= 1;
|
|
delete this.source;
|
|
}
|
|
else {
|
|
this.source.buffer = null;
|
|
}
|
|
}
|
|
}
|
|
stop(delay) {
|
|
if (!WEBAudio.audioContext) {
|
|
return;
|
|
}
|
|
if (this.source) {
|
|
if (this.source.buffer) {
|
|
try {
|
|
this.source.stop(WEBAudio.audioContext.currentTime + delay);
|
|
}
|
|
catch (e) { }
|
|
if (delay == 0) {
|
|
this.disconnectSource();
|
|
}
|
|
}
|
|
else if (this.source.mediaElement) {
|
|
this.source.stop(delay);
|
|
}
|
|
}
|
|
}
|
|
isPaused() {
|
|
if (!this.source) {
|
|
return true;
|
|
}
|
|
if (this.source.isPausedMockNode) {
|
|
return true;
|
|
}
|
|
if (this.source.mediaElement) {
|
|
return (!this.source.isPlaying || this.source.pauseRequested) ?? true;
|
|
}
|
|
return false;
|
|
}
|
|
pause() {
|
|
const { source } = this;
|
|
if (!source) {
|
|
return;
|
|
}
|
|
if (source.mediaElement) {
|
|
source._pauseMediaElement?.();
|
|
return;
|
|
}
|
|
if (source.isPausedMockNode) {
|
|
return;
|
|
}
|
|
const pausedSource = {
|
|
isPausedMockNode: true,
|
|
loop: this.loop,
|
|
loopStart: this.loopStart,
|
|
loopEnd: this.loopEnd,
|
|
buffer: source.buffer,
|
|
playbackRate: source.playbackRateValue,
|
|
playbackPausedAtPosition: source.estimatePlaybackPosition(),
|
|
setPitch(v) {
|
|
this.playbackRate = v;
|
|
},
|
|
_reset() { },
|
|
};
|
|
this.stop(0);
|
|
this.disconnectSource();
|
|
this.source = pausedSource;
|
|
}
|
|
resume() {
|
|
if (!WEBAudio.audioContext) {
|
|
return;
|
|
}
|
|
if (!this.source) {
|
|
return;
|
|
}
|
|
if (this.source.mediaElement) {
|
|
this.source.start(this.deleyTime, this.deleyOffset);
|
|
delete this.deleyTime;
|
|
delete this.deleyOffset;
|
|
return;
|
|
}
|
|
const pausedSource = this.source;
|
|
if (!pausedSource.isPausedMockNode) {
|
|
return;
|
|
}
|
|
delete this.source;
|
|
if (!pausedSource.buffer) {
|
|
return;
|
|
}
|
|
this.playBuffer(WEBAudio.audioContext.currentTime - Math.min(0, pausedSource.playbackPausedAtPosition), pausedSource.buffer, Math.max(0, pausedSource.playbackPausedAtPosition));
|
|
const getSource = this.source;
|
|
if (getSource) {
|
|
getSource.loop = pausedSource.loop;
|
|
getSource.loopStart = pausedSource.loopStart;
|
|
getSource.loopEnd = pausedSource.loopEnd;
|
|
getSource.setPitch(pausedSource.playbackRate);
|
|
}
|
|
}
|
|
setVolume(volume, isDefault) {
|
|
if (!WEBAudio.audioContext) {
|
|
return;
|
|
}
|
|
if (WEBAudio.isMute) {
|
|
volume = 0;
|
|
}
|
|
|
|
if (isDefault && volume == 1) {
|
|
return;
|
|
}
|
|
if (this.source) {
|
|
if (this.source.buffer && this.gain) {
|
|
|
|
this.gain.gain.value = volume;
|
|
}
|
|
else if (this.source.mediaElement) {
|
|
this.source.mediaElement.volume = volume;
|
|
}
|
|
}
|
|
}
|
|
setup(url) {
|
|
if (!WEBAudio.audioContext) {
|
|
return;
|
|
}
|
|
if (this.source && !this.source.isPausedMockNode) {
|
|
if (!this.source.url) {
|
|
if (typeof url !== 'undefined') {
|
|
|
|
this.stop(0);
|
|
}
|
|
else {
|
|
|
|
}
|
|
}
|
|
else if (typeof url === 'undefined') {
|
|
if (typeof this.source !== 'undefined') {
|
|
|
|
this.source._reset();
|
|
}
|
|
this.disconnectSource();
|
|
}
|
|
else {
|
|
|
|
|
|
this.source._reset();
|
|
this.disconnectSource();
|
|
}
|
|
}
|
|
if (!url) {
|
|
this.source = WEBAudio.audioContext.createBufferSource();
|
|
WEBAudio.bufferSourceNodeLength += 1;
|
|
const { source } = this;
|
|
Object.defineProperty(this.source, 'playbackRateValue', {
|
|
get() {
|
|
return source?.playbackRate?.value ?? 0;
|
|
},
|
|
set(v) {
|
|
if (!source) {
|
|
return;
|
|
}
|
|
if (typeof source.playbackRate === 'undefined') {
|
|
return;
|
|
}
|
|
source.playbackRate.value = v;
|
|
},
|
|
});
|
|
}
|
|
else {
|
|
const { audio: getAudio, id: instanceId } = createInnerAudio();
|
|
getAudio.src = url;
|
|
const innerFixPlay = () => {
|
|
if (!this.source) {
|
|
return;
|
|
}
|
|
this.source.needCanPlay = true;
|
|
if (this.source.fixPlayTicker) {
|
|
|
|
clearTimeout(this.source.fixPlayTicker);
|
|
delete this.source.fixPlayTicker;
|
|
}
|
|
|
|
this.source.fixPlayTicker = setTimeout(() => {
|
|
if (this.source && this.source.mediaElement && this.source.needCanPlay && !this.source.isPlaying) {
|
|
this.source.mediaElement.play();
|
|
}
|
|
}, 100);
|
|
};
|
|
const innerPlay = (callback) => {
|
|
if (this.source && this.source.mediaElement) {
|
|
if (isSupportBufferURL && this.source.readyToPlay) {
|
|
if (this.source.stopCache) {
|
|
this.source.stopCache = false;
|
|
this.source.playAfterStop = true;
|
|
}
|
|
else if (!this.source.isPlaying) {
|
|
|
|
if (isAndroid) {
|
|
innerFixPlay();
|
|
}
|
|
this.source.mediaElement.play();
|
|
callback?.();
|
|
}
|
|
}
|
|
else {
|
|
const fn = () => {
|
|
if (!this.source) {
|
|
return;
|
|
}
|
|
this.source.needCanPlay = false;
|
|
this.source.readyToPlay = true;
|
|
if (typeof this.source.mediaElement !== 'undefined') {
|
|
// @ts-ignore
|
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
const { duration } = this.source.mediaElement;
|
|
|
|
|
|
this.source.canPlayFnList.forEach((fn) => {
|
|
this.source?.mediaElement?.offCanplay(fn);
|
|
});
|
|
this.source.canPlayFnList = [];
|
|
}
|
|
if (this.source.stopCache) {
|
|
this.source.stopCache = false;
|
|
this.source.playAfterStop = true;
|
|
}
|
|
else if (!this.source.isPlaying) {
|
|
|
|
if (isAndroid) {
|
|
innerFixPlay();
|
|
}
|
|
if (typeof this.source.mediaElement !== 'undefined') {
|
|
this.source.mediaElement.play();
|
|
callback?.();
|
|
}
|
|
}
|
|
};
|
|
if (!this.source.canPlayFnList) {
|
|
this.source.canPlayFnList = [];
|
|
}
|
|
this.source.canPlayFnList.push(fn);
|
|
this.source.mediaElement.onCanplay(fn);
|
|
innerFixPlay();
|
|
}
|
|
}
|
|
};
|
|
// eslint-disable-next-line @typescript-eslint/naming-convention
|
|
const _reset = () => {
|
|
if (!this.source) {
|
|
return;
|
|
}
|
|
this.source.readyToPlay = false;
|
|
this.source.isPlaying = false;
|
|
this.source.stopCache = false;
|
|
this.source.playAfterStop = false;
|
|
this.source.needCanPlay = false;
|
|
if (this.source.stopTicker) {
|
|
clearTimeout(this.source.stopTicker);
|
|
this.source.stopTicker = undefined;
|
|
}
|
|
};
|
|
// eslint-disable-next-line @typescript-eslint/naming-convention
|
|
const _pauseMediaElement = () => {
|
|
if (typeof this.source === 'undefined') {
|
|
return;
|
|
}
|
|
if (this.source.playTimeout) {
|
|
this.source.pauseRequested = true;
|
|
}
|
|
else if (this.source.isPlaying && this.source.mediaElement) {
|
|
this.source.mediaElement.pause();
|
|
}
|
|
};
|
|
// eslint-disable-next-line @typescript-eslint/naming-convention
|
|
const _startPlayback = (offset) => {
|
|
if (typeof this.source === 'undefined' || !this.source.mediaElement) {
|
|
return;
|
|
}
|
|
if (this.source.playTimeout) {
|
|
this.source.mediaElement.seek(offset);
|
|
this.source.pauseRequested = false;
|
|
return;
|
|
}
|
|
innerPlay(() => {
|
|
|
|
if (this.source && this.source.mediaElement) {
|
|
this.source.mediaElement.seek(offset);
|
|
}
|
|
});
|
|
};
|
|
const start = (startTime, offset) => {
|
|
if (typeof this.source === 'undefined') {
|
|
return;
|
|
}
|
|
if (typeof startTime === 'undefined' && typeof offset === 'undefined') {
|
|
innerPlay();
|
|
return;
|
|
}
|
|
if (typeof startTime === 'undefined') {
|
|
startTime = 0;
|
|
}
|
|
if (typeof offset === 'undefined') {
|
|
offset = 0;
|
|
}
|
|
const startDelayThresholdMS = 4;
|
|
const startDelayMS = startTime * 1e3;
|
|
if (startDelayMS > startDelayThresholdMS) {
|
|
if (this.source.playTimeout) {
|
|
clearTimeout(this.source.playTimeout);
|
|
}
|
|
this.source.playTimeout = setTimeout(() => {
|
|
if (typeof this.source !== 'undefined') {
|
|
delete this.source.playTimeout;
|
|
this.source._startPlayback?.(offset || 0);
|
|
}
|
|
}, startDelayMS);
|
|
}
|
|
else {
|
|
this.source._startPlayback?.(offset);
|
|
}
|
|
};
|
|
const stop = (stopTime) => {
|
|
if (typeof this.source === 'undefined') {
|
|
return;
|
|
}
|
|
if (typeof stopTime === 'undefined') {
|
|
stopTime = 0;
|
|
}
|
|
const stopDelayThresholdMS = 4;
|
|
const stopDelayMS = stopTime * 1e3;
|
|
if (stopDelayMS > stopDelayThresholdMS) {
|
|
setTimeout(() => {
|
|
if (this.source && this.source.mediaElement) {
|
|
this.source.stopCache = true;
|
|
this.source.mediaElement.stop();
|
|
}
|
|
}, stopDelayMS);
|
|
}
|
|
else if (this.source.mediaElement) {
|
|
this.source.stopCache = true;
|
|
this.source.mediaElement.stop();
|
|
}
|
|
};
|
|
this.source = {
|
|
instanceId,
|
|
mediaElement: getAudio,
|
|
url,
|
|
playbackStartTime: 0,
|
|
playbackRate: 1,
|
|
pauseRequested: false,
|
|
_reset,
|
|
_pauseMediaElement,
|
|
_startPlayback,
|
|
start,
|
|
stop,
|
|
};
|
|
// @ts-ignore
|
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
const { buffered, referrerPolicy, volume } = getAudio;
|
|
const { source } = this;
|
|
Object.defineProperty(this.source, 'loopStart', {
|
|
get() {
|
|
return 0;
|
|
},
|
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
set(v) { },
|
|
});
|
|
Object.defineProperty(source, 'loopEnd', {
|
|
get() {
|
|
return 0;
|
|
},
|
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
set(v) { },
|
|
});
|
|
Object.defineProperty(source, 'loop', {
|
|
get() {
|
|
return source?.mediaElement?.loop ?? false;
|
|
},
|
|
set(v) {
|
|
if (!source || !source.mediaElement) {
|
|
return;
|
|
}
|
|
source.mediaElement.loop = v;
|
|
},
|
|
});
|
|
Object.defineProperty(source, 'playbackRateValue', {
|
|
get() {
|
|
|
|
return source?.playbackRate ?? 1;
|
|
},
|
|
set(v) {
|
|
if (!source || !source.mediaElement) {
|
|
return;
|
|
}
|
|
|
|
if (!isSupportPlayBackRate) {
|
|
source.mediaElement.playbackRate = 1;
|
|
}
|
|
else {
|
|
source.playbackRate = v;
|
|
source.mediaElement.playbackRate = v;
|
|
}
|
|
},
|
|
});
|
|
Object.defineProperty(source, 'currentTime', {
|
|
get() {
|
|
return source?.mediaElement?.currentTime ?? 0;
|
|
},
|
|
set(v) {
|
|
if (!source || !source.mediaElement) {
|
|
return;
|
|
}
|
|
if (typeof source.mediaElement.seek === 'function') {
|
|
source.mediaElement.seek(v);
|
|
}
|
|
else {
|
|
source.mediaElement.currentTime = v;
|
|
}
|
|
},
|
|
});
|
|
}
|
|
if (!this.source) {
|
|
return;
|
|
}
|
|
this.source.estimatePlaybackPosition = () => {
|
|
if (!this.source) {
|
|
return 0;
|
|
}
|
|
let t;
|
|
if (WEBAudio.audioContext) {
|
|
t = (WEBAudio.audioContext.currentTime - this.source.playbackStartTime) * this.source.playbackRateValue;
|
|
}
|
|
else {
|
|
t = -this.source.playbackStartTime * this.source.playbackRateValue;
|
|
}
|
|
if (typeof this.source.loopStart !== 'undefined' && typeof this.source.loopEnd !== 'undefined') {
|
|
if (this.source.loop && t >= this.source.loopStart) {
|
|
t = ((t - this.source.loopStart) % (this.source.loopEnd - this.source.loopStart)) + this.source.loopStart;
|
|
}
|
|
}
|
|
return t;
|
|
};
|
|
this.source.setPitch = (newPitch) => {
|
|
if (!this.source) {
|
|
return 0;
|
|
}
|
|
const curPosition = this.source.estimatePlaybackPosition();
|
|
if (curPosition >= 0) {
|
|
if (WEBAudio.audioContext) {
|
|
this.source.playbackStartTime = WEBAudio.audioContext.currentTime - curPosition / newPitch;
|
|
}
|
|
}
|
|
this.source.playbackRateValue = newPitch;
|
|
};
|
|
this.setupPanning();
|
|
}
|
|
setupPanning() {
|
|
if (typeof this.source === 'undefined') {
|
|
return;
|
|
}
|
|
if (this.source.isPausedMockNode) {
|
|
return;
|
|
}
|
|
if (this.source.disconnect && this.source.connect) {
|
|
this.source.disconnect();
|
|
if (this.gain) {
|
|
this.source.connect(this.gain);
|
|
}
|
|
}
|
|
}
|
|
isStopped() {
|
|
return !this.source;
|
|
}
|
|
}
|
|
export default {
|
|
_JS_Sound_Create_Channel(callback, userData) {
|
|
if (!WEBAudio.audioContext || WEBAudio.audioWebEnabled === 0) {
|
|
return 0;
|
|
}
|
|
const channel = new AudioChannelInstance(callback, userData);
|
|
WEBAudio.audioInstances[++WEBAudio.audioInstanceIdCounter] = channel;
|
|
return WEBAudio.audioInstanceIdCounter;
|
|
},
|
|
_JS_Sound_GetLength(bufferInstance) {
|
|
if (WEBAudio.audioWebEnabled === 0) {
|
|
return defaultSoundLength;
|
|
}
|
|
const soundClip = WEBAudio.audioInstances[bufferInstance];
|
|
if (!soundClip) {
|
|
return defaultSoundLength;
|
|
}
|
|
const length = soundClip.getLength();
|
|
return length;
|
|
},
|
|
_JS_Sound_GetLoadState(bufferInstance) {
|
|
if (WEBAudio.audioWebEnabled === 0) {
|
|
return 2;
|
|
}
|
|
const soundClip = WEBAudio.audioInstances[bufferInstance];
|
|
if (!soundClip || soundClip.error) {
|
|
return 2;
|
|
}
|
|
if (soundClip.buffer) {
|
|
return 0;
|
|
}
|
|
if (soundClip.url && soundClip.length) {
|
|
return 0;
|
|
}
|
|
return 1;
|
|
},
|
|
_JS_Sound_Init() {
|
|
try {
|
|
if (wx && wx.createWebAudioContext) {
|
|
WEBAudio.audioContext = wx.createWebAudioContext();
|
|
console.log('use wx WebAudio');
|
|
}
|
|
if (!WEBAudio.audioContext) {
|
|
printErrMsg('Minigame Web Audio API not suppoted');
|
|
return;
|
|
}
|
|
WEBAudio.audioWebSupport = 1;
|
|
WEBAudio.audioWebEnabled = 1;
|
|
let webAutoResumeTicker = null;
|
|
wx.onHide(() => {
|
|
if (webAutoResumeTicker) {
|
|
clearTimeout(webAutoResumeTicker);
|
|
webAutoResumeTicker = null;
|
|
}
|
|
WEBAudio.audioContext?.suspend();
|
|
});
|
|
wx.onShow(() => {
|
|
|
|
if (isIOS175) {
|
|
WEBAudio.audioContext?.close();
|
|
WEBAudio.audioContext = wx.createWebAudioContext();
|
|
Object.values(WEBAudio.audioInstances).forEach(it => it.resetGain());
|
|
}
|
|
else {
|
|
WEBAudio.audioContext?.resume();
|
|
}
|
|
});
|
|
if (webAudioNeedResume) {
|
|
|
|
webAutoResumeTicker = setTimeout(() => {
|
|
resumeWebAudio();
|
|
}, 2000);
|
|
}
|
|
}
|
|
catch (e) {
|
|
printErrMsg('Web Audio API is not supported in this browser');
|
|
}
|
|
},
|
|
_JS_Sound_IsStopped(channelInstance) {
|
|
if (WEBAudio.audioWebEnabled == 0) {
|
|
return true;
|
|
}
|
|
const channel = WEBAudio.audioInstances[channelInstance];
|
|
if (!channel) {
|
|
return true;
|
|
}
|
|
return channel.isStopped();
|
|
},
|
|
_JS_Sound_Load(ptr, length, decompress) {
|
|
if (!WEBAudio.audioContext || WEBAudio.audioWebEnabled === 0) {
|
|
return 0;
|
|
}
|
|
const audioData = GameGlobal.unityNamespace.Module.HEAPU8.buffer.slice(ptr, ptr + length);
|
|
|
|
if (length > 131072) {
|
|
decompress = 0;
|
|
}
|
|
else {
|
|
decompress = 1;
|
|
}
|
|
|
|
if (isPc) {
|
|
decompress = 1;
|
|
}
|
|
|
|
if (isAndroid && !isSupportInnerAudio) {
|
|
decompress = 1;
|
|
}
|
|
let soundClip;
|
|
if (decompress && WEBAudio.audioWebSupport) {
|
|
soundClip = jsAudioCreateUncompressedSoundClipFromCompressedAudio(audioData, ptr, length);
|
|
}
|
|
else {
|
|
soundClip = jsAudioCreateCompressedSoundClip(audioData, ptr, length);
|
|
}
|
|
WEBAudio.audioInstances[++WEBAudio.audioInstanceIdCounter] = soundClip;
|
|
return WEBAudio.audioInstanceIdCounter;
|
|
},
|
|
_JS_Sound_Load_PCM(channels, length, sampleRate, ptr) {
|
|
if (!WEBAudio.audioContext || WEBAudio.audioWebSupport === 0 || WEBAudio.audioWebEnabled === 0) {
|
|
return 0;
|
|
}
|
|
const sound = jsAudioCreateUncompressedSoundClipFromPCM(channels, length, sampleRate, ptr);
|
|
WEBAudio.audioInstances[++WEBAudio.audioInstanceIdCounter] = sound;
|
|
return WEBAudio.audioInstanceIdCounter;
|
|
},
|
|
_JS_Sound_Play(bufferInstance, channelInstance, offset, delay) {
|
|
if (!WEBAudio.audioContext || WEBAudio.audioWebEnabled === 0) {
|
|
return;
|
|
}
|
|
WXWASMSDK._JS_Sound_Stop(channelInstance, 0);
|
|
const soundClip = WEBAudio.audioInstances[bufferInstance];
|
|
const channel = WEBAudio.audioInstances[channelInstance];
|
|
if (soundClip && soundClip.url) {
|
|
try {
|
|
channel.playUrl(delay, soundClip.url, offset, unityAudioVolume.get(channel), soundClip);
|
|
}
|
|
catch (e) {
|
|
printErrMsg(`playUrl error. Exception: ${e}`);
|
|
}
|
|
}
|
|
else if (soundClip && soundClip.buffer) {
|
|
try {
|
|
channel.playBuffer(WEBAudio.audioContext.currentTime + delay, soundClip.buffer, offset, channel);
|
|
}
|
|
catch (e) {
|
|
printErrMsg(`playBuffer error. Exception: ${e}`);
|
|
}
|
|
}
|
|
else {
|
|
console.log('Trying to play sound which is not loaded.');
|
|
}
|
|
},
|
|
_JS_Sound_ReleaseInstance(instance) {
|
|
if (WEBAudio.audioWebEnabled === 0) {
|
|
return;
|
|
}
|
|
const object = WEBAudio.audioInstances[instance];
|
|
if (object) {
|
|
object.release();
|
|
}
|
|
delete WEBAudio.audioInstances[instance];
|
|
},
|
|
_JS_Sound_ResumeIfNeeded() {
|
|
if (WEBAudio.audioWebSupport === 0 || WEBAudio.audioWebEnabled === 0) {
|
|
return;
|
|
}
|
|
resumeWebAudio();
|
|
},
|
|
_JS_Sound_Set3D(channelInstance, threeD) {
|
|
if (WEBAudio.audio3DSupport === 0 || WEBAudio.audioWebEnabled === 0) {
|
|
return;
|
|
}
|
|
const channel = WEBAudio.audioInstances[channelInstance];
|
|
if (channel.threeD != threeD) {
|
|
channel.threeD = threeD;
|
|
if (!channel.source) {
|
|
channel.setup();
|
|
}
|
|
channel.setupPanning();
|
|
}
|
|
},
|
|
_JS_Sound_SetListenerOrientation(x, y, z, xUp, yUp, zUp) {
|
|
if (!WEBAudio.audioContext
|
|
|| WEBAudio.audio3DSupport === 0
|
|
|| WEBAudio.audioWebSupport === 0
|
|
|| WEBAudio.audioWebEnabled === 0) {
|
|
return;
|
|
}
|
|
x = x > 0 ? 0 : x;
|
|
y = y > 0 ? 0 : y;
|
|
z = z > 0 ? 0 : z;
|
|
xUp = xUp < 0 ? 0 : xUp;
|
|
yUp = yUp < 0 ? 0 : yUp;
|
|
zUp = zUp < 0 ? 0 : zUp;
|
|
if (x == WEBAudio.lOrientation.x
|
|
&& y == WEBAudio.lOrientation.y
|
|
&& z == WEBAudio.lOrientation.z
|
|
&& xUp == WEBAudio.lOrientation.xUp
|
|
&& yUp == WEBAudio.lOrientation.yUp
|
|
&& zUp == WEBAudio.lOrientation.zUp) {
|
|
return;
|
|
}
|
|
WEBAudio.lOrientation.x = x;
|
|
WEBAudio.lOrientation.y = y;
|
|
WEBAudio.lOrientation.z = z;
|
|
WEBAudio.lOrientation.xUp = xUp;
|
|
WEBAudio.lOrientation.yUp = yUp;
|
|
WEBAudio.lOrientation.zUp = zUp;
|
|
if (WEBAudio.audioContext.listener.forwardX) {
|
|
WEBAudio.audioContext.listener.forwardX = -x;
|
|
WEBAudio.audioContext.listener.forwardY = -y;
|
|
WEBAudio.audioContext.listener.forwardZ = -z;
|
|
WEBAudio.audioContext.listener.upX = xUp;
|
|
WEBAudio.audioContext.listener.upY = yUp;
|
|
WEBAudio.audioContext.listener.upZ = zUp;
|
|
}
|
|
else {
|
|
WEBAudio.audioContext.listener.setOrientation(-x, -y, -z, xUp, yUp, zUp);
|
|
}
|
|
},
|
|
_JS_Sound_SetListenerPosition(x, y, z) {
|
|
if (!WEBAudio.audioContext
|
|
|| WEBAudio.audio3DSupport === 0
|
|
|| WEBAudio.audioWebSupport === 0
|
|
|| WEBAudio.audioWebEnabled === 0) {
|
|
return;
|
|
}
|
|
x = x < 0 ? 0 : x;
|
|
y = y < 0 ? 0 : y;
|
|
z = z < 0 ? 0 : z;
|
|
if (x == WEBAudio.lPosition.x && y == WEBAudio.lPosition.y && z == WEBAudio.lPosition.z) {
|
|
return;
|
|
}
|
|
WEBAudio.lPosition.x = x;
|
|
WEBAudio.lPosition.y = y;
|
|
WEBAudio.lPosition.z = z;
|
|
if (WEBAudio.audioContext.listener.positionX) {
|
|
WEBAudio.audioContext.listener.positionX = x;
|
|
WEBAudio.audioContext.listener.positionY = y;
|
|
WEBAudio.audioContext.listener.positionZ = z;
|
|
}
|
|
else {
|
|
WEBAudio.audioContext.listener.setPosition(x, y, z);
|
|
}
|
|
},
|
|
_JS_Sound_SetLoop(channelInstance, loop) {
|
|
if (WEBAudio.audioWebEnabled === 0) {
|
|
return;
|
|
}
|
|
const channel = WEBAudio.audioInstances[channelInstance];
|
|
if (!channel.source) {
|
|
channel.setup();
|
|
}
|
|
if (!channel.source) {
|
|
return;
|
|
}
|
|
channel.setLoop(loop > 0);
|
|
},
|
|
_JS_Sound_SetLoopPoints(channelInstance, loopStart, loopEnd) {
|
|
if (WEBAudio.audioWebEnabled === 0) {
|
|
return;
|
|
}
|
|
const channel = WEBAudio.audioInstances[channelInstance];
|
|
if (!channel.source) {
|
|
channel.setup();
|
|
}
|
|
if (!channel.source) {
|
|
return;
|
|
}
|
|
channel.setLoopPoints(loopStart, loopEnd);
|
|
},
|
|
_JS_Sound_SetPaused(channelInstance, paused) {
|
|
if (WEBAudio.audioWebEnabled === 0) {
|
|
return;
|
|
}
|
|
const channel = WEBAudio.audioInstances[channelInstance];
|
|
if (!!paused !== channel.isPaused()) {
|
|
if (paused) {
|
|
channel.pause();
|
|
}
|
|
else {
|
|
channel.resume();
|
|
}
|
|
}
|
|
},
|
|
_JS_Sound_SetPitch(channelInstance, v) {
|
|
if (WEBAudio.audioWebSupport === 0 || WEBAudio.audioWebEnabled === 0) {
|
|
return;
|
|
}
|
|
try {
|
|
WEBAudio.audioInstances[channelInstance].source?.setPitch(v);
|
|
}
|
|
catch (e) {
|
|
printErrMsg(`Invalid audio pitch ${v} specified to WebAudio backend!`);
|
|
}
|
|
},
|
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
_JS_Sound_SetPosition(channelInstance, x, y, z) {
|
|
if (WEBAudio.audio3DSupport === 0 || WEBAudio.audioWebSupport === 0 || WEBAudio.audioWebEnabled === 0) {
|
|
return;
|
|
}
|
|
console.error('不支持3d音效');
|
|
},
|
|
_JS_Sound_SetVolume(channelInstance, v) {
|
|
if (WEBAudio.audioWebEnabled === 0) {
|
|
return;
|
|
}
|
|
try {
|
|
const volume = Number(v.toFixed(2));
|
|
const channel = WEBAudio.audioInstances[channelInstance];
|
|
const cur = unityAudioVolume.get(channel);
|
|
if (cur === volume) {
|
|
return;
|
|
}
|
|
unityAudioVolume.set(channel, volume);
|
|
channel.setVolume(volume, cur == undefined);
|
|
}
|
|
catch (e) {
|
|
printErrMsg(`Invalid audio volume ${v} specified to WebAudio backend!`);
|
|
}
|
|
},
|
|
_JS_Sound_Stop(channelInstance, delay) {
|
|
if (WEBAudio.audioWebEnabled === 0) {
|
|
return;
|
|
}
|
|
const channel = WEBAudio.audioInstances[channelInstance];
|
|
channel.stop(delay);
|
|
},
|
|
_JS_Sound_GetData(bufferInstance, ptr, length) {
|
|
if (WEBAudio.audioWebEnabled === 0) {
|
|
return 0;
|
|
}
|
|
const soundClip = WEBAudio.audioInstances[bufferInstance];
|
|
if (!soundClip) {
|
|
return 0;
|
|
}
|
|
return soundClip.getData(ptr, length) ?? 0;
|
|
},
|
|
_JS_Sound_GetMetaData(buffer, bufferInstance, metaData) {
|
|
if (WEBAudio.audioWebEnabled === 0) {
|
|
buffer[metaData >> 2] = 0;
|
|
buffer[(metaData >> 2) + 1] = 0;
|
|
return false;
|
|
}
|
|
const soundClip = WEBAudio.audioInstances[bufferInstance];
|
|
if (!soundClip) {
|
|
buffer[metaData >> 2] = 0;
|
|
buffer[(metaData >> 2) + 1] = 0;
|
|
return false;
|
|
}
|
|
buffer[metaData >> 2] = soundClip.getNumberOfChannels() ?? 0;
|
|
buffer[(metaData >> 2) + 1] = soundClip.getFrequency() ?? 0;
|
|
return true;
|
|
},
|
|
_JS_Sound_GetAudioBufferSampleRate(soundInstance) {
|
|
if (WEBAudio.audioWebEnabled === 0) {
|
|
return WEBAudio.FAKEMOD_SAMPLERATE;
|
|
}
|
|
const audioInstance = WEBAudio.audioInstances[soundInstance];
|
|
if (!audioInstance) {
|
|
return WEBAudio.FAKEMOD_SAMPLERATE;
|
|
}
|
|
// eslint-disable-next-line no-nested-ternary
|
|
const buffer = audioInstance.buffer
|
|
? audioInstance.buffer
|
|
: audioInstance.source
|
|
? audioInstance.source?.buffer
|
|
: null;
|
|
if (!buffer) {
|
|
return WEBAudio.FAKEMOD_SAMPLERATE;
|
|
}
|
|
return buffer.sampleRate;
|
|
},
|
|
_JS_Sound_GetAudioContextSampleRate() {
|
|
if (WEBAudio.audioWebEnabled === 0 || !WEBAudio.audioContext) {
|
|
return WEBAudio.FAKEMOD_SAMPLERATE;
|
|
}
|
|
return WEBAudio.audioContext.sampleRate;
|
|
},
|
|
};
|