diff --git a/src/sing/phraseRendering.ts b/src/sing/phraseRendering.ts deleted file mode 100644 index baf3cc603a..0000000000 --- a/src/sing/phraseRendering.ts +++ /dev/null @@ -1,794 +0,0 @@ -/** - * フレーズごとに音声合成するフレーズレンダラーと、それに必要な処理。 - * レンダリングが必要かどうかの判定やキャッシュの作成も行う。 - */ - -import { - Note, - PhraseKey, - Singer, - SingingVoice, - SingingVoiceKey, - Tempo, - Track, - SingingVolumeKey, - SingingVolume, - EditorFrameAudioQueryKey, - EditorFrameAudioQuery, -} from "@/store/type"; -import { FramePhoneme, Note as NoteForRequestToEngine } from "@/openapi"; -import { applyPitchEdit, decibelToLinear, tickToSecond } from "@/sing/domain"; -import { calculateHash, linearInterpolation } from "@/sing/utility"; -import { EngineId, StyleId, TrackId } from "@/type/preload"; -import { createLogger } from "@/domain/frontend/log"; -import { cloneWithUnwrapProxy } from "@/helpers/cloneWithUnwrapProxy"; -import { getOrThrow } from "@/helpers/mapHelper"; - -const logger = createLogger("sing/phraseRendering"); - -/** - * リクエスト用のノーツ(と休符)を作成する。 - */ -const createNotesForRequestToEngine = ( - firstRestDuration: number, - lastRestDurationSeconds: number, - notes: Note[], - tempos: Tempo[], - tpqn: number, - frameRate: number, -) => { - const notesForRequestToEngine: NoteForRequestToEngine[] = []; - - // 先頭の休符を変換 - const firstRestStartSeconds = tickToSecond( - notes[0].position - firstRestDuration, - tempos, - tpqn, - ); - const firstRestStartFrame = Math.round(firstRestStartSeconds * frameRate); - const firstRestEndSeconds = tickToSecond(notes[0].position, tempos, tpqn); - const firstRestEndFrame = Math.round(firstRestEndSeconds * frameRate); - notesForRequestToEngine.push({ - key: undefined, - frameLength: firstRestEndFrame - firstRestStartFrame, - lyric: "", - }); - - // ノートを変換 - for (const note of notes) { - const noteOnSeconds = tickToSecond(note.position, tempos, tpqn); - const noteOnFrame = Math.round(noteOnSeconds * frameRate); - const noteOffSeconds = tickToSecond( - note.position + note.duration, - tempos, - tpqn, - ); - const noteOffFrame = Math.round(noteOffSeconds * frameRate); - notesForRequestToEngine.push({ - id: note.id, - key: note.noteNumber, - frameLength: noteOffFrame - noteOnFrame, - lyric: note.lyric, - }); - } - - // 末尾に休符を追加 - const lastRestFrameLength = Math.round(lastRestDurationSeconds * frameRate); - notesForRequestToEngine.push({ - key: undefined, - frameLength: lastRestFrameLength, - lyric: "", - }); - - // frameLengthが1以上になるようにする - for (let i = 0; i < notesForRequestToEngine.length; i++) { - const frameLength = notesForRequestToEngine[i].frameLength; - const frameToShift = Math.max(0, 1 - frameLength); - notesForRequestToEngine[i].frameLength += frameToShift; - if (i < notesForRequestToEngine.length - 1) { - notesForRequestToEngine[i + 1].frameLength -= frameToShift; - } - } - - return notesForRequestToEngine; -}; - -const shiftKeyOfNotes = (notes: NoteForRequestToEngine[], keyShift: number) => { - for (const note of notes) { - if (note.key != undefined) { - note.key += keyShift; - } - } -}; - -const getPhonemes = (query: EditorFrameAudioQuery) => { - return query.phonemes.map((value) => value.phoneme).join(" "); -}; - -const shiftPitch = (f0: number[], pitchShift: number) => { - for (let i = 0; i < f0.length; i++) { - f0[i] *= Math.pow(2, pitchShift / 12); - } -}; - -const shiftVolume = (volume: number[], volumeShift: number) => { - for (let i = 0; i < volume.length; i++) { - volume[i] *= decibelToLinear(volumeShift); - } -}; - -/** - * 末尾のpauの区間のvolumeを0にする。(歌とpauの呼吸音が重ならないようにする) - * fadeOutDurationSecondsが0の場合は即座にvolumeを0にする。 - */ -const muteLastPauSection = ( - volume: number[], - phonemes: FramePhoneme[], - frameRate: number, - fadeOutDurationSeconds: number, -) => { - const lastPhoneme = phonemes.at(-1); - if (lastPhoneme == undefined || lastPhoneme.phoneme !== "pau") { - throw new Error("No pau exists at the end."); - } - - let lastPauStartFrame = 0; - for (let i = 0; i < phonemes.length - 1; i++) { - lastPauStartFrame += phonemes[i].frameLength; - } - - const lastPauFrameLength = lastPhoneme.frameLength; - let fadeOutFrameLength = Math.round(fadeOutDurationSeconds * frameRate); - fadeOutFrameLength = Math.max(0, fadeOutFrameLength); - fadeOutFrameLength = Math.min(lastPauFrameLength, fadeOutFrameLength); - - // フェードアウト処理を行う - if (fadeOutFrameLength === 1) { - volume[lastPauStartFrame] *= 0.5; - } else { - for (let i = 0; i < fadeOutFrameLength; i++) { - volume[lastPauStartFrame + i] *= linearInterpolation( - 0, - 1, - fadeOutFrameLength - 1, - 0, - i, - ); - } - } - // 音量を0にする - for (let i = fadeOutFrameLength; i < lastPauFrameLength; i++) { - volume[lastPauStartFrame + i] = 0; - } -}; - -const singingTeacherStyleId = StyleId(6000); // TODO: 設定できるようにする -const lastRestDurationSeconds = 0.5; // TODO: 設定できるようにする -const fadeOutDurationSeconds = 0.15; // TODO: 設定できるようにする - -/** - * フレーズレンダリングに必要なデータのスナップショット - */ -type Snapshot = Readonly<{ - tpqn: number; - tempos: Tempo[]; - tracks: Map; - engineFrameRates: Map; - editorFrameRate: number; -}>; - -/** - * フレーズ - */ -type Phrase = Readonly<{ - firstRestDuration: number; - notes: Note[]; - startTime: number; - queryKey: { - get: () => EditorFrameAudioQueryKey | undefined; - set: (value: EditorFrameAudioQueryKey | undefined) => void; - }; - singingVolumeKey: { - get: () => SingingVolumeKey | undefined; - set: (value: SingingVolumeKey | undefined) => void; - }; - singingVoiceKey: { - get: () => SingingVoiceKey | undefined; - set: (value: SingingVoiceKey | undefined) => void; - }; -}>; - -/** - * フレーズレンダリングで必要となる外部のキャッシュや関数 - */ -type ExternalDependencies = Readonly<{ - queryCache: Map; - singingVolumeCache: Map; - singingVoiceCache: Map; - - phrases: { - get: (phraseKey: PhraseKey) => Phrase; - }; - phraseQueries: { - get: (queryKey: EditorFrameAudioQueryKey) => EditorFrameAudioQuery; - set: ( - queryKey: EditorFrameAudioQueryKey, - query: EditorFrameAudioQuery, - ) => void; - delete: (queryKey: EditorFrameAudioQueryKey) => void; - }; - phraseSingingVolumes: { - get: (singingVolumeKey: SingingVolumeKey) => SingingVolume; - set: ( - singingVolumeKey: SingingVolumeKey, - singingVolume: SingingVolume, - ) => void; - delete: (singingVolumeKey: SingingVolumeKey) => void; - }; - phraseSingingVoices: { - set: (singingVoiceKey: SingingVoiceKey, singingVoice: SingingVoice) => void; - delete: (singingVoiceKey: SingingVoiceKey) => void; - }; - - fetchQuery: ( - engineId: EngineId, - engineFrameRate: number, - notes: NoteForRequestToEngine[], - ) => Promise; - fetchSingFrameVolume: ( - notes: NoteForRequestToEngine[], - query: EditorFrameAudioQuery, - engineId: EngineId, - styleId: StyleId, - ) => Promise; - synthesizeSingingVoice: ( - singer: Singer, - query: EditorFrameAudioQuery, - ) => Promise; -}>; - -/** - * フレーズレンダリングのコンテキスト - */ -type Context = Readonly<{ - snapshot: Snapshot; - trackId: TrackId; - phraseKey: PhraseKey; - externalDependencies: ExternalDependencies; -}>; - -export type PhraseRenderStageId = - | "queryGeneration" - | "singingVolumeGeneration" - | "singingVoiceSynthesis"; - -/** - * フレーズレンダリングのステージのインターフェイス。 - * フレーズレンダラー内で順に実行される。 - */ -type BaseStage = Readonly<{ - id: PhraseRenderStageId; - - /** - * このステージが実行されるべきかを判定する。 - * @param context コンテキスト - * @returns 実行が必要かどうかのブール値 - */ - shouldBeExecuted: (context: Context) => Promise; - - /** - * 前回の処理結果を削除する。 - * @param context コンテキスト - */ - deleteExecutionResult: (context: Context) => void; - - /** - * ステージの処理を実行する。 - * @param context コンテキスト - */ - execute: (context: Context) => Promise; -}>; - -// クエリ生成ステージ - -/** - * クエリの生成に必要なデータ - */ -type QuerySource = Readonly<{ - engineId: EngineId; - engineFrameRate: number; - tpqn: number; - tempos: Tempo[]; - firstRestDuration: number; - notes: Note[]; - keyRangeAdjustment: number; -}>; - -const generateQuerySource = (context: Context): QuerySource => { - const { phrases } = context.externalDependencies; - - const track = getOrThrow(context.snapshot.tracks, context.trackId); - if (track.singer == undefined) { - throw new Error("track.singer is undefined."); - } - const engineFrameRate = getOrThrow( - context.snapshot.engineFrameRates, - track.singer.engineId, - ); - const phrase = phrases.get(context.phraseKey); - return { - engineId: track.singer.engineId, - engineFrameRate, - tpqn: context.snapshot.tpqn, - tempos: context.snapshot.tempos, - firstRestDuration: phrase.firstRestDuration, - notes: phrase.notes, - keyRangeAdjustment: track.keyRangeAdjustment, - }; -}; - -const calculateQueryKey = async (querySource: QuerySource) => { - const hash = await calculateHash(querySource); - return EditorFrameAudioQueryKey(hash); -}; - -const generateQuery = async ( - querySource: QuerySource, - externalDependencies: ExternalDependencies, -): Promise => { - const notesForRequestToEngine = createNotesForRequestToEngine( - querySource.firstRestDuration, - lastRestDurationSeconds, - querySource.notes, - querySource.tempos, - querySource.tpqn, - querySource.engineFrameRate, - ); - - shiftKeyOfNotes(notesForRequestToEngine, -querySource.keyRangeAdjustment); - - const query = await externalDependencies.fetchQuery( - querySource.engineId, - querySource.engineFrameRate, - notesForRequestToEngine, - ); - - shiftPitch(query.f0, querySource.keyRangeAdjustment); - return query; -}; - -const queryGenerationStage: BaseStage = { - id: "queryGeneration", - shouldBeExecuted: async (context: Context) => { - const { phrases } = context.externalDependencies; - - const track = getOrThrow(context.snapshot.tracks, context.trackId); - if (track.singer == undefined) { - return false; - } - const phrase = phrases.get(context.phraseKey); - const phraseQueryKey = phrase.queryKey.get(); - const querySource = generateQuerySource(context); - const queryKey = await calculateQueryKey(querySource); - return phraseQueryKey == undefined || phraseQueryKey !== queryKey; - }, - deleteExecutionResult: (context: Context) => { - const { phrases, phraseQueries } = context.externalDependencies; - - const phrase = phrases.get(context.phraseKey); - const phraseQueryKey = phrase.queryKey.get(); - if (phraseQueryKey != undefined) { - phraseQueries.delete(phraseQueryKey); - phrase.queryKey.set(undefined); - } - }, - execute: async (context: Context) => { - const { phrases, phraseQueries, queryCache } = context.externalDependencies; - - const querySource = generateQuerySource(context); - const queryKey = await calculateQueryKey(querySource); - - let query = queryCache.get(queryKey); - if (query != undefined) { - logger.info(`Loaded query from cache.`); - } else { - query = await generateQuery(querySource, context.externalDependencies); - const phonemes = getPhonemes(query); - logger.info(`Generated query. phonemes: ${phonemes}`); - queryCache.set(queryKey, query); - } - - const phrase = phrases.get(context.phraseKey); - const phraseQueryKey = phrase.queryKey.get(); - if (phraseQueryKey != undefined) { - phraseQueries.delete(phraseQueryKey); - } - phraseQueries.set(queryKey, query); - phrase.queryKey.set(queryKey); - }, -}; - -// 歌唱ボリューム生成ステージ - -/** - * 歌唱ボリュームの生成に必要なデータ - */ -type SingingVolumeSource = Readonly<{ - engineId: EngineId; - engineFrameRate: number; - tpqn: number; - tempos: Tempo[]; - firstRestDuration: number; - notes: Note[]; - keyRangeAdjustment: number; - volumeRangeAdjustment: number; - queryForVolumeGeneration: EditorFrameAudioQuery; -}>; - -const generateSingingVolumeSource = (context: Context): SingingVolumeSource => { - const { phrases, phraseQueries } = context.externalDependencies; - - const track = getOrThrow(context.snapshot.tracks, context.trackId); - if (track.singer == undefined) { - throw new Error("track.singer is undefined."); - } - const phrase = phrases.get(context.phraseKey); - const phraseQueryKey = phrase.queryKey.get(); - if (phraseQueryKey == undefined) { - throw new Error("phraseQueryKey is undefined."); - } - const query = phraseQueries.get(phraseQueryKey); - const clonedQuery = cloneWithUnwrapProxy(query); - applyPitchEdit( - clonedQuery, - phrase.startTime, - track.pitchEditData, - context.snapshot.editorFrameRate, - ); - return { - engineId: track.singer.engineId, - engineFrameRate: query.frameRate, - tpqn: context.snapshot.tpqn, - tempos: context.snapshot.tempos, - firstRestDuration: phrase.firstRestDuration, - notes: phrase.notes, - keyRangeAdjustment: track.keyRangeAdjustment, - volumeRangeAdjustment: track.volumeRangeAdjustment, - queryForVolumeGeneration: clonedQuery, - }; -}; - -const calculateSingingVolumeKey = async ( - singingVolumeSource: SingingVolumeSource, -) => { - const hash = await calculateHash(singingVolumeSource); - return SingingVolumeKey(hash); -}; - -const generateSingingVolume = async ( - singingVolumeSource: SingingVolumeSource, - externalDependencies: ExternalDependencies, -) => { - const notesForRequestToEngine = createNotesForRequestToEngine( - singingVolumeSource.firstRestDuration, - lastRestDurationSeconds, - singingVolumeSource.notes, - singingVolumeSource.tempos, - singingVolumeSource.tpqn, - singingVolumeSource.engineFrameRate, - ); - const queryForVolumeGeneration = singingVolumeSource.queryForVolumeGeneration; - - shiftKeyOfNotes( - notesForRequestToEngine, - -singingVolumeSource.keyRangeAdjustment, - ); - shiftPitch( - queryForVolumeGeneration.f0, - -singingVolumeSource.keyRangeAdjustment, - ); - - const singingVolume = await externalDependencies.fetchSingFrameVolume( - notesForRequestToEngine, - queryForVolumeGeneration, - singingVolumeSource.engineId, - singingTeacherStyleId, - ); - - shiftVolume(singingVolume, singingVolumeSource.volumeRangeAdjustment); - muteLastPauSection( - singingVolume, - queryForVolumeGeneration.phonemes, - singingVolumeSource.engineFrameRate, - fadeOutDurationSeconds, - ); - return singingVolume; -}; - -const singingVolumeGenerationStage: BaseStage = { - id: "singingVolumeGeneration", - shouldBeExecuted: async (context: Context) => { - const { phrases } = context.externalDependencies; - - const track = getOrThrow(context.snapshot.tracks, context.trackId); - if (track.singer == undefined) { - return false; - } - const singingVolumeSource = generateSingingVolumeSource(context); - const singingVolumeKey = - await calculateSingingVolumeKey(singingVolumeSource); - const phrase = phrases.get(context.phraseKey); - const phraseSingingVolumeKey = phrase.singingVolumeKey.get(); - return ( - phraseSingingVolumeKey == undefined || - phraseSingingVolumeKey !== singingVolumeKey - ); - }, - deleteExecutionResult: (context: Context) => { - const { phrases, phraseSingingVolumes } = context.externalDependencies; - - const phrase = phrases.get(context.phraseKey); - const phraseSingingVolumeKey = phrase.singingVolumeKey.get(); - if (phraseSingingVolumeKey != undefined) { - phraseSingingVolumes.delete(phraseSingingVolumeKey); - phrase.singingVolumeKey.set(undefined); - } - }, - execute: async (context: Context) => { - const { phrases, phraseSingingVolumes, singingVolumeCache } = - context.externalDependencies; - - const singingVolumeSource = generateSingingVolumeSource(context); - const singingVolumeKey = - await calculateSingingVolumeKey(singingVolumeSource); - - let singingVolume = singingVolumeCache.get(singingVolumeKey); - if (singingVolume != undefined) { - logger.info(`Loaded singing volume from cache.`); - } else { - singingVolume = await generateSingingVolume( - singingVolumeSource, - context.externalDependencies, - ); - logger.info(`Generated singing volume.`); - singingVolumeCache.set(singingVolumeKey, singingVolume); - } - - const phrase = phrases.get(context.phraseKey); - const phraseSingingVolumeKey = phrase.singingVolumeKey.get(); - if (phraseSingingVolumeKey != undefined) { - phraseSingingVolumes.delete(phraseSingingVolumeKey); - } - phraseSingingVolumes.set(singingVolumeKey, singingVolume); - phrase.singingVolumeKey.set(singingVolumeKey); - }, -}; - -// 歌唱音声合成ステージ - -/** - * 歌唱音声の合成に必要なデータ - */ -type SingingVoiceSource = Readonly<{ - singer: Singer; - queryForSingingVoiceSynthesis: EditorFrameAudioQuery; -}>; - -const generateSingingVoiceSource = (context: Context): SingingVoiceSource => { - const { phrases, phraseQueries, phraseSingingVolumes } = - context.externalDependencies; - - const track = getOrThrow(context.snapshot.tracks, context.trackId); - if (track.singer == undefined) { - throw new Error("track.singer is undefined."); - } - const phrase = phrases.get(context.phraseKey); - const phraseQueryKey = phrase.queryKey.get(); - const phraseSingingVolumeKey = phrase.singingVolumeKey.get(); - if (phraseQueryKey == undefined) { - throw new Error("phraseQueryKey is undefined."); - } - if (phraseSingingVolumeKey == undefined) { - throw new Error("phraseSingingVolumeKey is undefined."); - } - const query = phraseQueries.get(phraseQueryKey); - const singingVolume = phraseSingingVolumes.get(phraseSingingVolumeKey); - const clonedQuery = cloneWithUnwrapProxy(query); - const clonedSingingVolume = cloneWithUnwrapProxy(singingVolume); - applyPitchEdit( - clonedQuery, - phrase.startTime, - track.pitchEditData, - context.snapshot.editorFrameRate, - ); - clonedQuery.volume = clonedSingingVolume; - return { - singer: track.singer, - queryForSingingVoiceSynthesis: clonedQuery, - }; -}; - -const calculateSingingVoiceKey = async ( - singingVoiceSource: SingingVoiceSource, -) => { - const hash = await calculateHash(singingVoiceSource); - return SingingVoiceKey(hash); -}; - -const synthesizeSingingVoice = async ( - singingVoiceSource: SingingVoiceSource, - externalDependencies: ExternalDependencies, -) => { - const singingVoice = await externalDependencies.synthesizeSingingVoice( - singingVoiceSource.singer, - singingVoiceSource.queryForSingingVoiceSynthesis, - ); - return singingVoice; -}; - -const singingVoiceSynthesisStage: BaseStage = { - id: "singingVoiceSynthesis", - shouldBeExecuted: async (context: Context) => { - const { phrases } = context.externalDependencies; - - const track = getOrThrow(context.snapshot.tracks, context.trackId); - if (track.singer == undefined) { - return false; - } - const singingVoiceSource = generateSingingVoiceSource(context); - const singingVoiceKey = await calculateSingingVoiceKey(singingVoiceSource); - const phrase = phrases.get(context.phraseKey); - const phraseSingingVoiceKey = phrase.singingVoiceKey.get(); - return ( - phraseSingingVoiceKey == undefined || - phraseSingingVoiceKey !== singingVoiceKey - ); - }, - deleteExecutionResult: (context: Context) => { - const { phrases, phraseSingingVoices } = context.externalDependencies; - - const phrase = phrases.get(context.phraseKey); - const phraseSingingVoiceKey = phrase.singingVoiceKey.get(); - if (phraseSingingVoiceKey != undefined) { - phraseSingingVoices.delete(phraseSingingVoiceKey); - phrase.singingVoiceKey.set(undefined); - } - }, - execute: async (context: Context) => { - const { phrases, phraseSingingVoices, singingVoiceCache } = - context.externalDependencies; - - const singingVoiceSource = generateSingingVoiceSource(context); - const singingVoiceKey = await calculateSingingVoiceKey(singingVoiceSource); - - let singingVoice = singingVoiceCache.get(singingVoiceKey); - if (singingVoice != undefined) { - logger.info(`Loaded singing voice from cache.`); - } else { - singingVoice = await synthesizeSingingVoice( - singingVoiceSource, - context.externalDependencies, - ); - logger.info(`Generated singing voice.`); - singingVoiceCache.set(singingVoiceKey, singingVoice); - } - - const phrase = phrases.get(context.phraseKey); - const phraseSingingVoiceKey = phrase.singingVoiceKey.get(); - if (phraseSingingVoiceKey != undefined) { - phraseSingingVoices.delete(phraseSingingVoiceKey); - } - phraseSingingVoices.set(singingVoiceKey, singingVoice); - phrase.singingVoiceKey.set(singingVoiceKey); - }, -}; - -// フレーズレンダラー - -/** - * フレーズレンダラー。 - * 各フレーズごとに、ステージを進めながらレンダリング処理を行う。 - */ -export type PhraseRenderer = Readonly<{ - /** - * 一番最初のステージのIDを返す。 - * 一度もレンダリングを行っていないフレーズは、 - * この(一番最初の)ステージからレンダリング処理を開始する必要がある。 - * @returns ステージID - */ - getFirstRenderStageId: () => PhraseRenderStageId; - - /** - * レンダリングが必要なフレーズかどうかを判断し、 - * レンダリングが必要であればどのステージから開始されるべきかを判断して、そのステージのIDを返す。 - * レンダリングが必要ない場合、undefinedが返される。 - * @param snapshot スナップショット - * @param trackId トラックID - * @param phraseKey フレーズキー - * @returns ステージID または undefined - */ - determineStartStage: ( - snapshot: Snapshot, - trackId: TrackId, - phraseKey: PhraseKey, - ) => Promise; - - /** - * 指定されたフレーズのレンダリング処理を、指定されたステージから開始する。 - * レンダリング処理を開始する前に、前回のレンダリング処理結果の削除が行われる。 - * @param snapshot スナップショット - * @param trackId トラックID - * @param phraseKey フレーズキー - * @param startStageId 開始ステージID - */ - render: ( - snapshot: Snapshot, - trackId: TrackId, - phraseKey: PhraseKey, - startStageId: PhraseRenderStageId, - ) => Promise; -}>; - -const stages: readonly BaseStage[] = [ - queryGenerationStage, - singingVolumeGenerationStage, - singingVoiceSynthesisStage, -]; - -/** - * フレーズレンダラーを作成する。 - * @param externalDependencies レンダリング処理で必要となる外部のキャッシュや関数 - * @returns フレーズレンダラー - */ -export const createPhraseRenderer = ( - externalDependencies: ExternalDependencies, -): PhraseRenderer => { - return { - getFirstRenderStageId: () => { - return stages[0].id; - }, - determineStartStage: async ( - snapshot: Snapshot, - trackId: TrackId, - phraseKey: PhraseKey, - ) => { - const context: Context = { - snapshot, - trackId, - phraseKey, - externalDependencies, - }; - for (const stage of stages) { - if (await stage.shouldBeExecuted(context)) { - return stage.id; - } - } - return undefined; - }, - render: async ( - snapshot: Snapshot, - trackId: TrackId, - phraseKey: PhraseKey, - startStageId: PhraseRenderStageId, - ) => { - const context: Context = { - snapshot, - trackId, - phraseKey, - externalDependencies, - }; - const startStageIndex = stages.findIndex((value) => { - return value.id === startStageId; - }); - if (startStageIndex === -1) { - throw new Error("Stage not found."); - } - for (let i = stages.length - 1; i >= startStageIndex; i--) { - stages[i].deleteExecutionResult(context); - } - for (let i = startStageIndex; i < stages.length; i++) { - await stages[i].execute(context); - } - }, - }; -}; diff --git a/src/store/singing.ts b/src/store/singing.ts index 6512aa0349..d787822713 100644 --- a/src/store/singing.ts +++ b/src/store/singing.ts @@ -40,7 +40,7 @@ import { StyleId, TrackId, } from "@/type/preload"; -import { Note as NoteForRequestToEngine } from "@/openapi"; +import { FramePhoneme, Note as NoteForRequestToEngine } from "@/openapi"; import { ResultError, getValueOrThrow } from "@/type/result"; import { AudioEvent, @@ -84,11 +84,15 @@ import { getNumMeasures, isTracksEmpty, shouldPlayTracks, + decibelToLinear, + applyPitchEdit, } from "@/sing/domain"; import { getOverlappingNoteIds } from "@/sing/storeHelper"; import { AnimationTimer, + calculateHash, createPromiseThatResolvesWhen, + linearInterpolation, round, } from "@/sing/utility"; import { getWorkaroundKeyRangeAdjustment } from "@/sing/workaroundKeyRangeAdjustment"; @@ -100,13 +104,282 @@ import { ufProjectToVoicevox } from "@/sing/utaformatixProject/toVoicevox"; import { uuid4 } from "@/helpers/random"; import { convertToWavFileData } from "@/sing/convertToWavFileData"; import { generateWriteErrorMessage } from "@/helpers/fileHelper"; -import { - PhraseRenderStageId, - createPhraseRenderer, -} from "@/sing/phraseRendering"; const logger = createLogger("store/singing"); +/** + * フレーズレンダリングに必要なデータのスナップショット + */ +type SnapshotForPhraseRender = Readonly<{ + tpqn: number; + tempos: Tempo[]; + tracks: Map; + engineFrameRates: Map; + editorFrameRate: number; +}>; + +/** + * フレーズレンダリングのコンテキスト + */ +type PhraseRenderContext = Readonly<{ + snapshot: SnapshotForPhraseRender; + trackId: TrackId; + phraseKey: PhraseKey; +}>; + +type PhraseRenderStageId = + | "queryGeneration" + | "singingVolumeGeneration" + | "singingVoiceSynthesis"; + +/** + * フレーズレンダリングのステージのインターフェイス。 + * フレーズレンダラー内で順に実行される。 + */ +type PhraseRenderBaseStage = Readonly<{ + id: PhraseRenderStageId; + + /** + * このステージが実行されるべきかを判定する。 + * @param context コンテキスト + * @returns 実行が必要かどうかのブール値 + */ + shouldBeExecuted: (context: PhraseRenderContext) => Promise; + + /** + * 前回の処理結果を削除する。 + * @param context コンテキスト + */ + deleteExecutionResult: (context: PhraseRenderContext) => void; + + /** + * ステージの処理を実行する。 + * @param context コンテキスト + */ + execute: (context: PhraseRenderContext) => Promise; +}>; + +/** + * クエリの生成に必要なデータ + */ +type QuerySource = Readonly<{ + engineId: EngineId; + engineFrameRate: number; + tpqn: number; + tempos: Tempo[]; + firstRestDuration: number; + notes: Note[]; + keyRangeAdjustment: number; +}>; + +/** + * 歌唱ボリュームの生成に必要なデータ + */ +type SingingVolumeSource = Readonly<{ + engineId: EngineId; + engineFrameRate: number; + tpqn: number; + tempos: Tempo[]; + firstRestDuration: number; + notes: Note[]; + keyRangeAdjustment: number; + volumeRangeAdjustment: number; + queryForVolumeGeneration: EditorFrameAudioQuery; +}>; + +/** + * 歌唱音声の合成に必要なデータ + */ +type SingingVoiceSource = Readonly<{ + singer: Singer; + queryForSingingVoiceSynthesis: EditorFrameAudioQuery; +}>; + +/** + * フレーズレンダラー。 + * 各フレーズごとに、ステージを進めながらレンダリング処理を行う。 + */ +type PhraseRenderer = Readonly<{ + /** + * 一番最初のステージのIDを返す。 + * 一度もレンダリングを行っていないフレーズは、 + * この(一番最初の)ステージからレンダリング処理を開始する必要がある。 + * @returns ステージID + */ + getFirstRenderStageId: () => PhraseRenderStageId; + + /** + * レンダリングが必要なフレーズかどうかを判断し、 + * レンダリングが必要であればどのステージから開始されるべきかを判断して、そのステージのIDを返す。 + * レンダリングが必要ない場合、undefinedが返される。 + * @param snapshot スナップショット + * @param trackId トラックID + * @param phraseKey フレーズキー + * @returns ステージID または undefined + */ + determineStartStage: ( + snapshot: SnapshotForPhraseRender, + trackId: TrackId, + phraseKey: PhraseKey, + ) => Promise; + + /** + * 指定されたフレーズのレンダリング処理を、指定されたステージから開始する。 + * レンダリング処理を開始する前に、前回のレンダリング処理結果の削除が行われる。 + * @param snapshot スナップショット + * @param trackId トラックID + * @param phraseKey フレーズキー + * @param startStageId 開始ステージID + */ + render: ( + snapshot: SnapshotForPhraseRender, + trackId: TrackId, + phraseKey: PhraseKey, + startStageId: PhraseRenderStageId, + ) => Promise; +}>; + +/** + * リクエスト用のノーツ(と休符)を作成する。 + */ +const createNotesForRequestToEngine = ( + firstRestDuration: number, + lastRestDurationSeconds: number, + notes: Note[], + tempos: Tempo[], + tpqn: number, + frameRate: number, +) => { + const notesForRequestToEngine: NoteForRequestToEngine[] = []; + + // 先頭の休符を変換 + const firstRestStartSeconds = tickToSecond( + notes[0].position - firstRestDuration, + tempos, + tpqn, + ); + const firstRestStartFrame = Math.round(firstRestStartSeconds * frameRate); + const firstRestEndSeconds = tickToSecond(notes[0].position, tempos, tpqn); + const firstRestEndFrame = Math.round(firstRestEndSeconds * frameRate); + notesForRequestToEngine.push({ + key: undefined, + frameLength: firstRestEndFrame - firstRestStartFrame, + lyric: "", + }); + + // ノートを変換 + for (const note of notes) { + const noteOnSeconds = tickToSecond(note.position, tempos, tpqn); + const noteOnFrame = Math.round(noteOnSeconds * frameRate); + const noteOffSeconds = tickToSecond( + note.position + note.duration, + tempos, + tpqn, + ); + const noteOffFrame = Math.round(noteOffSeconds * frameRate); + notesForRequestToEngine.push({ + id: note.id, + key: note.noteNumber, + frameLength: noteOffFrame - noteOnFrame, + lyric: note.lyric, + }); + } + + // 末尾に休符を追加 + const lastRestFrameLength = Math.round(lastRestDurationSeconds * frameRate); + notesForRequestToEngine.push({ + key: undefined, + frameLength: lastRestFrameLength, + lyric: "", + }); + + // frameLengthが1以上になるようにする + for (let i = 0; i < notesForRequestToEngine.length; i++) { + const frameLength = notesForRequestToEngine[i].frameLength; + const frameToShift = Math.max(0, 1 - frameLength); + notesForRequestToEngine[i].frameLength += frameToShift; + if (i < notesForRequestToEngine.length - 1) { + notesForRequestToEngine[i + 1].frameLength -= frameToShift; + } + } + + return notesForRequestToEngine; +}; + +const shiftKeyOfNotes = (notes: NoteForRequestToEngine[], keyShift: number) => { + for (const note of notes) { + if (note.key != undefined) { + note.key += keyShift; + } + } +}; + +const getPhonemes = (query: EditorFrameAudioQuery) => { + return query.phonemes.map((value) => value.phoneme).join(" "); +}; + +const shiftPitch = (f0: number[], pitchShift: number) => { + for (let i = 0; i < f0.length; i++) { + f0[i] *= Math.pow(2, pitchShift / 12); + } +}; + +const shiftVolume = (volume: number[], volumeShift: number) => { + for (let i = 0; i < volume.length; i++) { + volume[i] *= decibelToLinear(volumeShift); + } +}; + +const calculateQueryKey = async (querySource: QuerySource) => { + const hash = await calculateHash(querySource); + return EditorFrameAudioQueryKey(hash); +}; + +/** + * 末尾のpauの区間のvolumeを0にする。(歌とpauの呼吸音が重ならないようにする) + * fadeOutDurationSecondsが0の場合は即座にvolumeを0にする。 + */ +const muteLastPauSection = ( + volume: number[], + phonemes: FramePhoneme[], + frameRate: number, + fadeOutDurationSeconds: number, +) => { + const lastPhoneme = phonemes.at(-1); + if (lastPhoneme == undefined || lastPhoneme.phoneme !== "pau") { + throw new Error("No pau exists at the end."); + } + + let lastPauStartFrame = 0; + for (let i = 0; i < phonemes.length - 1; i++) { + lastPauStartFrame += phonemes[i].frameLength; + } + + const lastPauFrameLength = lastPhoneme.frameLength; + let fadeOutFrameLength = Math.round(fadeOutDurationSeconds * frameRate); + fadeOutFrameLength = Math.max(0, fadeOutFrameLength); + fadeOutFrameLength = Math.min(lastPauFrameLength, fadeOutFrameLength); + + // フェードアウト処理を行う + if (fadeOutFrameLength === 1) { + volume[lastPauStartFrame] *= 0.5; + } else { + for (let i = 0; i < fadeOutFrameLength; i++) { + volume[lastPauStartFrame + i] *= linearInterpolation( + 0, + 1, + fadeOutFrameLength - 1, + 0, + i, + ); + } + } + // 音量を0にする + for (let i = fadeOutFrameLength; i < lastPauFrameLength; i++) { + volume[lastPauStartFrame + i] = 0; + } +}; + const generateAudioEvents = async ( audioContext: BaseAudioContext, time: number, @@ -1338,6 +1611,10 @@ export const singingStore = createPartialStore({ */ RENDER: { async action({ state, getters, mutations, actions }) { + const singingTeacherStyleId = StyleId(6000); // TODO: 設定できるようにする + const lastRestDurationSeconds = 0.5; // TODO: 設定できるようにする + const fadeOutDurationSeconds = 0.15; // TODO: 設定できるようにする + const calcPhraseFirstRestDuration = ( prevPhraseLastNote: Note | undefined, phraseFirstNote: Note, @@ -1459,7 +1736,28 @@ export const singingStore = createPartialStore({ return foundPhrases; }; - const singingTeacherStyleId = StyleId(6000); // TODO: 設定できるようにする + const generateQuerySource = ( + context: PhraseRenderContext, + ): QuerySource => { + const track = getOrThrow(context.snapshot.tracks, context.trackId); + if (track.singer == undefined) { + throw new Error("track.singer is undefined."); + } + const engineFrameRate = getOrThrow( + context.snapshot.engineFrameRates, + track.singer.engineId, + ); + const phrase = getOrThrow(state.phrases, context.phraseKey); + return { + engineId: track.singer.engineId, + engineFrameRate, + tpqn: context.snapshot.tpqn, + tempos: context.snapshot.tempos, + firstRestDuration: phrase.firstRestDuration, + notes: phrase.notes, + keyRangeAdjustment: track.keyRangeAdjustment, + }; + }; const fetchQuery = async ( engineId: EngineId, @@ -1496,14 +1794,278 @@ export const singingStore = createPartialStore({ } }; + const generateQuery = async ( + querySource: QuerySource, + ): Promise => { + const notesForRequestToEngine = createNotesForRequestToEngine( + querySource.firstRestDuration, + lastRestDurationSeconds, + querySource.notes, + querySource.tempos, + querySource.tpqn, + querySource.engineFrameRate, + ); + + shiftKeyOfNotes( + notesForRequestToEngine, + -querySource.keyRangeAdjustment, + ); + + const query = await fetchQuery( + querySource.engineId, + querySource.engineFrameRate, + notesForRequestToEngine, + ); + + shiftPitch(query.f0, querySource.keyRangeAdjustment); + return query; + }; + + const queryGenerationStage: PhraseRenderBaseStage = { + id: "queryGeneration", + shouldBeExecuted: async (context: PhraseRenderContext) => { + const track = getOrThrow(context.snapshot.tracks, context.trackId); + if (track.singer == undefined) { + return false; + } + const phrase = getOrThrow(state.phrases, context.phraseKey); + const phraseQueryKey = phrase.queryKey; + const querySource = generateQuerySource(context); + const queryKey = await calculateQueryKey(querySource); + return phraseQueryKey == undefined || phraseQueryKey !== queryKey; + }, + deleteExecutionResult: (context: PhraseRenderContext) => { + const phrase = getOrThrow(state.phrases, context.phraseKey); + const phraseQueryKey = phrase.queryKey; + if (phraseQueryKey != undefined) { + mutations.DELETE_PHRASE_QUERY({ queryKey: phraseQueryKey }); + mutations.SET_QUERY_KEY_TO_PHRASE({ + phraseKey: context.phraseKey, + queryKey: undefined, + }); + } + }, + execute: async (context: PhraseRenderContext) => { + const querySource = generateQuerySource(context); + const queryKey = await calculateQueryKey(querySource); + + let query = queryCache.get(queryKey); + if (query != undefined) { + logger.info(`Loaded query from cache.`); + } else { + query = await generateQuery(querySource); + const phonemes = getPhonemes(query); + logger.info(`Generated query. phonemes: ${phonemes}`); + queryCache.set(queryKey, query); + } + + const phrase = getOrThrow(state.phrases, context.phraseKey); + const phraseQueryKey = phrase.queryKey; + if (phraseQueryKey != undefined) { + mutations.DELETE_PHRASE_QUERY({ queryKey: phraseQueryKey }); + } + mutations.SET_PHRASE_QUERY({ queryKey, query }); + mutations.SET_QUERY_KEY_TO_PHRASE({ + phraseKey: context.phraseKey, + queryKey, + }); + }, + }; + + const generateSingingVolumeSource = ( + context: PhraseRenderContext, + ): SingingVolumeSource => { + const track = getOrThrow(context.snapshot.tracks, context.trackId); + if (track.singer == undefined) { + throw new Error("track.singer is undefined."); + } + const phrase = getOrThrow(state.phrases, context.phraseKey); + const phraseQueryKey = phrase.queryKey; + if (phraseQueryKey == undefined) { + throw new Error("phraseQueryKey is undefined."); + } + const query = getOrThrow(state.phraseQueries, phraseQueryKey); + const clonedQuery = cloneWithUnwrapProxy(query); + applyPitchEdit( + clonedQuery, + phrase.startTime, + track.pitchEditData, + context.snapshot.editorFrameRate, + ); + return { + engineId: track.singer.engineId, + engineFrameRate: query.frameRate, + tpqn: context.snapshot.tpqn, + tempos: context.snapshot.tempos, + firstRestDuration: phrase.firstRestDuration, + notes: phrase.notes, + keyRangeAdjustment: track.keyRangeAdjustment, + volumeRangeAdjustment: track.volumeRangeAdjustment, + queryForVolumeGeneration: clonedQuery, + }; + }; + + const calculateSingingVolumeKey = async ( + singingVolumeSource: SingingVolumeSource, + ) => { + const hash = await calculateHash(singingVolumeSource); + return SingingVolumeKey(hash); + }; + + const generateSingingVolume = async ( + singingVolumeSource: SingingVolumeSource, + ) => { + const notesForRequestToEngine = createNotesForRequestToEngine( + singingVolumeSource.firstRestDuration, + lastRestDurationSeconds, + singingVolumeSource.notes, + singingVolumeSource.tempos, + singingVolumeSource.tpqn, + singingVolumeSource.engineFrameRate, + ); + const queryForVolumeGeneration = + singingVolumeSource.queryForVolumeGeneration; + + shiftKeyOfNotes( + notesForRequestToEngine, + -singingVolumeSource.keyRangeAdjustment, + ); + shiftPitch( + queryForVolumeGeneration.f0, + -singingVolumeSource.keyRangeAdjustment, + ); + + const singingVolume = await actions.FETCH_SING_FRAME_VOLUME({ + notes: notesForRequestToEngine, + query: queryForVolumeGeneration, + engineId: singingVolumeSource.engineId, + styleId: singingTeacherStyleId, + }); + + shiftVolume(singingVolume, singingVolumeSource.volumeRangeAdjustment); + muteLastPauSection( + singingVolume, + queryForVolumeGeneration.phonemes, + singingVolumeSource.engineFrameRate, + fadeOutDurationSeconds, + ); + return singingVolume; + }; + + const singingVolumeGenerationStage: PhraseRenderBaseStage = { + id: "singingVolumeGeneration", + shouldBeExecuted: async (context: PhraseRenderContext) => { + const track = getOrThrow(context.snapshot.tracks, context.trackId); + if (track.singer == undefined) { + return false; + } + const singingVolumeSource = generateSingingVolumeSource(context); + const singingVolumeKey = + await calculateSingingVolumeKey(singingVolumeSource); + const phrase = getOrThrow(state.phrases, context.phraseKey); + const phraseSingingVolumeKey = phrase.singingVolumeKey; + return ( + phraseSingingVolumeKey == undefined || + phraseSingingVolumeKey !== singingVolumeKey + ); + }, + deleteExecutionResult: (context: PhraseRenderContext) => { + const phrase = getOrThrow(state.phrases, context.phraseKey); + const phraseSingingVolumeKey = phrase.singingVolumeKey; + if (phraseSingingVolumeKey != undefined) { + mutations.DELETE_PHRASE_SINGING_VOLUME({ + singingVolumeKey: phraseSingingVolumeKey, + }); + mutations.SET_SINGING_VOLUME_KEY_TO_PHRASE({ + phraseKey: context.phraseKey, + singingVolumeKey: undefined, + }); + } + }, + execute: async (context: PhraseRenderContext) => { + const singingVolumeSource = generateSingingVolumeSource(context); + const singingVolumeKey = + await calculateSingingVolumeKey(singingVolumeSource); + + let singingVolume = singingVolumeCache.get(singingVolumeKey); + if (singingVolume != undefined) { + logger.info(`Loaded singing volume from cache.`); + } else { + singingVolume = await generateSingingVolume(singingVolumeSource); + logger.info(`Generated singing volume.`); + singingVolumeCache.set(singingVolumeKey, singingVolume); + } + + const phrase = getOrThrow(state.phrases, context.phraseKey); + const phraseSingingVolumeKey = phrase.singingVolumeKey; + if (phraseSingingVolumeKey != undefined) { + mutations.DELETE_PHRASE_SINGING_VOLUME({ + singingVolumeKey: phraseSingingVolumeKey, + }); + } + mutations.SET_PHRASE_SINGING_VOLUME({ + singingVolumeKey, + singingVolume, + }); + mutations.SET_SINGING_VOLUME_KEY_TO_PHRASE({ + phraseKey: context.phraseKey, + singingVolumeKey, + }); + }, + }; + + const generateSingingVoiceSource = ( + context: PhraseRenderContext, + ): SingingVoiceSource => { + const track = getOrThrow(context.snapshot.tracks, context.trackId); + if (track.singer == undefined) { + throw new Error("track.singer is undefined."); + } + const phrase = getOrThrow(state.phrases, context.phraseKey); + const phraseQueryKey = phrase.queryKey; + const phraseSingingVolumeKey = phrase.singingVolumeKey; + if (phraseQueryKey == undefined) { + throw new Error("phraseQueryKey is undefined."); + } + if (phraseSingingVolumeKey == undefined) { + throw new Error("phraseSingingVolumeKey is undefined."); + } + const query = getOrThrow(state.phraseQueries, phraseQueryKey); + const singingVolume = getOrThrow( + state.phraseSingingVolumes, + phraseSingingVolumeKey, + ); + const clonedQuery = cloneWithUnwrapProxy(query); + const clonedSingingVolume = cloneWithUnwrapProxy(singingVolume); + applyPitchEdit( + clonedQuery, + phrase.startTime, + track.pitchEditData, + context.snapshot.editorFrameRate, + ); + clonedQuery.volume = clonedSingingVolume; + return { + singer: track.singer, + queryForSingingVoiceSynthesis: clonedQuery, + }; + }; + + const calculateSingingVoiceKey = async ( + singingVoiceSource: SingingVoiceSource, + ) => { + const hash = await calculateHash(singingVoiceSource); + return SingingVoiceKey(hash); + }; + const synthesizeSingingVoice = async ( - singer: Singer, - query: EditorFrameAudioQuery, + singingVoiceSource: SingingVoiceSource, ) => { + const singer = singingVoiceSource.singer; + const query = singingVoiceSource.queryForSingingVoiceSynthesis; + if (!getters.IS_ENGINE_READY(singer.engineId)) { throw new Error("Engine not ready."); } - try { const instance = await actions.INSTANTIATE_ENGINE_CONNECTOR({ engineId: singer.engineId, @@ -1524,6 +2086,114 @@ export const singingStore = createPartialStore({ } }; + const singingVoiceSynthesisStage: PhraseRenderBaseStage = { + id: "singingVoiceSynthesis", + shouldBeExecuted: async (context: PhraseRenderContext) => { + const track = getOrThrow(context.snapshot.tracks, context.trackId); + if (track.singer == undefined) { + return false; + } + const singingVoiceSource = generateSingingVoiceSource(context); + const singingVoiceKey = + await calculateSingingVoiceKey(singingVoiceSource); + const phrase = getOrThrow(state.phrases, context.phraseKey); + const phraseSingingVoiceKey = phrase.singingVoiceKey; + return ( + phraseSingingVoiceKey == undefined || + phraseSingingVoiceKey !== singingVoiceKey + ); + }, + deleteExecutionResult: (context: PhraseRenderContext) => { + const phrase = getOrThrow(state.phrases, context.phraseKey); + const phraseSingingVoiceKey = phrase.singingVoiceKey; + if (phraseSingingVoiceKey != undefined) { + phraseSingingVoices.delete(phraseSingingVoiceKey); + mutations.SET_SINGING_VOICE_KEY_TO_PHRASE({ + phraseKey: context.phraseKey, + singingVoiceKey: undefined, + }); + } + }, + execute: async (context: PhraseRenderContext) => { + const singingVoiceSource = generateSingingVoiceSource(context); + const singingVoiceKey = + await calculateSingingVoiceKey(singingVoiceSource); + + let singingVoice = singingVoiceCache.get(singingVoiceKey); + if (singingVoice != undefined) { + logger.info(`Loaded singing voice from cache.`); + } else { + singingVoice = await synthesizeSingingVoice(singingVoiceSource); + logger.info(`Generated singing voice.`); + singingVoiceCache.set(singingVoiceKey, singingVoice); + } + + const phrase = getOrThrow(state.phrases, context.phraseKey); + const phraseSingingVoiceKey = phrase.singingVoiceKey; + if (phraseSingingVoiceKey != undefined) { + phraseSingingVoices.delete(phraseSingingVoiceKey); + } + phraseSingingVoices.set(singingVoiceKey, singingVoice); + mutations.SET_SINGING_VOICE_KEY_TO_PHRASE({ + phraseKey: context.phraseKey, + singingVoiceKey, + }); + }, + }; + + const stages: readonly PhraseRenderBaseStage[] = [ + queryGenerationStage, + singingVolumeGenerationStage, + singingVoiceSynthesisStage, + ]; + + const phraseRenderer: PhraseRenderer = { + getFirstRenderStageId: () => { + return stages[0].id; + }, + determineStartStage: async ( + snapshot: SnapshotForPhraseRender, + trackId: TrackId, + phraseKey: PhraseKey, + ) => { + const context: PhraseRenderContext = { + snapshot, + trackId, + phraseKey, + }; + for (const stage of stages) { + if (await stage.shouldBeExecuted(context)) { + return stage.id; + } + } + return undefined; + }, + render: async ( + snapshot: SnapshotForPhraseRender, + trackId: TrackId, + phraseKey: PhraseKey, + startStageId: PhraseRenderStageId, + ) => { + const context: PhraseRenderContext = { + snapshot, + trackId, + phraseKey, + }; + const startStageIndex = stages.findIndex((value) => { + return value.id === startStageId; + }); + if (startStageIndex === -1) { + throw new Error("Stage not found."); + } + for (let i = stages.length - 1; i >= startStageIndex; i--) { + stages[i].deleteExecutionResult(context); + } + for (let i = startStageIndex; i < stages.length; i++) { + await stages[i].execute(context); + } + }, + }; + // NOTE: 型推論でawaitの前か後かが考慮されないので、関数を介して取得する(型がbooleanになるようにする) const startRenderingRequested = () => state.startRenderingRequested; const stopRenderingRequested = () => state.stopRenderingRequested; @@ -1571,80 +2241,6 @@ export const singingStore = createPartialStore({ editorFrameRate: state.editorFrameRate, } as const; - const phraseRenderer = createPhraseRenderer({ - queryCache, - singingVolumeCache, - singingVoiceCache, - phrases: { - get: (phraseKey: PhraseKey) => { - const phrase = getOrThrow(state.phrases, phraseKey); - return { - firstRestDuration: phrase.firstRestDuration, - notes: phrase.notes, - startTime: phrase.startTime, - queryKey: { - get: () => getOrThrow(state.phrases, phraseKey).queryKey, - set: (value) => - mutations.SET_QUERY_KEY_TO_PHRASE({ - phraseKey, - queryKey: value, - }), - }, - singingVolumeKey: { - get: () => - getOrThrow(state.phrases, phraseKey).singingVolumeKey, - set: (value) => - mutations.SET_SINGING_VOLUME_KEY_TO_PHRASE({ - phraseKey, - singingVolumeKey: value, - }), - }, - singingVoiceKey: { - get: () => - getOrThrow(state.phrases, phraseKey).singingVoiceKey, - set: (value) => - mutations.SET_SINGING_VOICE_KEY_TO_PHRASE({ - phraseKey, - singingVoiceKey: value, - }), - }, - }; - }, - }, - phraseQueries: { - get: (queryKey) => getOrThrow(state.phraseQueries, queryKey), - set: (queryKey, query) => - mutations.SET_PHRASE_QUERY({ queryKey, query }), - delete: (queryKey) => mutations.DELETE_PHRASE_QUERY({ queryKey }), - }, - phraseSingingVolumes: { - get: (singingVolumeKey) => - getOrThrow(state.phraseSingingVolumes, singingVolumeKey), - set: (singingVolumeKey, singingVolume) => - mutations.SET_PHRASE_SINGING_VOLUME({ - singingVolumeKey, - singingVolume, - }), - delete: (singingVolumeKey) => - mutations.DELETE_PHRASE_SINGING_VOLUME({ singingVolumeKey }), - }, - phraseSingingVoices: { - set: (singingVoiceKey, singingVoice) => - phraseSingingVoices.set(singingVoiceKey, singingVoice), - delete: (singingVoiceKey) => - phraseSingingVoices.delete(singingVoiceKey), - }, - fetchQuery, - fetchSingFrameVolume: (notes, query, engineId, styleId) => - actions.FETCH_SING_FRAME_VOLUME({ - notes, - query, - engineId, - styleId, - }), - synthesizeSingingVoice, - }); - const renderStartStageIds = new Map(); // フレーズを更新する