voice-changer/client/lib/worklet/src/voice-changer-worklet-processor.ts

175 lines
5.2 KiB
TypeScript
Raw Normal View History

2023-01-11 21:49:22 +03:00
export const RequestType = {
"voice": "voice",
"config": "config",
2023-02-19 22:49:34 +03:00
"start": "start",
2023-02-28 05:54:40 +03:00
"stop": "stop",
"trancateBuffer": "trancateBuffer",
2023-01-11 21:49:22 +03:00
} as const
export type RequestType = typeof RequestType[keyof typeof RequestType]
export const ResponseType = {
"volume": "volume",
2023-02-19 19:00:28 +03:00
"inputData": "inputData"
} as const
export type ResponseType = typeof ResponseType[keyof typeof ResponseType]
2023-01-11 21:49:22 +03:00
export type VoiceChangerWorkletProcessorRequest = {
requestType: RequestType,
2023-02-19 22:49:34 +03:00
voice: Float32Array,
2023-01-11 21:49:22 +03:00
numTrancateTreshold: number
volTrancateThreshold: number
volTrancateLength: number
}
2023-01-04 20:28:36 +03:00
export type VoiceChangerWorkletProcessorResponse = {
responseType: ResponseType,
volume?: number,
recordData?: Float32Array[]
2023-02-19 19:00:28 +03:00
inputData?: Float32Array
}
2023-01-04 20:28:36 +03:00
class VoiceChangerWorkletProcessor extends AudioWorkletProcessor {
private BLOCK_SIZE = 128
private initialized = false;
private volume = 0
2023-01-11 22:52:01 +03:00
private numTrancateTreshold = 150
2023-01-11 21:49:22 +03:00
private volTrancateThreshold = 0.0005
private volTrancateLength = 32
private volTrancateCount = 0
private isRecording = false
2023-01-04 20:28:36 +03:00
playBuffer: Float32Array[] = []
/**
* @constructor
*/
constructor() {
super();
this.initialized = true;
this.port.onmessage = this.handleMessage.bind(this);
}
2023-01-11 21:49:22 +03:00
calcVol = (data: Float32Array, prevVol: number) => {
const sum = data.reduce((prev, cur) => {
return prev + cur * cur
}, 0)
const rms = Math.sqrt(sum / data.length)
return Math.max(rms, prevVol * 0.95)
}
2023-02-28 05:54:40 +03:00
trancateBuffer = () => {
console.log("[worklet] Buffer truncated")
while (this.playBuffer.length > 2) {
this.playBuffer.shift()
}
}
2023-01-04 20:28:36 +03:00
handleMessage(event: any) {
const request = event.data as VoiceChangerWorkletProcessorRequest
2023-01-11 21:49:22 +03:00
if (request.requestType === "config") {
this.numTrancateTreshold = request.numTrancateTreshold
this.volTrancateLength = request.volTrancateLength
this.volTrancateThreshold = request.volTrancateThreshold
console.log("[worklet] worklet configured", request)
return
2023-02-19 22:49:34 +03:00
} else if (request.requestType === "start") {
if (this.isRecording) {
console.warn("[worklet] recoring is already started")
return
}
this.isRecording = true
return
2023-02-19 22:49:34 +03:00
} else if (request.requestType === "stop") {
if (!this.isRecording) {
console.warn("[worklet] recoring is not started")
return
}
this.isRecording = false
return
2023-02-28 05:54:40 +03:00
} else if (request.requestType === "trancateBuffer") {
this.trancateBuffer()
return
2023-01-11 21:49:22 +03:00
}
if (this.playBuffer.length > this.numTrancateTreshold) {
2023-02-28 05:54:40 +03:00
this.trancateBuffer()
2023-01-04 20:28:36 +03:00
}
2023-02-19 22:49:34 +03:00
const f32Data = request.voice
const chunkNum = f32Data.length / this.BLOCK_SIZE
for (let i = 0; i < chunkNum; i++) {
const block = f32Data.slice(i * this.BLOCK_SIZE, (i + 1) * this.BLOCK_SIZE)
this.playBuffer.push(block)
2023-01-04 20:28:36 +03:00
}
}
2023-02-19 19:00:28 +03:00
pushData = (inputData: Float32Array) => {
const volumeResponse: VoiceChangerWorkletProcessorResponse = {
responseType: ResponseType.inputData,
inputData: inputData
}
this.port.postMessage(volumeResponse);
}
2023-01-04 20:28:36 +03:00
process(_inputs: Float32Array[][], outputs: Float32Array[][], _parameters: Record<string, Float32Array>) {
if (!this.initialized) {
console.warn("[worklet] worklet_process not ready");
return true;
}
2023-02-19 20:21:51 +03:00
if (this.isRecording) {
if (_inputs.length > 0 && _inputs[0].length > 0) {
this.pushData(_inputs[0][0])
}
2023-02-19 19:00:28 +03:00
}
2023-01-04 20:28:36 +03:00
if (this.playBuffer.length === 0) {
2023-01-11 22:52:01 +03:00
// console.log("[worklet] no play buffer")
2023-01-04 20:28:36 +03:00
return true
}
2023-01-11 21:49:22 +03:00
//// 一定期間無音状態が続いている場合はスキップ。
let voice: Float32Array | undefined
while (true) {
voice = this.playBuffer.shift()
if (!voice) {
break
}
this.volume = this.calcVol(voice, this.volume)
if (this.volume < this.volTrancateThreshold) {
this.volTrancateCount += 1
} else {
this.volTrancateCount = 0
}
2023-01-04 20:28:36 +03:00
// V.1.5.0よりsilent skipで音飛びするようになったので無効化
2023-01-11 22:52:01 +03:00
if (this.volTrancateCount < this.volTrancateLength || this.volTrancateLength < 0) {
2023-01-11 21:49:22 +03:00
break
} else {
break
2023-01-11 22:52:01 +03:00
// console.log("silent...skip")
2023-01-11 21:49:22 +03:00
}
}
2023-01-04 20:28:36 +03:00
2023-01-11 21:49:22 +03:00
if (voice) {
const volumeResponse: VoiceChangerWorkletProcessorResponse = {
responseType: ResponseType.volume,
volume: this.volume
}
this.port.postMessage(volumeResponse);
2023-01-11 21:49:22 +03:00
outputs[0][0].set(voice)
2023-02-23 20:51:52 +03:00
if (outputs[0].length == 2) {
outputs[0][1].set(voice)
}
2023-01-11 21:49:22 +03:00
}
2023-01-04 20:28:36 +03:00
return true;
}
}
registerProcessor("voice-changer-worklet-processor", VoiceChangerWorkletProcessor);