2023-07-27 14:31:52 +09:00
|
|
|
|
/*
|
2024-02-13 15:59:27 +00:00
|
|
|
|
* SPDX-FileCopyrightText: syuilo and misskey-project
|
2023-07-27 14:31:52 +09:00
|
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
*/
|
|
|
|
|
|
2023-11-27 17:33:42 +09:00
|
|
|
|
import type { SoundStore } from '@/store.js';
|
2023-11-04 10:09:21 +09:00
|
|
|
|
import { defaultStore } from '@/store.js';
|
2020-11-25 21:31:34 +09:00
|
|
|
|
|
2023-11-26 14:38:34 +09:00
|
|
|
|
let ctx: AudioContext;
|
2023-11-15 18:03:15 +09:00
|
|
|
|
const cache = new Map<string, AudioBuffer>();
|
2023-11-26 13:20:46 +09:00
|
|
|
|
let canPlay = true;
|
2020-11-25 21:31:34 +09:00
|
|
|
|
|
2022-12-22 09:01:58 +09:00
|
|
|
|
export const soundsTypes = [
|
2023-11-27 17:33:42 +09:00
|
|
|
|
// 音声なし
|
2022-12-22 09:01:58 +09:00
|
|
|
|
null,
|
2023-11-27 17:33:42 +09:00
|
|
|
|
|
|
|
|
|
// ドライブの音声
|
|
|
|
|
'_driveFile_',
|
|
|
|
|
|
|
|
|
|
// プリインストール
|
2023-03-03 09:41:33 +09:00
|
|
|
|
'syuilo/n-aec',
|
|
|
|
|
'syuilo/n-aec-4va',
|
|
|
|
|
'syuilo/n-aec-4vb',
|
|
|
|
|
'syuilo/n-aec-8va',
|
|
|
|
|
'syuilo/n-aec-8vb',
|
|
|
|
|
'syuilo/n-cea',
|
|
|
|
|
'syuilo/n-cea-4va',
|
|
|
|
|
'syuilo/n-cea-4vb',
|
|
|
|
|
'syuilo/n-cea-8va',
|
|
|
|
|
'syuilo/n-cea-8vb',
|
|
|
|
|
'syuilo/n-eca',
|
|
|
|
|
'syuilo/n-eca-4va',
|
|
|
|
|
'syuilo/n-eca-4vb',
|
|
|
|
|
'syuilo/n-eca-8va',
|
|
|
|
|
'syuilo/n-eca-8vb',
|
|
|
|
|
'syuilo/n-ea',
|
|
|
|
|
'syuilo/n-ea-4va',
|
|
|
|
|
'syuilo/n-ea-4vb',
|
|
|
|
|
'syuilo/n-ea-8va',
|
|
|
|
|
'syuilo/n-ea-8vb',
|
|
|
|
|
'syuilo/n-ea-harmony',
|
2022-12-22 09:01:58 +09:00
|
|
|
|
'syuilo/up',
|
|
|
|
|
'syuilo/down',
|
|
|
|
|
'syuilo/pope1',
|
|
|
|
|
'syuilo/pope2',
|
|
|
|
|
'syuilo/waon',
|
|
|
|
|
'syuilo/popo',
|
|
|
|
|
'syuilo/triple',
|
2023-11-26 13:04:44 +09:00
|
|
|
|
'syuilo/bubble1',
|
|
|
|
|
'syuilo/bubble2',
|
2022-12-22 09:01:58 +09:00
|
|
|
|
'syuilo/poi1',
|
|
|
|
|
'syuilo/poi2',
|
|
|
|
|
'syuilo/pirori',
|
|
|
|
|
'syuilo/pirori-wet',
|
|
|
|
|
'syuilo/pirori-square-wet',
|
|
|
|
|
'syuilo/square-pico',
|
|
|
|
|
'syuilo/reverved',
|
|
|
|
|
'syuilo/ryukyu',
|
|
|
|
|
'syuilo/kick',
|
|
|
|
|
'syuilo/snare',
|
|
|
|
|
'syuilo/queue-jammed',
|
|
|
|
|
'aisha/1',
|
|
|
|
|
'aisha/2',
|
|
|
|
|
'aisha/3',
|
|
|
|
|
'noizenecio/kick_gaba1',
|
|
|
|
|
'noizenecio/kick_gaba2',
|
|
|
|
|
'noizenecio/kick_gaba3',
|
|
|
|
|
'noizenecio/kick_gaba4',
|
|
|
|
|
'noizenecio/kick_gaba5',
|
|
|
|
|
'noizenecio/kick_gaba6',
|
|
|
|
|
'noizenecio/kick_gaba7',
|
|
|
|
|
] as const;
|
|
|
|
|
|
2023-11-27 17:33:42 +09:00
|
|
|
|
export const operationTypes = [
|
|
|
|
|
'noteMy',
|
|
|
|
|
'note',
|
|
|
|
|
'antenna',
|
|
|
|
|
'channel',
|
|
|
|
|
'notification',
|
|
|
|
|
'reaction',
|
|
|
|
|
] as const;
|
|
|
|
|
|
|
|
|
|
/** サウンドの種類 */
|
|
|
|
|
export type SoundType = typeof soundsTypes[number];
|
|
|
|
|
|
|
|
|
|
/** スプライトの種類 */
|
|
|
|
|
export type OperationType = typeof operationTypes[number];
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* 音声を読み込む
|
2024-01-08 12:46:20 +09:00
|
|
|
|
* @param url url
|
2023-11-27 17:33:42 +09:00
|
|
|
|
* @param options `useCache`: デフォルトは`true` 一度再生した音声はキャッシュする
|
|
|
|
|
*/
|
2024-01-08 12:46:20 +09:00
|
|
|
|
export async function loadAudio(url: string, options?: { useCache?: boolean; }) {
|
2023-11-27 17:33:42 +09:00
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
2023-11-26 14:38:34 +09:00
|
|
|
|
if (ctx == null) {
|
|
|
|
|
ctx = new AudioContext();
|
|
|
|
|
}
|
2023-11-27 17:33:42 +09:00
|
|
|
|
if (options?.useCache ?? true) {
|
2024-01-08 12:46:20 +09:00
|
|
|
|
if (cache.has(url)) {
|
|
|
|
|
return cache.get(url) as AudioBuffer;
|
2023-11-27 17:33:42 +09:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let response: Response;
|
|
|
|
|
|
2024-01-08 12:46:20 +09:00
|
|
|
|
try {
|
|
|
|
|
response = await fetch(url);
|
|
|
|
|
} catch (err) {
|
|
|
|
|
return;
|
2021-08-14 18:36:22 +09:00
|
|
|
|
}
|
2023-11-15 18:03:15 +09:00
|
|
|
|
|
|
|
|
|
const arrayBuffer = await response.arrayBuffer();
|
|
|
|
|
const audioBuffer = await ctx.decodeAudioData(arrayBuffer);
|
|
|
|
|
|
2023-11-27 17:33:42 +09:00
|
|
|
|
if (options?.useCache ?? true) {
|
2024-01-08 12:46:20 +09:00
|
|
|
|
cache.set(url, audioBuffer);
|
2023-11-15 18:03:15 +09:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return audioBuffer;
|
2021-08-14 18:36:22 +09:00
|
|
|
|
}
|
|
|
|
|
|
2023-11-27 17:33:42 +09:00
|
|
|
|
/**
|
|
|
|
|
* 既定のスプライトを再生する
|
|
|
|
|
* @param type スプライトの種類を指定
|
|
|
|
|
*/
|
2024-01-09 13:25:33 +09:00
|
|
|
|
export function playMisskeySfx(operationType: OperationType) {
|
2023-11-27 17:33:42 +09:00
|
|
|
|
const sound = defaultStore.state[`sound_${operationType}`];
|
2024-02-20 15:26:11 +09:00
|
|
|
|
if (sound.type == null || !canPlay || !navigator.userActivation.hasBeenActive) return;
|
2023-11-26 13:20:46 +09:00
|
|
|
|
|
|
|
|
|
canPlay = false;
|
2024-01-09 13:25:33 +09:00
|
|
|
|
playMisskeySfxFile(sound).finally(() => {
|
2023-11-26 13:20:46 +09:00
|
|
|
|
// ごく短時間に音が重複しないように
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
canPlay = true;
|
|
|
|
|
}, 25);
|
|
|
|
|
});
|
2020-11-25 21:31:34 +09:00
|
|
|
|
}
|
|
|
|
|
|
2023-11-27 17:33:42 +09:00
|
|
|
|
/**
|
|
|
|
|
* サウンド設定形式で指定された音声を再生する
|
|
|
|
|
* @param soundStore サウンド設定
|
|
|
|
|
*/
|
2024-01-09 13:25:33 +09:00
|
|
|
|
export async function playMisskeySfxFile(soundStore: SoundStore) {
|
2024-01-08 12:46:20 +09:00
|
|
|
|
if (soundStore.type === null || (soundStore.type === '_driveFile_' && !soundStore.fileUrl)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2024-01-09 13:25:33 +09:00
|
|
|
|
const masterVolume = defaultStore.state.sound_masterVolume;
|
|
|
|
|
if (isMute() || masterVolume === 0 || soundStore.volume === 0) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2024-01-08 12:46:20 +09:00
|
|
|
|
const url = soundStore.type === '_driveFile_' ? soundStore.fileUrl : `/client-assets/sounds/${soundStore.type}.mp3`;
|
|
|
|
|
const buffer = await loadAudio(url);
|
2023-11-27 17:33:42 +09:00
|
|
|
|
if (!buffer) return;
|
2024-01-09 13:25:33 +09:00
|
|
|
|
const volume = soundStore.volume * masterVolume;
|
|
|
|
|
createSourceNode(buffer, { volume }).soundSource.start();
|
2023-11-21 20:05:04 +09:00
|
|
|
|
}
|
|
|
|
|
|
2024-01-09 13:25:33 +09:00
|
|
|
|
export async function playUrl(url: string, opts: {
|
|
|
|
|
volume?: number;
|
|
|
|
|
pan?: number;
|
|
|
|
|
playbackRate?: number;
|
|
|
|
|
}) {
|
|
|
|
|
if (opts.volume === 0) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2024-01-08 12:46:20 +09:00
|
|
|
|
const buffer = await loadAudio(url);
|
2024-01-06 20:15:28 +09:00
|
|
|
|
if (!buffer) return;
|
2024-01-09 13:25:33 +09:00
|
|
|
|
createSourceNode(buffer, opts).soundSource.start();
|
2024-01-06 20:15:28 +09:00
|
|
|
|
}
|
|
|
|
|
|
2024-01-09 13:25:33 +09:00
|
|
|
|
export function createSourceNode(buffer: AudioBuffer, opts: {
|
|
|
|
|
volume?: number;
|
|
|
|
|
pan?: number;
|
|
|
|
|
playbackRate?: number;
|
|
|
|
|
}): {
|
2024-01-08 12:46:20 +09:00
|
|
|
|
soundSource: AudioBufferSourceNode;
|
|
|
|
|
panNode: StereoPannerNode;
|
|
|
|
|
gainNode: GainNode;
|
2024-01-09 13:25:33 +09:00
|
|
|
|
} {
|
2024-01-06 20:15:28 +09:00
|
|
|
|
const panNode = ctx.createStereoPanner();
|
2024-01-09 13:25:33 +09:00
|
|
|
|
panNode.pan.value = opts.pan ?? 0;
|
2024-01-06 20:15:28 +09:00
|
|
|
|
|
2023-11-15 18:03:15 +09:00
|
|
|
|
const gainNode = ctx.createGain();
|
2024-01-09 13:25:33 +09:00
|
|
|
|
|
|
|
|
|
gainNode.gain.value = opts.volume ?? 1;
|
2023-11-15 18:03:15 +09:00
|
|
|
|
|
|
|
|
|
const soundSource = ctx.createBufferSource();
|
2023-11-21 20:05:04 +09:00
|
|
|
|
soundSource.buffer = buffer;
|
2024-01-09 13:25:33 +09:00
|
|
|
|
soundSource.playbackRate.value = opts.playbackRate ?? 1;
|
2024-01-06 20:15:28 +09:00
|
|
|
|
soundSource
|
|
|
|
|
.connect(panNode)
|
|
|
|
|
.connect(gainNode)
|
|
|
|
|
.connect(ctx.destination);
|
2023-11-21 20:05:04 +09:00
|
|
|
|
|
2024-01-08 12:46:20 +09:00
|
|
|
|
return { soundSource, panNode, gainNode };
|
2020-11-25 21:31:34 +09:00
|
|
|
|
}
|
2023-11-26 16:12:02 +09:00
|
|
|
|
|
2023-11-27 17:33:42 +09:00
|
|
|
|
/**
|
|
|
|
|
* 音声の長さをミリ秒で取得する
|
|
|
|
|
* @param file ファイルのURL(ドライブIDではない)
|
|
|
|
|
*/
|
|
|
|
|
export async function getSoundDuration(file: string): Promise<number> {
|
|
|
|
|
const audioEl = document.createElement('audio');
|
|
|
|
|
audioEl.src = file;
|
|
|
|
|
return new Promise((resolve) => {
|
|
|
|
|
const si = setInterval(() => {
|
|
|
|
|
if (audioEl.readyState > 0) {
|
|
|
|
|
resolve(audioEl.duration * 1000);
|
|
|
|
|
clearInterval(si);
|
|
|
|
|
audioEl.remove();
|
|
|
|
|
}
|
|
|
|
|
}, 100);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* ミュートすべきかどうかを判断する
|
|
|
|
|
*/
|
2023-11-26 16:12:02 +09:00
|
|
|
|
export function isMute(): boolean {
|
|
|
|
|
if (defaultStore.state.sound_notUseSound) {
|
|
|
|
|
// サウンドを出力しない
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// noinspection RedundantIfStatementJS
|
|
|
|
|
if (defaultStore.state.sound_useSoundOnlyWhenActive && document.visibilityState === 'hidden') {
|
|
|
|
|
// ブラウザがアクティブな時のみサウンドを出力する
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|