2022-10-07 04:37:21 +00:00
|
|
|
package funkin.ui.debug.charting;
|
|
|
|
|
2023-10-17 04:38:28 +00:00
|
|
|
import haxe.ui.notifications.NotificationType;
|
|
|
|
import haxe.ui.notifications.NotificationManager;
|
2023-09-08 21:46:44 +00:00
|
|
|
import funkin.data.song.SongData.SongEventData;
|
|
|
|
import funkin.data.song.SongData.SongNoteData;
|
|
|
|
import funkin.data.song.SongDataUtils;
|
2022-10-11 00:04:09 +00:00
|
|
|
|
|
|
|
using Lambda;
|
2022-10-07 04:37:21 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Actions in the chart editor are backed by the Command pattern
|
|
|
|
* (see Bob Nystrom's book "Game Programming Patterns" for more info)
|
|
|
|
*
|
|
|
|
* To make a function compatible with the undo/redo history, create a new class
|
|
|
|
* that implements ChartEditorCommand, then call `ChartEditorState.performCommand(new Command())`
|
|
|
|
*/
|
|
|
|
interface ChartEditorCommand
|
|
|
|
{
|
2023-01-23 03:25:45 +00:00
|
|
|
/**
|
|
|
|
* Calling this function should perform the action that this command represents.
|
|
|
|
* @param state The ChartEditorState to perform the action on.
|
|
|
|
*/
|
|
|
|
public function execute(state:ChartEditorState):Void;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calling this function should perform the inverse of the action that this command represents,
|
|
|
|
* effectively undoing the action.
|
|
|
|
* @param state The ChartEditorState to undo the action on.
|
|
|
|
*/
|
|
|
|
public function undo(state:ChartEditorState):Void;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a short description of the action (for the UI).
|
|
|
|
* For example, return `Add Left Note` to display `Undo Add Left Note` in the menu.
|
|
|
|
*/
|
|
|
|
public function toString():String;
|
2022-10-07 04:37:21 +00:00
|
|
|
}
|
|
|
|
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2022-10-11 00:04:09 +00:00
|
|
|
class AddNotesCommand implements ChartEditorCommand
|
2022-10-07 04:37:21 +00:00
|
|
|
{
|
2023-01-23 03:25:45 +00:00
|
|
|
var notes:Array<SongNoteData>;
|
|
|
|
var appendToSelection:Bool;
|
|
|
|
|
|
|
|
public function new(notes:Array<SongNoteData>, appendToSelection:Bool = false)
|
|
|
|
{
|
|
|
|
this.notes = notes;
|
|
|
|
this.appendToSelection = appendToSelection;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function execute(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
for (note in notes)
|
|
|
|
{
|
|
|
|
state.currentSongChartNoteData.push(note);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (appendToSelection)
|
|
|
|
{
|
|
|
|
state.currentNoteSelection = state.currentNoteSelection.concat(notes);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
state.currentNoteSelection = notes;
|
|
|
|
state.currentEventSelection = [];
|
|
|
|
}
|
|
|
|
|
2023-10-11 23:39:52 +00:00
|
|
|
ChartEditorAudioHandler.playSound(Paths.sound('chartingSounds/noteLay'));
|
2023-01-23 03:25:45 +00:00
|
|
|
|
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
|
|
|
|
state.sortChartData();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function undo(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentSongChartNoteData = SongDataUtils.subtractNotes(state.currentSongChartNoteData, notes);
|
|
|
|
state.currentNoteSelection = [];
|
|
|
|
state.currentEventSelection = [];
|
2023-10-11 23:39:52 +00:00
|
|
|
ChartEditorAudioHandler.playSound(Paths.sound('chartingSounds/undo'));
|
2023-01-23 03:25:45 +00:00
|
|
|
|
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
|
|
|
|
state.sortChartData();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
if (notes.length == 1)
|
|
|
|
{
|
|
|
|
var dir:String = notes[0].getDirectionName();
|
|
|
|
return 'Add $dir Note';
|
|
|
|
}
|
|
|
|
|
|
|
|
return 'Add ${notes.length} Notes';
|
|
|
|
}
|
2022-10-07 04:37:21 +00:00
|
|
|
}
|
|
|
|
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2022-10-11 07:14:57 +00:00
|
|
|
class RemoveNotesCommand implements ChartEditorCommand
|
2022-10-07 04:37:21 +00:00
|
|
|
{
|
2023-01-23 03:25:45 +00:00
|
|
|
var notes:Array<SongNoteData>;
|
|
|
|
|
|
|
|
public function new(notes:Array<SongNoteData>)
|
|
|
|
{
|
|
|
|
this.notes = notes;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function execute(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentSongChartNoteData = SongDataUtils.subtractNotes(state.currentSongChartNoteData, notes);
|
|
|
|
state.currentNoteSelection = [];
|
|
|
|
state.currentEventSelection = [];
|
2023-10-15 02:49:24 +00:00
|
|
|
|
|
|
|
ChartEditorAudioHandler.playSound(Paths.sound('chartingSounds/noteErase'));
|
2023-01-23 03:25:45 +00:00
|
|
|
|
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
|
|
|
|
state.sortChartData();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function undo(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
for (note in notes)
|
|
|
|
{
|
|
|
|
state.currentSongChartNoteData.push(note);
|
|
|
|
}
|
|
|
|
state.currentNoteSelection = notes;
|
|
|
|
state.currentEventSelection = [];
|
2023-10-11 23:39:52 +00:00
|
|
|
ChartEditorAudioHandler.playSound(Paths.sound('chartingSounds/undo'));
|
2023-01-23 03:25:45 +00:00
|
|
|
|
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
|
|
|
|
state.sortChartData();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
if (notes.length == 1 && notes[0] != null)
|
|
|
|
{
|
|
|
|
var dir:String = notes[0].getDirectionName();
|
|
|
|
return 'Remove $dir Note';
|
|
|
|
}
|
|
|
|
|
|
|
|
return 'Remove ${notes.length} Notes';
|
|
|
|
}
|
2022-10-07 04:37:21 +00:00
|
|
|
}
|
|
|
|
|
2023-01-23 00:55:30 +00:00
|
|
|
/**
|
|
|
|
* Appends one or more items to the selection.
|
|
|
|
*/
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2023-01-23 00:55:30 +00:00
|
|
|
class SelectItemsCommand implements ChartEditorCommand
|
2022-10-07 04:37:21 +00:00
|
|
|
{
|
2023-01-23 03:25:45 +00:00
|
|
|
var notes:Array<SongNoteData>;
|
|
|
|
var events:Array<SongEventData>;
|
|
|
|
|
|
|
|
public function new(notes:Array<SongNoteData>, events:Array<SongEventData>)
|
|
|
|
{
|
|
|
|
this.notes = notes;
|
|
|
|
this.events = events;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function execute(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
for (note in this.notes)
|
|
|
|
{
|
|
|
|
state.currentNoteSelection.push(note);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (event in this.events)
|
|
|
|
{
|
|
|
|
state.currentEventSelection.push(event);
|
|
|
|
}
|
|
|
|
|
2023-10-21 05:04:50 +00:00
|
|
|
// state.noteDisplayDirty = true;
|
|
|
|
// state.notePreviewDirty = true;
|
2023-01-23 03:25:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public function undo(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentNoteSelection = SongDataUtils.subtractNotes(state.currentNoteSelection, this.notes);
|
|
|
|
state.currentEventSelection = SongDataUtils.subtractEvents(state.currentEventSelection, this.events);
|
|
|
|
|
2023-10-21 05:04:50 +00:00
|
|
|
// state.noteDisplayDirty = true;
|
|
|
|
// state.notePreviewDirty = true;
|
2023-01-23 03:25:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
var len:Int = notes.length + events.length;
|
|
|
|
|
|
|
|
if (notes.length == 0)
|
|
|
|
{
|
|
|
|
if (events.length == 1)
|
|
|
|
{
|
|
|
|
return 'Select Event';
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return 'Select ${events.length} Events';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (events.length == 0)
|
|
|
|
{
|
|
|
|
if (notes.length == 1)
|
|
|
|
{
|
|
|
|
return 'Select Note';
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return 'Select ${notes.length} Notes';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 'Select ${len} Items';
|
|
|
|
}
|
2022-10-07 04:37:21 +00:00
|
|
|
}
|
2022-10-11 00:04:09 +00:00
|
|
|
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2023-01-23 00:55:30 +00:00
|
|
|
class AddEventsCommand implements ChartEditorCommand
|
2022-10-11 00:04:09 +00:00
|
|
|
{
|
2023-01-23 03:25:45 +00:00
|
|
|
var events:Array<SongEventData>;
|
|
|
|
var appendToSelection:Bool;
|
|
|
|
|
2023-08-28 19:03:29 +00:00
|
|
|
public function new(events:Array<SongEventData>, appendToSelection:Bool = false)
|
2023-01-23 03:25:45 +00:00
|
|
|
{
|
|
|
|
this.events = events;
|
|
|
|
this.appendToSelection = appendToSelection;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function execute(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
for (event in events)
|
|
|
|
{
|
|
|
|
state.currentSongChartEventData.push(event);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (appendToSelection)
|
|
|
|
{
|
|
|
|
state.currentEventSelection = state.currentEventSelection.concat(events);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
state.currentNoteSelection = [];
|
|
|
|
state.currentEventSelection = events;
|
|
|
|
}
|
|
|
|
|
2023-10-11 23:39:52 +00:00
|
|
|
ChartEditorAudioHandler.playSound(Paths.sound('chartingSounds/noteLay'));
|
2023-01-23 03:25:45 +00:00
|
|
|
|
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
|
|
|
|
state.sortChartData();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function undo(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentSongChartEventData = SongDataUtils.subtractEvents(state.currentSongChartEventData, events);
|
|
|
|
|
|
|
|
state.currentNoteSelection = [];
|
|
|
|
state.currentEventSelection = [];
|
|
|
|
|
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
|
|
|
|
state.sortChartData();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
var len:Int = events.length;
|
|
|
|
return 'Add $len Events';
|
|
|
|
}
|
2023-01-23 00:55:30 +00:00
|
|
|
}
|
|
|
|
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2023-01-23 00:55:30 +00:00
|
|
|
class RemoveEventsCommand implements ChartEditorCommand
|
|
|
|
{
|
2023-01-23 03:25:45 +00:00
|
|
|
var events:Array<SongEventData>;
|
|
|
|
|
|
|
|
public function new(events:Array<SongEventData>)
|
|
|
|
{
|
|
|
|
this.events = events;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function execute(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentSongChartEventData = SongDataUtils.subtractEvents(state.currentSongChartEventData, events);
|
|
|
|
state.currentEventSelection = [];
|
2023-10-15 02:49:24 +00:00
|
|
|
|
|
|
|
ChartEditorAudioHandler.playSound(Paths.sound('chartingSounds/noteErase'));
|
2023-01-23 03:25:45 +00:00
|
|
|
|
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
|
|
|
|
state.sortChartData();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function undo(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
for (event in events)
|
|
|
|
{
|
|
|
|
state.currentSongChartEventData.push(event);
|
|
|
|
}
|
|
|
|
state.currentEventSelection = events;
|
2023-10-11 23:39:52 +00:00
|
|
|
ChartEditorAudioHandler.playSound(Paths.sound('chartingSounds/undo'));
|
2023-01-23 03:25:45 +00:00
|
|
|
|
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
|
|
|
|
state.sortChartData();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
if (events.length == 1 && events[0] != null)
|
|
|
|
{
|
|
|
|
return 'Remove Event';
|
|
|
|
}
|
|
|
|
|
|
|
|
return 'Remove ${events.length} Events';
|
|
|
|
}
|
2022-10-11 00:04:09 +00:00
|
|
|
}
|
|
|
|
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2023-01-23 00:55:30 +00:00
|
|
|
class RemoveItemsCommand implements ChartEditorCommand
|
2022-10-11 00:04:09 +00:00
|
|
|
{
|
2023-01-23 03:25:45 +00:00
|
|
|
var notes:Array<SongNoteData>;
|
|
|
|
var events:Array<SongEventData>;
|
2022-10-11 00:04:09 +00:00
|
|
|
|
2023-01-23 03:25:45 +00:00
|
|
|
public function new(notes:Array<SongNoteData>, events:Array<SongEventData>)
|
|
|
|
{
|
|
|
|
this.notes = notes;
|
|
|
|
this.events = events;
|
|
|
|
}
|
2022-10-11 00:04:09 +00:00
|
|
|
|
2023-01-23 03:25:45 +00:00
|
|
|
public function execute(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentSongChartNoteData = SongDataUtils.subtractNotes(state.currentSongChartNoteData, notes);
|
|
|
|
state.currentSongChartEventData = SongDataUtils.subtractEvents(state.currentSongChartEventData, events);
|
2023-01-23 00:55:30 +00:00
|
|
|
|
2023-01-23 03:25:45 +00:00
|
|
|
state.currentNoteSelection = [];
|
|
|
|
state.currentEventSelection = [];
|
2023-01-23 00:55:30 +00:00
|
|
|
|
2023-10-15 02:49:24 +00:00
|
|
|
ChartEditorAudioHandler.playSound(Paths.sound('chartingSounds/noteErase'));
|
2023-01-23 00:55:30 +00:00
|
|
|
|
2023-01-23 03:25:45 +00:00
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
2023-01-23 00:55:30 +00:00
|
|
|
|
2023-01-23 03:25:45 +00:00
|
|
|
state.sortChartData();
|
|
|
|
}
|
2023-01-23 00:55:30 +00:00
|
|
|
|
2023-01-23 03:25:45 +00:00
|
|
|
public function undo(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
for (note in notes)
|
|
|
|
{
|
|
|
|
state.currentSongChartNoteData.push(note);
|
|
|
|
}
|
2023-01-23 00:55:30 +00:00
|
|
|
|
2023-01-23 03:25:45 +00:00
|
|
|
for (event in events)
|
|
|
|
{
|
|
|
|
state.currentSongChartEventData.push(event);
|
|
|
|
}
|
2023-01-23 00:55:30 +00:00
|
|
|
|
2023-01-23 03:25:45 +00:00
|
|
|
state.currentNoteSelection = notes;
|
|
|
|
state.currentEventSelection = events;
|
2023-01-23 00:55:30 +00:00
|
|
|
|
2023-10-11 23:39:52 +00:00
|
|
|
ChartEditorAudioHandler.playSound(Paths.sound('chartingSounds/undo'));
|
2023-01-23 00:55:30 +00:00
|
|
|
|
2023-01-23 03:25:45 +00:00
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
2023-01-23 00:55:30 +00:00
|
|
|
|
2023-01-23 03:25:45 +00:00
|
|
|
state.sortChartData();
|
|
|
|
}
|
2023-01-23 00:55:30 +00:00
|
|
|
|
2023-01-23 03:25:45 +00:00
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
return 'Remove ${notes.length + events.length} Items';
|
|
|
|
}
|
2023-01-23 00:55:30 +00:00
|
|
|
}
|
|
|
|
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2023-01-23 00:55:30 +00:00
|
|
|
class SwitchDifficultyCommand implements ChartEditorCommand
|
|
|
|
{
|
2023-01-23 03:25:45 +00:00
|
|
|
var prevDifficulty:String;
|
|
|
|
var newDifficulty:String;
|
|
|
|
var prevVariation:String;
|
|
|
|
var newVariation:String;
|
|
|
|
|
|
|
|
public function new(prevDifficulty:String, newDifficulty:String, prevVariation:String, newVariation:String)
|
|
|
|
{
|
|
|
|
this.prevDifficulty = prevDifficulty;
|
|
|
|
this.newDifficulty = newDifficulty;
|
|
|
|
this.prevVariation = prevVariation;
|
|
|
|
this.newVariation = newVariation;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function execute(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.selectedVariation = newVariation != null ? newVariation : prevVariation;
|
|
|
|
state.selectedDifficulty = newDifficulty != null ? newDifficulty : prevDifficulty;
|
|
|
|
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function undo(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.selectedVariation = prevVariation != null ? prevVariation : newVariation;
|
|
|
|
state.selectedDifficulty = prevDifficulty != null ? prevDifficulty : newDifficulty;
|
|
|
|
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
return 'Switch Difficulty';
|
|
|
|
}
|
2023-01-23 00:55:30 +00:00
|
|
|
}
|
|
|
|
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2023-01-23 00:55:30 +00:00
|
|
|
class DeselectItemsCommand implements ChartEditorCommand
|
|
|
|
{
|
2023-01-23 03:25:45 +00:00
|
|
|
var notes:Array<SongNoteData>;
|
|
|
|
var events:Array<SongEventData>;
|
|
|
|
|
|
|
|
public function new(notes:Array<SongNoteData>, events:Array<SongEventData>)
|
|
|
|
{
|
|
|
|
this.notes = notes;
|
|
|
|
this.events = events;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function execute(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentNoteSelection = SongDataUtils.subtractNotes(state.currentNoteSelection, this.notes);
|
|
|
|
state.currentEventSelection = SongDataUtils.subtractEvents(state.currentEventSelection, this.events);
|
|
|
|
|
2023-10-21 05:04:50 +00:00
|
|
|
// state.noteDisplayDirty = true;
|
|
|
|
// state.notePreviewDirty = true;
|
2023-01-23 03:25:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public function undo(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
for (note in this.notes)
|
|
|
|
{
|
|
|
|
state.currentNoteSelection.push(note);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (event in this.events)
|
|
|
|
{
|
|
|
|
state.currentEventSelection.push(event);
|
|
|
|
}
|
|
|
|
|
2023-10-21 05:04:50 +00:00
|
|
|
// state.noteDisplayDirty = true;
|
|
|
|
// state.notePreviewDirty = true;
|
2023-01-23 03:25:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
var noteCount = notes.length + events.length;
|
|
|
|
|
|
|
|
if (noteCount == 1)
|
|
|
|
{
|
|
|
|
var dir:String = notes[0].getDirectionName();
|
|
|
|
return 'Deselect $dir Items';
|
|
|
|
}
|
|
|
|
|
|
|
|
return 'Deselect ${noteCount} Items';
|
|
|
|
}
|
2022-10-11 00:04:09 +00:00
|
|
|
}
|
|
|
|
|
2022-10-26 05:14:29 +00:00
|
|
|
/**
|
|
|
|
* Sets the selection rather than appends it.
|
|
|
|
* Deselects any notes that are not in the new selection.
|
|
|
|
*/
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2023-01-23 00:55:30 +00:00
|
|
|
class SetItemSelectionCommand implements ChartEditorCommand
|
2022-10-26 05:14:29 +00:00
|
|
|
{
|
2023-01-23 03:25:45 +00:00
|
|
|
var notes:Array<SongNoteData>;
|
|
|
|
var events:Array<SongEventData>;
|
|
|
|
var previousNoteSelection:Array<SongNoteData>;
|
|
|
|
var previousEventSelection:Array<SongEventData>;
|
|
|
|
|
|
|
|
public function new(notes:Array<SongNoteData>, events:Array<SongEventData>, previousNoteSelection:Array<SongNoteData>,
|
|
|
|
previousEventSelection:Array<SongEventData>)
|
|
|
|
{
|
|
|
|
this.notes = notes;
|
|
|
|
this.events = events;
|
|
|
|
this.previousNoteSelection = previousNoteSelection == null ? [] : previousNoteSelection;
|
|
|
|
this.previousEventSelection = previousEventSelection == null ? [] : previousEventSelection;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function execute(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentNoteSelection = notes;
|
|
|
|
state.currentEventSelection = events;
|
|
|
|
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function undo(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentNoteSelection = previousNoteSelection;
|
|
|
|
state.currentEventSelection = previousEventSelection;
|
|
|
|
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
return 'Select ${notes.length} Items';
|
|
|
|
}
|
2022-10-26 05:14:29 +00:00
|
|
|
}
|
|
|
|
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2023-01-23 00:55:30 +00:00
|
|
|
class SelectAllItemsCommand implements ChartEditorCommand
|
2022-10-11 00:04:09 +00:00
|
|
|
{
|
2023-01-23 03:25:45 +00:00
|
|
|
var previousNoteSelection:Array<SongNoteData>;
|
|
|
|
var previousEventSelection:Array<SongEventData>;
|
|
|
|
|
|
|
|
public function new(?previousNoteSelection:Array<SongNoteData>, ?previousEventSelection:Array<SongEventData>)
|
|
|
|
{
|
|
|
|
this.previousNoteSelection = previousNoteSelection == null ? [] : previousNoteSelection;
|
|
|
|
this.previousEventSelection = previousEventSelection == null ? [] : previousEventSelection;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function execute(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentNoteSelection = state.currentSongChartNoteData;
|
|
|
|
state.currentEventSelection = state.currentSongChartEventData;
|
|
|
|
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function undo(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentNoteSelection = previousNoteSelection;
|
|
|
|
state.currentEventSelection = previousEventSelection;
|
|
|
|
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
return 'Select All Items';
|
|
|
|
}
|
2022-10-11 00:04:09 +00:00
|
|
|
}
|
|
|
|
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2023-01-23 00:55:30 +00:00
|
|
|
class InvertSelectedItemsCommand implements ChartEditorCommand
|
2022-10-26 05:14:29 +00:00
|
|
|
{
|
2023-01-23 03:25:45 +00:00
|
|
|
var previousNoteSelection:Array<SongNoteData>;
|
|
|
|
var previousEventSelection:Array<SongEventData>;
|
|
|
|
|
|
|
|
public function new(?previousNoteSelection:Array<SongNoteData>, ?previousEventSelection:Array<SongEventData>)
|
|
|
|
{
|
|
|
|
this.previousNoteSelection = previousNoteSelection == null ? [] : previousNoteSelection;
|
|
|
|
this.previousEventSelection = previousEventSelection == null ? [] : previousEventSelection;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function execute(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentNoteSelection = SongDataUtils.subtractNotes(state.currentSongChartNoteData, previousNoteSelection);
|
|
|
|
state.currentEventSelection = SongDataUtils.subtractEvents(state.currentSongChartEventData, previousEventSelection);
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function undo(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentNoteSelection = previousNoteSelection;
|
|
|
|
state.currentEventSelection = previousEventSelection;
|
|
|
|
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
return 'Invert Selected Items';
|
|
|
|
}
|
2022-10-26 05:14:29 +00:00
|
|
|
}
|
|
|
|
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2023-01-23 00:55:30 +00:00
|
|
|
class DeselectAllItemsCommand implements ChartEditorCommand
|
2022-10-11 00:04:09 +00:00
|
|
|
{
|
2023-01-23 03:25:45 +00:00
|
|
|
var previousNoteSelection:Array<SongNoteData>;
|
|
|
|
var previousEventSelection:Array<SongEventData>;
|
|
|
|
|
|
|
|
public function new(?previousNoteSelection:Array<SongNoteData>, ?previousEventSelection:Array<SongEventData>)
|
|
|
|
{
|
|
|
|
this.previousNoteSelection = previousNoteSelection == null ? [] : previousNoteSelection;
|
|
|
|
this.previousEventSelection = previousEventSelection == null ? [] : previousEventSelection;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function execute(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentNoteSelection = [];
|
|
|
|
state.currentEventSelection = [];
|
|
|
|
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function undo(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentNoteSelection = previousNoteSelection;
|
|
|
|
state.currentEventSelection = previousEventSelection;
|
|
|
|
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
return 'Deselect All Items';
|
|
|
|
}
|
2022-10-11 00:04:09 +00:00
|
|
|
}
|
|
|
|
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2023-01-23 00:55:30 +00:00
|
|
|
class CutItemsCommand implements ChartEditorCommand
|
2022-10-11 00:04:09 +00:00
|
|
|
{
|
2023-01-23 03:25:45 +00:00
|
|
|
var notes:Array<SongNoteData>;
|
|
|
|
var events:Array<SongEventData>;
|
|
|
|
|
|
|
|
public function new(notes:Array<SongNoteData>, events:Array<SongEventData>)
|
|
|
|
{
|
|
|
|
this.notes = notes;
|
|
|
|
this.events = events;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function execute(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
// Copy the notes.
|
|
|
|
SongDataUtils.writeItemsToClipboard(
|
|
|
|
{
|
|
|
|
notes: SongDataUtils.buildNoteClipboard(notes),
|
|
|
|
events: SongDataUtils.buildEventClipboard(events)
|
|
|
|
});
|
|
|
|
|
|
|
|
// Delete the notes.
|
|
|
|
state.currentSongChartNoteData = SongDataUtils.subtractNotes(state.currentSongChartNoteData, notes);
|
|
|
|
state.currentSongChartEventData = SongDataUtils.subtractEvents(state.currentSongChartEventData, events);
|
|
|
|
state.currentNoteSelection = [];
|
|
|
|
state.currentEventSelection = [];
|
|
|
|
|
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
state.sortChartData();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function undo(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentSongChartNoteData = state.currentSongChartNoteData.concat(notes);
|
|
|
|
state.currentSongChartEventData = state.currentSongChartEventData.concat(events);
|
|
|
|
|
|
|
|
state.currentNoteSelection = notes;
|
|
|
|
state.currentEventSelection = events;
|
|
|
|
|
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
state.sortChartData();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
var len:Int = notes.length + events.length;
|
|
|
|
|
|
|
|
if (notes.length == 0) return 'Cut $len Events to Clipboard';
|
|
|
|
else if (events.length == 0) return 'Cut $len Notes to Clipboard';
|
|
|
|
else
|
|
|
|
return 'Cut $len Items to Clipboard';
|
|
|
|
}
|
2022-10-11 00:04:09 +00:00
|
|
|
}
|
|
|
|
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2022-10-26 05:14:29 +00:00
|
|
|
class FlipNotesCommand implements ChartEditorCommand
|
|
|
|
{
|
2023-08-31 22:47:23 +00:00
|
|
|
var notes:Array<SongNoteData> = [];
|
|
|
|
var flippedNotes:Array<SongNoteData> = [];
|
2023-01-23 03:25:45 +00:00
|
|
|
|
|
|
|
public function new(notes:Array<SongNoteData>)
|
|
|
|
{
|
|
|
|
this.notes = notes;
|
2023-08-31 22:47:23 +00:00
|
|
|
this.flippedNotes = SongDataUtils.flipNotes(notes);
|
2023-01-23 03:25:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public function execute(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
// Delete the notes.
|
|
|
|
state.currentSongChartNoteData = SongDataUtils.subtractNotes(state.currentSongChartNoteData, notes);
|
|
|
|
|
|
|
|
// Add the flipped notes.
|
|
|
|
state.currentSongChartNoteData = state.currentSongChartNoteData.concat(flippedNotes);
|
|
|
|
|
|
|
|
state.currentNoteSelection = flippedNotes;
|
|
|
|
state.currentEventSelection = [];
|
|
|
|
|
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
state.sortChartData();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function undo(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
state.currentSongChartNoteData = SongDataUtils.subtractNotes(state.currentSongChartNoteData, flippedNotes);
|
|
|
|
state.currentSongChartNoteData = state.currentSongChartNoteData.concat(notes);
|
|
|
|
|
|
|
|
state.currentNoteSelection = notes;
|
|
|
|
state.currentEventSelection = [];
|
|
|
|
|
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
|
|
|
|
state.sortChartData();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
var len:Int = notes.length;
|
|
|
|
return 'Flip $len Notes';
|
|
|
|
}
|
2022-10-26 05:14:29 +00:00
|
|
|
}
|
|
|
|
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2023-01-23 00:55:30 +00:00
|
|
|
class PasteItemsCommand implements ChartEditorCommand
|
2022-10-11 00:04:09 +00:00
|
|
|
{
|
2023-01-23 03:25:45 +00:00
|
|
|
var targetTimestamp:Float;
|
|
|
|
// Notes we added with this command, for undo.
|
2023-08-31 22:47:23 +00:00
|
|
|
var addedNotes:Array<SongNoteData> = [];
|
|
|
|
var addedEvents:Array<SongEventData> = [];
|
2023-01-23 03:25:45 +00:00
|
|
|
|
|
|
|
public function new(targetTimestamp:Float)
|
|
|
|
{
|
|
|
|
this.targetTimestamp = targetTimestamp;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function execute(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
var currentClipboard:SongClipboardItems = SongDataUtils.readItemsFromClipboard();
|
|
|
|
|
2023-10-17 04:38:28 +00:00
|
|
|
if (currentClipboard.valid != true)
|
|
|
|
{
|
|
|
|
#if !mac
|
|
|
|
NotificationManager.instance.addNotification(
|
|
|
|
{
|
|
|
|
title: 'Failed to Paste',
|
|
|
|
body: 'Could not parse clipboard contents.',
|
|
|
|
type: NotificationType.Error,
|
|
|
|
expiryMs: ChartEditorState.NOTIFICATION_DISMISS_TIME
|
|
|
|
});
|
|
|
|
#end
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
trace(currentClipboard.notes);
|
|
|
|
|
2023-01-23 03:25:45 +00:00
|
|
|
addedNotes = SongDataUtils.offsetSongNoteData(currentClipboard.notes, Std.int(targetTimestamp));
|
|
|
|
addedEvents = SongDataUtils.offsetSongEventData(currentClipboard.events, Std.int(targetTimestamp));
|
|
|
|
|
|
|
|
state.currentSongChartNoteData = state.currentSongChartNoteData.concat(addedNotes);
|
|
|
|
state.currentSongChartEventData = state.currentSongChartEventData.concat(addedEvents);
|
|
|
|
state.currentNoteSelection = addedNotes.copy();
|
|
|
|
state.currentEventSelection = addedEvents.copy();
|
|
|
|
|
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
|
|
|
|
state.sortChartData();
|
2023-10-17 04:38:28 +00:00
|
|
|
|
|
|
|
#if !mac
|
|
|
|
NotificationManager.instance.addNotification(
|
|
|
|
{
|
|
|
|
title: 'Paste Successful',
|
|
|
|
body: 'Successfully pasted clipboard contents.',
|
|
|
|
type: NotificationType.Success,
|
|
|
|
expiryMs: ChartEditorState.NOTIFICATION_DISMISS_TIME
|
|
|
|
});
|
|
|
|
#end
|
2023-01-23 03:25:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public function undo(state:ChartEditorState):Void
|
|
|
|
{
|
2023-10-11 23:39:52 +00:00
|
|
|
ChartEditorAudioHandler.playSound(Paths.sound('chartingSounds/undo'));
|
|
|
|
|
2023-01-23 03:25:45 +00:00
|
|
|
state.currentSongChartNoteData = SongDataUtils.subtractNotes(state.currentSongChartNoteData, addedNotes);
|
|
|
|
state.currentSongChartEventData = SongDataUtils.subtractEvents(state.currentSongChartEventData, addedEvents);
|
|
|
|
state.currentNoteSelection = [];
|
|
|
|
state.currentEventSelection = [];
|
|
|
|
|
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
|
|
|
|
state.sortChartData();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
var currentClipboard:SongClipboardItems = SongDataUtils.readItemsFromClipboard();
|
|
|
|
|
|
|
|
var len:Int = currentClipboard.notes.length + currentClipboard.events.length;
|
|
|
|
|
|
|
|
if (currentClipboard.notes.length == 0) return 'Paste $len Events';
|
|
|
|
else if (currentClipboard.events.length == 0) return 'Paste $len Notes';
|
|
|
|
else
|
|
|
|
return 'Paste $len Items';
|
|
|
|
}
|
2022-10-11 07:14:57 +00:00
|
|
|
}
|
2022-10-26 05:14:29 +00:00
|
|
|
|
2023-08-30 06:24:35 +00:00
|
|
|
@:nullSafety
|
2022-10-26 05:14:29 +00:00
|
|
|
class ExtendNoteLengthCommand implements ChartEditorCommand
|
|
|
|
{
|
2023-01-23 03:25:45 +00:00
|
|
|
var note:SongNoteData;
|
|
|
|
var oldLength:Float;
|
|
|
|
var newLength:Float;
|
|
|
|
|
|
|
|
public function new(note:SongNoteData, newLength:Float)
|
|
|
|
{
|
|
|
|
this.note = note;
|
|
|
|
this.oldLength = note.length;
|
|
|
|
this.newLength = newLength;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function execute(state:ChartEditorState):Void
|
|
|
|
{
|
|
|
|
note.length = newLength;
|
|
|
|
|
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
|
|
|
|
state.sortChartData();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function undo(state:ChartEditorState):Void
|
|
|
|
{
|
2023-10-11 23:39:52 +00:00
|
|
|
ChartEditorAudioHandler.playSound(Paths.sound('chartingSounds/undo'));
|
|
|
|
|
2023-01-23 03:25:45 +00:00
|
|
|
note.length = oldLength;
|
|
|
|
|
|
|
|
state.saveDataDirty = true;
|
|
|
|
state.noteDisplayDirty = true;
|
|
|
|
state.notePreviewDirty = true;
|
|
|
|
|
|
|
|
state.sortChartData();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function toString():String
|
|
|
|
{
|
|
|
|
return 'Extend Note Length';
|
|
|
|
}
|
2022-10-26 05:14:29 +00:00
|
|
|
}
|