2023-09-26 03:24:07 +00:00
|
|
|
package funkin.ui.debug.charting;
|
|
|
|
|
|
|
|
import flixel.system.FlxAssets.FlxSoundAsset;
|
|
|
|
import flixel.system.FlxSound;
|
|
|
|
import flixel.system.FlxSound;
|
2023-10-17 04:38:28 +00:00
|
|
|
import funkin.audio.VoicesGroup;
|
|
|
|
import funkin.play.character.BaseCharacter.CharacterType;
|
|
|
|
import funkin.util.FileUtil;
|
|
|
|
import haxe.io.Bytes;
|
2023-09-26 03:24:07 +00:00
|
|
|
import haxe.io.Path;
|
2023-10-17 04:38:28 +00:00
|
|
|
import openfl.utils.Assets;
|
2023-09-26 03:24:07 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Functions for loading audio for the chart editor.
|
|
|
|
*/
|
|
|
|
@:nullSafety
|
|
|
|
@:allow(funkin.ui.debug.charting.ChartEditorState)
|
|
|
|
@:allow(funkin.ui.debug.charting.ChartEditorDialogHandler)
|
|
|
|
@:allow(funkin.ui.debug.charting.ChartEditorImportExportHandler)
|
|
|
|
class ChartEditorAudioHandler
|
|
|
|
{
|
|
|
|
/**
|
2023-10-17 04:38:28 +00:00
|
|
|
* Loads and stores byte data for a vocal track from an absolute file path
|
|
|
|
*
|
2023-09-26 03:24:07 +00:00
|
|
|
* @param path The absolute path to the audio file.
|
2023-10-17 04:38:28 +00:00
|
|
|
* @param charId The character this vocal track will be for.
|
|
|
|
* @param instId The instrumental this vocal track will be for.
|
2023-09-26 03:24:07 +00:00
|
|
|
* @return Success or failure.
|
|
|
|
*/
|
2023-10-17 04:38:28 +00:00
|
|
|
static function loadVocalsFromPath(state:ChartEditorState, path:Path, charId:String, instId:String = ''):Bool
|
2023-09-26 03:24:07 +00:00
|
|
|
{
|
|
|
|
#if sys
|
2023-10-17 04:38:28 +00:00
|
|
|
var fileBytes:Bytes = sys.io.File.getBytes(path.toString());
|
|
|
|
return loadVocalsFromBytes(state, fileBytes, charId, instId);
|
2023-09-26 03:24:07 +00:00
|
|
|
#else
|
|
|
|
trace("[WARN] This platform can't load audio from a file path, you'll need to fetch the bytes some other way.");
|
|
|
|
return false;
|
|
|
|
#end
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-10-17 04:38:28 +00:00
|
|
|
* Loads and stores byte data for a vocal track from an asset
|
2023-09-26 03:24:07 +00:00
|
|
|
*
|
2023-10-17 04:38:28 +00:00
|
|
|
* @param path The path to the asset. Use `Paths` to build this.
|
|
|
|
* @param charId The character this vocal track will be for.
|
|
|
|
* @param instId The instrumental this vocal track will be for.
|
2023-09-26 03:24:07 +00:00
|
|
|
* @return Success or failure.
|
|
|
|
*/
|
2023-10-17 04:38:28 +00:00
|
|
|
static function loadVocalsFromAsset(state:ChartEditorState, path:String, charId:String, instId:String = ''):Bool
|
2023-09-26 03:24:07 +00:00
|
|
|
{
|
2023-10-17 04:38:28 +00:00
|
|
|
var trackData:Null<Bytes> = Assets.getBytes(path);
|
|
|
|
if (trackData != null)
|
2023-09-26 03:24:07 +00:00
|
|
|
{
|
2023-10-17 04:38:28 +00:00
|
|
|
return loadVocalsFromBytes(state, trackData, charId, instId);
|
2023-09-26 03:24:07 +00:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-10-17 04:38:28 +00:00
|
|
|
* Loads and stores byte data for a vocal track
|
|
|
|
*
|
|
|
|
* @param bytes The audio byte data.
|
|
|
|
* @param charId The character this vocal track will be for.
|
|
|
|
* @param instId The instrumental this vocal track will be for.
|
2023-09-26 03:24:07 +00:00
|
|
|
*/
|
2023-10-17 04:38:28 +00:00
|
|
|
static function loadVocalsFromBytes(state:ChartEditorState, bytes:Bytes, charId:String, instId:String = ''):Bool
|
2023-09-26 03:24:07 +00:00
|
|
|
{
|
2023-10-17 04:38:28 +00:00
|
|
|
var trackId:String = '${charId}${instId == '' ? '' : '-${instId}'}';
|
|
|
|
state.audioVocalTrackData.set(trackId, bytes);
|
2023-09-26 03:24:07 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-10-17 04:38:28 +00:00
|
|
|
* Loads and stores byte data for an instrumental track from an absolute file path
|
2023-09-26 03:24:07 +00:00
|
|
|
*
|
|
|
|
* @param path The absolute path to the audio file.
|
2023-10-17 04:38:28 +00:00
|
|
|
* @param instId The instrumental this vocal track will be for.
|
2023-09-26 03:24:07 +00:00
|
|
|
* @return Success or failure.
|
|
|
|
*/
|
2023-10-17 04:38:28 +00:00
|
|
|
static function loadInstFromPath(state:ChartEditorState, path:Path, instId:String = ''):Bool
|
2023-09-26 03:24:07 +00:00
|
|
|
{
|
|
|
|
#if sys
|
2023-10-17 04:38:28 +00:00
|
|
|
var fileBytes:Bytes = sys.io.File.getBytes(path.toString());
|
|
|
|
return loadInstFromBytes(state, fileBytes, instId);
|
2023-09-26 03:24:07 +00:00
|
|
|
#else
|
|
|
|
trace("[WARN] This platform can't load audio from a file path, you'll need to fetch the bytes some other way.");
|
|
|
|
return false;
|
|
|
|
#end
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-10-17 04:38:28 +00:00
|
|
|
* Loads and stores byte data for an instrumental track from an asset
|
|
|
|
*
|
|
|
|
* @param path The path to the asset. Use `Paths` to build this.
|
|
|
|
* @param instId The instrumental this vocal track will be for.
|
2023-09-26 03:24:07 +00:00
|
|
|
* @return Success or failure.
|
|
|
|
*/
|
2023-10-17 04:38:28 +00:00
|
|
|
static function loadInstFromAsset(state:ChartEditorState, path:String, instId:String = ''):Bool
|
2023-09-26 03:24:07 +00:00
|
|
|
{
|
2023-10-17 04:38:28 +00:00
|
|
|
var trackData:Null<Bytes> = Assets.getBytes(path);
|
|
|
|
if (trackData != null)
|
2023-09-26 03:24:07 +00:00
|
|
|
{
|
2023-10-17 04:38:28 +00:00
|
|
|
return loadInstFromBytes(state, trackData, instId);
|
2023-09-26 03:24:07 +00:00
|
|
|
}
|
2023-10-17 04:38:28 +00:00
|
|
|
return false;
|
|
|
|
}
|
2023-09-26 03:24:07 +00:00
|
|
|
|
2023-10-17 04:38:28 +00:00
|
|
|
/**
|
|
|
|
* Loads and stores byte data for a vocal track
|
|
|
|
*
|
|
|
|
* @param bytes The audio byte data.
|
|
|
|
* @param charId The character this vocal track will be for.
|
|
|
|
* @param instId The instrumental this vocal track will be for.
|
|
|
|
*/
|
|
|
|
static function loadInstFromBytes(state:ChartEditorState, bytes:Bytes, instId:String = ''):Bool
|
|
|
|
{
|
|
|
|
if (instId == '') instId = 'default';
|
|
|
|
state.audioInstTrackData.set(instId, bytes);
|
|
|
|
return true;
|
|
|
|
}
|
2023-09-26 03:24:07 +00:00
|
|
|
|
2023-10-17 04:38:28 +00:00
|
|
|
public static function switchToInstrumental(state:ChartEditorState, instId:String = '', playerId:String, opponentId:String):Bool
|
|
|
|
{
|
|
|
|
var result:Bool = playInstrumental(state, instId);
|
|
|
|
if (!result) return false;
|
2023-09-26 03:24:07 +00:00
|
|
|
|
2023-10-17 04:38:28 +00:00
|
|
|
stopExistingVocals(state);
|
|
|
|
result = playVocals(state, BF, playerId, instId);
|
|
|
|
if (!result) return false;
|
|
|
|
result = playVocals(state, DAD, opponentId, instId);
|
|
|
|
if (!result) return false;
|
2023-09-26 03:24:07 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-10-17 04:38:28 +00:00
|
|
|
* Tell the Chart Editor to select a specific instrumental track, that is already loaded.
|
2023-09-26 03:24:07 +00:00
|
|
|
*/
|
2023-10-17 04:38:28 +00:00
|
|
|
static function playInstrumental(state:ChartEditorState, instId:String = ''):Bool
|
2023-09-26 03:24:07 +00:00
|
|
|
{
|
2023-10-17 04:38:28 +00:00
|
|
|
if (instId == '') instId = 'default';
|
|
|
|
var instTrackData:Null<Bytes> = state.audioInstTrackData.get(instId);
|
|
|
|
var instTrack:Null<FlxSound> = buildFlxSoundFromBytes(instTrackData);
|
|
|
|
if (instTrack == null) return false;
|
2023-09-26 03:24:07 +00:00
|
|
|
|
2023-10-17 04:38:28 +00:00
|
|
|
stopExistingInstrumental(state);
|
|
|
|
state.audioInstTrack = instTrack;
|
|
|
|
state.postLoadInstrumental();
|
|
|
|
return true;
|
|
|
|
}
|
2023-09-26 03:24:07 +00:00
|
|
|
|
2023-10-17 04:38:28 +00:00
|
|
|
static function stopExistingInstrumental(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
if (state.audioInstTrack != null)
|
|
|
|
{
|
|
|
|
state.audioInstTrack.stop();
|
|
|
|
state.audioInstTrack.destroy();
|
|
|
|
state.audioInstTrack = null;
|
2023-09-26 03:24:07 +00:00
|
|
|
}
|
2023-10-17 04:38:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tell the Chart Editor to select a specific vocal track, that is already loaded.
|
|
|
|
*/
|
|
|
|
static function playVocals(state:ChartEditorState, charType:CharacterType, charId:String, instId:String = ''):Bool
|
|
|
|
{
|
|
|
|
var trackId:String = '${charId}${instId == '' ? '' : '-${instId}'}';
|
|
|
|
var vocalTrackData:Null<Bytes> = state.audioVocalTrackData.get(trackId);
|
|
|
|
var vocalTrack:Null<FlxSound> = buildFlxSoundFromBytes(vocalTrackData);
|
2023-09-26 03:24:07 +00:00
|
|
|
|
2023-10-17 04:38:28 +00:00
|
|
|
if (state.audioVocalTrackGroup == null) state.audioVocalTrackGroup = new VoicesGroup();
|
|
|
|
|
|
|
|
if (vocalTrack != null)
|
|
|
|
{
|
|
|
|
switch (charType)
|
|
|
|
{
|
|
|
|
case BF:
|
|
|
|
state.audioVocalTrackGroup.addPlayerVoice(vocalTrack);
|
|
|
|
return true;
|
|
|
|
case DAD:
|
|
|
|
state.audioVocalTrackGroup.addOpponentVoice(vocalTrack);
|
|
|
|
return true;
|
|
|
|
case OTHER:
|
|
|
|
state.audioVocalTrackGroup.add(vocalTrack);
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
// Do nothing.
|
|
|
|
}
|
|
|
|
}
|
2023-09-26 03:24:07 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-10-17 04:38:28 +00:00
|
|
|
static function stopExistingVocals(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
if (state.audioVocalTrackGroup != null)
|
|
|
|
{
|
|
|
|
state.audioVocalTrackGroup.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-09-26 03:24:07 +00:00
|
|
|
/**
|
|
|
|
* Play a sound effect.
|
|
|
|
* Automatically cleans up after itself and recycles previous FlxSound instances if available, for performance.
|
2023-10-17 04:38:28 +00:00
|
|
|
* @param path The path to the sound effect. Use `Paths` to build this.
|
2023-09-26 03:24:07 +00:00
|
|
|
*/
|
|
|
|
public static function playSound(path:String):Void
|
|
|
|
{
|
|
|
|
var snd:FlxSound = FlxG.sound.list.recycle(FlxSound) ?? new FlxSound();
|
|
|
|
var asset:Null<FlxSoundAsset> = FlxG.sound.cache(path);
|
|
|
|
if (asset == null)
|
|
|
|
{
|
|
|
|
trace('WARN: Failed to play sound $path, asset not found.');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
snd.loadEmbedded(asset);
|
|
|
|
snd.autoDestroy = true;
|
|
|
|
FlxG.sound.list.add(snd);
|
|
|
|
snd.play();
|
|
|
|
}
|
2023-10-17 04:38:28 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Convert byte data into a playable sound.
|
|
|
|
*
|
|
|
|
* @param input The byte data.
|
|
|
|
* @return The playable sound, or `null` if loading failed.
|
|
|
|
*/
|
|
|
|
public static function buildFlxSoundFromBytes(input:Null<Bytes>):Null<FlxSound>
|
|
|
|
{
|
|
|
|
if (input == null) return null;
|
|
|
|
|
|
|
|
var openflSound:openfl.media.Sound = new openfl.media.Sound();
|
|
|
|
openflSound.loadCompressedDataFromByteArray(openfl.utils.ByteArray.fromBytes(input), input.length);
|
|
|
|
var output:FlxSound = FlxG.sound.load(openflSound, 1.0, false);
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
|
|
|
static function makeZIPEntriesFromInstrumentals(state:ChartEditorState):Array<haxe.zip.Entry>
|
|
|
|
{
|
|
|
|
var zipEntries = [];
|
|
|
|
|
|
|
|
for (key in state.audioInstTrackData.keys())
|
|
|
|
{
|
|
|
|
if (key == 'default')
|
|
|
|
{
|
|
|
|
var data:Null<Bytes> = state.audioInstTrackData.get('default');
|
|
|
|
if (data == null) continue;
|
|
|
|
zipEntries.push(FileUtil.makeZIPEntryFromBytes('Inst.ogg', data));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
var data:Null<Bytes> = state.audioInstTrackData.get(key);
|
|
|
|
if (data == null) continue;
|
|
|
|
zipEntries.push(FileUtil.makeZIPEntryFromBytes('Inst-${key}.ogg', data));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return zipEntries;
|
|
|
|
}
|
|
|
|
|
|
|
|
static function makeZIPEntriesFromVocals(state:ChartEditorState):Array<haxe.zip.Entry>
|
|
|
|
{
|
|
|
|
var zipEntries = [];
|
|
|
|
|
|
|
|
for (key in state.audioVocalTrackData.keys())
|
|
|
|
{
|
|
|
|
var data:Null<Bytes> = state.audioVocalTrackData.get(key);
|
|
|
|
if (data == null) continue;
|
|
|
|
zipEntries.push(FileUtil.makeZIPEntryFromBytes('Vocals-${key}.ogg', data));
|
|
|
|
}
|
|
|
|
|
|
|
|
return zipEntries;
|
|
|
|
}
|
2023-09-26 03:24:07 +00:00
|
|
|
}
|