mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
Script editor undo is now integrated with general undo system.
This commit is contained in:
parent
c0e3ac924a
commit
5bd8feb766
@ -276,7 +276,7 @@ export class QuestEntityControls {
|
|||||||
const entity_type =
|
const entity_type =
|
||||||
entity instanceof QuestNpc ? entity.type.name : (entity as QuestObject).type.name;
|
entity instanceof QuestNpc ? entity.type.name : (entity as QuestObject).type.name;
|
||||||
|
|
||||||
quest_editor_store.undo_stack.push_action(
|
quest_editor_store.undo.push_action(
|
||||||
`Move ${entity_type}`,
|
`Move ${entity_type}`,
|
||||||
() => {
|
() => {
|
||||||
entity.position = initial_position;
|
entity.position = initial_position;
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { observable } from "mobx";
|
import { observable } from "mobx";
|
||||||
import { Server } from "../domain";
|
import { Server } from "../domain";
|
||||||
import { UndoStack } from "../undo";
|
import { undo_manager } from "../undo";
|
||||||
|
|
||||||
class ApplicationStore {
|
class ApplicationStore {
|
||||||
@observable current_server: Server = Server.Ephinea;
|
@observable current_server: Server = Server.Ephinea;
|
||||||
@ -23,10 +23,10 @@ class ApplicationStore {
|
|||||||
|
|
||||||
switch (binding) {
|
switch (binding) {
|
||||||
case "Ctrl-Z":
|
case "Ctrl-Z":
|
||||||
UndoStack.current && UndoStack.current.undo();
|
undo_manager.undo();
|
||||||
break;
|
break;
|
||||||
case "Ctrl-Shift-Z":
|
case "Ctrl-Shift-Z":
|
||||||
UndoStack.current && UndoStack.current.redo();
|
undo_manager.redo();
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
{
|
{
|
||||||
|
@ -6,7 +6,7 @@ import { parse_quest, write_quest_qst } from "../data_formats/parsing/quest";
|
|||||||
import { Vec3 } from "../data_formats/vector";
|
import { Vec3 } from "../data_formats/vector";
|
||||||
import { Area, Episode, Quest, QuestEntity, Section } from "../domain";
|
import { Area, Episode, Quest, QuestEntity, Section } from "../domain";
|
||||||
import { read_file } from "../read_file";
|
import { read_file } from "../read_file";
|
||||||
import { UndoStack } from "../undo";
|
import { UndoStack, SimpleUndo } from "../undo";
|
||||||
import { application_store } from "./ApplicationStore";
|
import { application_store } from "./ApplicationStore";
|
||||||
import { area_store } from "./AreaStore";
|
import { area_store } from "./AreaStore";
|
||||||
import { create_new_quest } from "./quest_creation";
|
import { create_new_quest } from "./quest_creation";
|
||||||
@ -16,8 +16,8 @@ const logger = Logger.get("stores/QuestEditorStore");
|
|||||||
class QuestEditorStore {
|
class QuestEditorStore {
|
||||||
@observable debug = false;
|
@observable debug = false;
|
||||||
|
|
||||||
readonly undo_stack = new UndoStack();
|
readonly undo = new UndoStack();
|
||||||
readonly script_undo_stack = new UndoStack();
|
readonly script_undo = new SimpleUndo("Text edits", () => {}, () => {});
|
||||||
|
|
||||||
@observable current_quest_filename?: string;
|
@observable current_quest_filename?: string;
|
||||||
@observable current_quest?: Quest;
|
@observable current_quest?: Quest;
|
||||||
@ -121,7 +121,8 @@ class QuestEditorStore {
|
|||||||
@action
|
@action
|
||||||
private set_quest = flow(function* set_quest(this: QuestEditorStore, quest?: Quest) {
|
private set_quest = flow(function* set_quest(this: QuestEditorStore, quest?: Quest) {
|
||||||
if (quest !== this.current_quest) {
|
if (quest !== this.current_quest) {
|
||||||
this.undo_stack.clear();
|
this.undo.reset();
|
||||||
|
this.script_undo.reset();
|
||||||
this.selected_entity = undefined;
|
this.selected_entity = undefined;
|
||||||
this.current_quest = quest;
|
this.current_quest = quest;
|
||||||
|
|
||||||
|
@ -73,7 +73,7 @@ export class QuestEditorComponent extends Component {
|
|||||||
private layout?: GoldenLayout;
|
private layout?: GoldenLayout;
|
||||||
|
|
||||||
componentDidMount(): void {
|
componentDidMount(): void {
|
||||||
quest_editor_store.undo_stack.make_current();
|
quest_editor_store.undo.make_current();
|
||||||
|
|
||||||
window.addEventListener("resize", this.resize);
|
window.addEventListener("resize", this.resize);
|
||||||
|
|
||||||
@ -116,7 +116,7 @@ export class QuestEditorComponent extends Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
componentWillUnmount(): void {
|
componentWillUnmount(): void {
|
||||||
quest_editor_store.undo_stack.ensure_not_current();
|
quest_editor_store.undo.ensure_not_current();
|
||||||
|
|
||||||
window.removeEventListener("resize", this.resize);
|
window.removeEventListener("resize", this.resize);
|
||||||
|
|
||||||
@ -147,10 +147,9 @@ export class QuestEditorComponent extends Component {
|
|||||||
scrip_editor_element.compareDocumentPosition(e.target) &
|
scrip_editor_element.compareDocumentPosition(e.target) &
|
||||||
Node.DOCUMENT_POSITION_CONTAINED_BY
|
Node.DOCUMENT_POSITION_CONTAINED_BY
|
||||||
) {
|
) {
|
||||||
// quest_editor_store.script_undo_stack.make_current();
|
quest_editor_store.script_undo.make_current();
|
||||||
quest_editor_store.undo_stack.ensure_not_current();
|
|
||||||
} else {
|
} else {
|
||||||
quest_editor_store.undo_stack.make_current();
|
quest_editor_store.undo.make_current();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -6,6 +6,7 @@ import { OPCODES } from "../../data_formats/parsing/quest/bin";
|
|||||||
import { Assembler } from "../../scripting/Assembler";
|
import { Assembler } from "../../scripting/Assembler";
|
||||||
import { quest_editor_store } from "../../stores/QuestEditorStore";
|
import { quest_editor_store } from "../../stores/QuestEditorStore";
|
||||||
import "./ScriptEditorComponent.less";
|
import "./ScriptEditorComponent.less";
|
||||||
|
import { Action } from "../../undo";
|
||||||
|
|
||||||
const ASM_SYNTAX: languages.IMonarchLanguage = {
|
const ASM_SYNTAX: languages.IMonarchLanguage = {
|
||||||
defaultToken: "invalid",
|
defaultToken: "invalid",
|
||||||
@ -182,7 +183,53 @@ class MonacoComponent extends Component<MonacoProps> {
|
|||||||
const assembly = this.assembler.disassemble(quest.instructions, quest.labels);
|
const assembly = this.assembler.disassemble(quest.instructions, quest.labels);
|
||||||
const model = editor.createModel(assembly.join("\n"), "psoasm");
|
const model = editor.createModel(assembly.join("\n"), "psoasm");
|
||||||
|
|
||||||
|
quest_editor_store.script_undo.action = new Action(
|
||||||
|
"Text edits",
|
||||||
|
() => {
|
||||||
|
if (this.editor) {
|
||||||
|
this.editor.trigger("undo stack", "undo", undefined);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
() => {
|
||||||
|
if (this.editor) {
|
||||||
|
this.editor.trigger("redo stack", "redo", undefined);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
let initial_version = model.getAlternativeVersionId();
|
||||||
|
let current_version = initial_version;
|
||||||
|
let last_version = initial_version;
|
||||||
|
|
||||||
const disposable = model.onDidChangeContent(e => {
|
const disposable = model.onDidChangeContent(e => {
|
||||||
|
const version = model.getAlternativeVersionId();
|
||||||
|
|
||||||
|
if (version < current_version) {
|
||||||
|
// Undoing.
|
||||||
|
quest_editor_store.script_undo.can_redo = true;
|
||||||
|
|
||||||
|
if (version === initial_version) {
|
||||||
|
quest_editor_store.script_undo.can_undo = false;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Redoing.
|
||||||
|
if (version <= last_version) {
|
||||||
|
if (version === last_version) {
|
||||||
|
quest_editor_store.script_undo.can_redo = false;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
quest_editor_store.script_undo.can_redo = false;
|
||||||
|
|
||||||
|
if (current_version > last_version) {
|
||||||
|
last_version = current_version;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
quest_editor_store.script_undo.can_undo = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
current_version = version;
|
||||||
|
|
||||||
if (!this.assembler) return;
|
if (!this.assembler) return;
|
||||||
this.assembler.update_assembly(e.changes);
|
this.assembler.update_assembly(e.changes);
|
||||||
});
|
});
|
||||||
|
@ -5,13 +5,12 @@ import { observer } from "mobx-react";
|
|||||||
import React, { ChangeEvent, Component, ReactNode } from "react";
|
import React, { ChangeEvent, Component, ReactNode } from "react";
|
||||||
import { Episode } from "../../domain";
|
import { Episode } from "../../domain";
|
||||||
import { quest_editor_store } from "../../stores/QuestEditorStore";
|
import { quest_editor_store } from "../../stores/QuestEditorStore";
|
||||||
|
import { undo_manager } from "../../undo";
|
||||||
import "./Toolbar.less";
|
import "./Toolbar.less";
|
||||||
import { UndoStack } from "../../undo";
|
|
||||||
|
|
||||||
@observer
|
@observer
|
||||||
export class Toolbar extends Component {
|
export class Toolbar extends Component {
|
||||||
render(): ReactNode {
|
render(): ReactNode {
|
||||||
const undo = UndoStack.current;
|
|
||||||
const quest = quest_editor_store.current_quest;
|
const quest = quest_editor_store.current_quest;
|
||||||
const areas = quest ? Array.from(quest.area_variants).map(a => a.area) : [];
|
const areas = quest ? Array.from(quest.area_variants).map(a => a.area) : [];
|
||||||
const area = quest_editor_store.current_area;
|
const area = quest_editor_store.current_area;
|
||||||
@ -49,10 +48,11 @@ export class Toolbar extends Component {
|
|||||||
icon="undo"
|
icon="undo"
|
||||||
onClick={this.undo}
|
onClick={this.undo}
|
||||||
title={
|
title={
|
||||||
"Undo" +
|
undo_manager.first_undo
|
||||||
(undo && undo.first_undo ? ` "${undo.first_undo.description}"` : "")
|
? `Undo "${undo_manager.first_undo.description}"`
|
||||||
|
: "Nothing to undo"
|
||||||
}
|
}
|
||||||
disabled={!(undo && undo.can_undo)}
|
disabled={!undo_manager.can_undo}
|
||||||
>
|
>
|
||||||
Undo
|
Undo
|
||||||
</Button>
|
</Button>
|
||||||
@ -60,10 +60,11 @@ export class Toolbar extends Component {
|
|||||||
icon="redo"
|
icon="redo"
|
||||||
onClick={this.redo}
|
onClick={this.redo}
|
||||||
title={
|
title={
|
||||||
"Redo" +
|
undo_manager.first_redo
|
||||||
(undo && undo.first_redo ? ` "${undo.first_redo.description}"` : "")
|
? `Redo "${undo_manager.first_redo.description}"`
|
||||||
|
: "Nothing to redo"
|
||||||
}
|
}
|
||||||
disabled={!(undo && undo.can_redo)}
|
disabled={!undo_manager.can_redo}
|
||||||
>
|
>
|
||||||
Redo
|
Redo
|
||||||
</Button>
|
</Button>
|
||||||
@ -95,11 +96,11 @@ export class Toolbar extends Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private undo(): void {
|
private undo(): void {
|
||||||
UndoStack.current && UndoStack.current.undo();
|
undo_manager.undo();
|
||||||
}
|
}
|
||||||
|
|
||||||
private redo(): void {
|
private redo(): void {
|
||||||
UndoStack.current && UndoStack.current.redo();
|
undo_manager.redo();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
160
src/undo.ts
160
src/undo.ts
@ -8,9 +8,147 @@ export class Action {
|
|||||||
) {}
|
) {}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class UndoStack {
|
class UndoManager {
|
||||||
@observable static current?: UndoStack;
|
@observable current?: Undo;
|
||||||
|
|
||||||
|
@computed
|
||||||
|
get can_undo(): boolean {
|
||||||
|
return this.current ? this.current.can_undo : false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed
|
||||||
|
get can_redo(): boolean {
|
||||||
|
return this.current ? this.current.can_redo : false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed
|
||||||
|
get first_undo(): Action | undefined {
|
||||||
|
return this.current && this.current.first_undo;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed
|
||||||
|
get first_redo(): Action | undefined {
|
||||||
|
return this.current && this.current.first_redo;
|
||||||
|
}
|
||||||
|
|
||||||
|
undo(): boolean {
|
||||||
|
return this.current ? this.current.undo() : false;
|
||||||
|
}
|
||||||
|
|
||||||
|
redo(): boolean {
|
||||||
|
return this.current ? this.current.redo() : false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export const undo_manager = new UndoManager();
|
||||||
|
|
||||||
|
interface Undo {
|
||||||
|
make_current(): void;
|
||||||
|
|
||||||
|
ensure_not_current(): void;
|
||||||
|
|
||||||
|
readonly can_undo: boolean;
|
||||||
|
|
||||||
|
readonly can_redo: boolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The first action that will be undone when calling undo().
|
||||||
|
*/
|
||||||
|
readonly first_undo: Action | undefined;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The first action that will be redone when calling redo().
|
||||||
|
*/
|
||||||
|
readonly first_redo: Action | undefined;
|
||||||
|
|
||||||
|
undo(): boolean;
|
||||||
|
|
||||||
|
redo(): boolean;
|
||||||
|
|
||||||
|
reset(): void;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Simply contains a single action. `can_undo` and `can_redo` must be managed manually.
|
||||||
|
*/
|
||||||
|
export class SimpleUndo implements Undo {
|
||||||
|
@observable.ref action: Action;
|
||||||
|
|
||||||
|
constructor(description: string, undo: () => void, redo: () => void) {
|
||||||
|
this.action = new Action(description, undo, redo);
|
||||||
|
}
|
||||||
|
|
||||||
|
@action
|
||||||
|
make_current(): void {
|
||||||
|
undo_manager.current = this;
|
||||||
|
}
|
||||||
|
|
||||||
|
@action
|
||||||
|
ensure_not_current(): void {
|
||||||
|
if (undo_manager.current === this) {
|
||||||
|
undo_manager.current = undefined;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@observable _can_undo = false;
|
||||||
|
|
||||||
|
get can_undo(): boolean {
|
||||||
|
return this._can_undo;
|
||||||
|
}
|
||||||
|
|
||||||
|
set can_undo(can_undo: boolean) {
|
||||||
|
this._can_undo = can_undo;
|
||||||
|
}
|
||||||
|
|
||||||
|
@observable _can_redo = false;
|
||||||
|
|
||||||
|
get can_redo(): boolean {
|
||||||
|
return this._can_redo;
|
||||||
|
}
|
||||||
|
|
||||||
|
set can_redo(can_redo: boolean) {
|
||||||
|
this._can_redo = can_redo;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed get first_undo(): Action | undefined {
|
||||||
|
return this.can_undo ? this.action : undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
@computed get first_redo(): Action | undefined {
|
||||||
|
return this.can_redo ? this.action : undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
@action
|
||||||
|
undo(): boolean {
|
||||||
|
if (this.can_undo) {
|
||||||
|
this.action.undo();
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@action
|
||||||
|
redo(): boolean {
|
||||||
|
if (this.can_redo) {
|
||||||
|
this.action.redo();
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@action
|
||||||
|
reset(): void {
|
||||||
|
this._can_undo = false;
|
||||||
|
this._can_redo = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Full-fledged linear undo/redo implementation.
|
||||||
|
*/
|
||||||
|
export class UndoStack implements Undo {
|
||||||
@observable private readonly stack: IObservableArray<Action> = observable.array([], {
|
@observable private readonly stack: IObservableArray<Action> = observable.array([], {
|
||||||
deep: false,
|
deep: false,
|
||||||
});
|
});
|
||||||
@ -19,13 +157,15 @@ export class UndoStack {
|
|||||||
*/
|
*/
|
||||||
@observable private index = 0;
|
@observable private index = 0;
|
||||||
|
|
||||||
|
@action
|
||||||
make_current(): void {
|
make_current(): void {
|
||||||
UndoStack.current = this;
|
undo_manager.current = this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@action
|
||||||
ensure_not_current(): void {
|
ensure_not_current(): void {
|
||||||
if (UndoStack.current === this) {
|
if (undo_manager.current === this) {
|
||||||
UndoStack.current = undefined;
|
undo_manager.current = undefined;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -62,6 +202,14 @@ export class UndoStack {
|
|||||||
this.index++;
|
this.index++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Pop an action off the stack without undoing.
|
||||||
|
*/
|
||||||
|
@action
|
||||||
|
pop(): Action | undefined {
|
||||||
|
return this.stack.splice(--this.index, 1)[0];
|
||||||
|
}
|
||||||
|
|
||||||
@action
|
@action
|
||||||
undo(): boolean {
|
undo(): boolean {
|
||||||
if (this.can_undo) {
|
if (this.can_undo) {
|
||||||
@ -83,7 +231,7 @@ export class UndoStack {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@action
|
@action
|
||||||
clear(): void {
|
reset(): void {
|
||||||
this.stack.clear();
|
this.stack.clear();
|
||||||
this.index = 0;
|
this.index = 0;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user