mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-06 08:08:28 +08:00
Added running
and paused
properties to QuestRunner.
Also tried to improve the situation with the circular dependencies.
This commit is contained in:
parent
a219200291
commit
b8f0cbfcb3
@ -2,16 +2,20 @@ import { ExecutionResult, VirtualMachine, ExecutionLocation } from "./scripting/
|
|||||||
import { QuestModel } from "./model/QuestModel";
|
import { QuestModel } from "./model/QuestModel";
|
||||||
import { VirtualMachineIO } from "./scripting/vm/io";
|
import { VirtualMachineIO } from "./scripting/vm/io";
|
||||||
import { AsmToken, SegmentType, InstructionSegment, Segment, Instruction } from "./scripting/instructions";
|
import { AsmToken, SegmentType, InstructionSegment, Segment, Instruction } from "./scripting/instructions";
|
||||||
import { quest_editor_store } from "./stores/QuestEditorStore";
|
import { quest_editor_store, Logger } from "./stores/QuestEditorStore";
|
||||||
import { asm_editor_store } from "./stores/AsmEditorStore";
|
|
||||||
import { defined, assert } from "../core/util";
|
import { defined, assert } from "../core/util";
|
||||||
import {
|
import {
|
||||||
OP_CALL,
|
OP_CALL,
|
||||||
OP_VA_CALL,
|
OP_VA_CALL,
|
||||||
OP_SWITCH_CALL,
|
OP_SWITCH_CALL,
|
||||||
} from "./scripting/opcodes";
|
} from "./scripting/opcodes";
|
||||||
|
import { WritableProperty } from "../core/observable/property/WritableProperty";
|
||||||
|
import { property } from "../core/observable";
|
||||||
|
import { Property } from "../core/observable/property/Property";
|
||||||
|
import { AsmEditorStore } from "./stores/AsmEditorStore";
|
||||||
|
|
||||||
const logger = quest_editor_store.get_logger("quest_editor/QuestRunner");
|
let asm_editor_store: AsmEditorStore | undefined;
|
||||||
|
let logger: Logger | undefined;
|
||||||
|
|
||||||
function srcloc_to_string(srcloc: AsmToken): string {
|
function srcloc_to_string(srcloc: AsmToken): string {
|
||||||
return `[${srcloc.line_no}:${srcloc.col}]`;
|
return `[${srcloc.line_no}:${srcloc.col}]`;
|
||||||
@ -31,11 +35,34 @@ export class QuestRunner {
|
|||||||
private readonly stepping_breakpoints: number[] = [];
|
private readonly stepping_breakpoints: number[] = [];
|
||||||
private break_on_next = false;
|
private break_on_next = false;
|
||||||
|
|
||||||
|
private readonly _running: WritableProperty<boolean> = property(false);
|
||||||
|
private readonly _paused: WritableProperty<boolean> = property(false);
|
||||||
|
/**
|
||||||
|
* There is a quest loaded and it is currently running.
|
||||||
|
*/
|
||||||
|
public readonly running: Property<boolean> = this._running;
|
||||||
|
/**
|
||||||
|
* A quest is running but the execution is currently paused.
|
||||||
|
*/
|
||||||
|
public readonly paused: Property<boolean> = this._paused;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
this.vm = new VirtualMachine(this.create_vm_io());
|
this.vm = new VirtualMachine(this.create_vm_io());
|
||||||
}
|
}
|
||||||
|
|
||||||
run(quest: QuestModel): void {
|
run(quest: QuestModel): void {
|
||||||
|
if (logger === undefined) {
|
||||||
|
// defer creation of logger to prevent problems caused by circular dependency with QuestEditorStore
|
||||||
|
logger = quest_editor_store.get_logger("quest_editor/QuestRunner");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (asm_editor_store === undefined) {
|
||||||
|
// same here... circular dependency problem
|
||||||
|
import("./stores/AsmEditorStore").then(imports => {
|
||||||
|
asm_editor_store = imports.asm_editor_store;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
if (this.animation_frame != undefined) {
|
if (this.animation_frame != undefined) {
|
||||||
cancelAnimationFrame(this.animation_frame);
|
cancelAnimationFrame(this.animation_frame);
|
||||||
}
|
}
|
||||||
@ -45,6 +72,8 @@ export class QuestRunner {
|
|||||||
this.vm.load_object_code(quest.object_code);
|
this.vm.load_object_code(quest.object_code);
|
||||||
this.vm.start_thread(0);
|
this.vm.start_thread(0);
|
||||||
|
|
||||||
|
this._running.val = true;
|
||||||
|
|
||||||
this.schedule_frame();
|
this.schedule_frame();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -106,6 +135,10 @@ export class QuestRunner {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public step_out(): void {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
private schedule_frame(): void {
|
private schedule_frame(): void {
|
||||||
this.animation_frame = requestAnimationFrame(this.execution_loop);
|
this.animation_frame = requestAnimationFrame(this.execution_loop);
|
||||||
}
|
}
|
||||||
@ -113,57 +146,70 @@ export class QuestRunner {
|
|||||||
private execution_loop = (): void => {
|
private execution_loop = (): void => {
|
||||||
let result: ExecutionResult;
|
let result: ExecutionResult;
|
||||||
|
|
||||||
|
let need_emit_unpause = this.paused.val;
|
||||||
|
|
||||||
exec_loop: while (true) {
|
exec_loop: while (true) {
|
||||||
result = this.vm.execute();
|
result = this.vm.execute();
|
||||||
|
|
||||||
const srcloc = this.vm.get_current_source_location();
|
const srcloc = this.vm.get_current_source_location();
|
||||||
if (srcloc) {
|
if (srcloc) {
|
||||||
|
// check if need to break
|
||||||
const hit_breakpoint =
|
const hit_breakpoint =
|
||||||
this.break_on_next ||
|
this.break_on_next ||
|
||||||
asm_editor_store.breakpoints.val.includes(srcloc.line_no) ||
|
asm_editor_store?.breakpoints.val.includes(srcloc.line_no) ||
|
||||||
this.stepping_breakpoints.includes(srcloc.line_no);
|
this.stepping_breakpoints.includes(srcloc.line_no);
|
||||||
|
|
||||||
|
this.break_on_next = false;
|
||||||
|
|
||||||
if (hit_breakpoint) {
|
if (hit_breakpoint) {
|
||||||
this.stepping_breakpoints.length = 0;
|
this.stepping_breakpoints.length = 0;
|
||||||
asm_editor_store.set_execution_location(srcloc.line_no);
|
asm_editor_store?.set_execution_location(srcloc.line_no);
|
||||||
break exec_loop;
|
break exec_loop;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
this.break_on_next = false;
|
// first instruction after pause did not break, set state to unpaused
|
||||||
|
if (need_emit_unpause) {
|
||||||
|
need_emit_unpause = false;
|
||||||
|
this._paused.val = false;
|
||||||
|
}
|
||||||
|
|
||||||
switch (result) {
|
switch (result) {
|
||||||
case ExecutionResult.WaitingVsync:
|
case ExecutionResult.WaitingVsync:
|
||||||
this.vm.vsync();
|
this.vm.vsync();
|
||||||
this.schedule_frame();
|
this.schedule_frame();
|
||||||
break;
|
break exec_loop;
|
||||||
case ExecutionResult.WaitingInput:
|
case ExecutionResult.WaitingInput:
|
||||||
// TODO: implement input from gui
|
// TODO: implement input from gui
|
||||||
this.schedule_frame();
|
this.schedule_frame();
|
||||||
break;
|
break exec_loop;
|
||||||
case ExecutionResult.WaitingSelection:
|
case ExecutionResult.WaitingSelection:
|
||||||
// TODO: implement input from gui
|
// TODO: implement input from gui
|
||||||
this.vm.list_select(0);
|
this.vm.list_select(0);
|
||||||
this.schedule_frame();
|
this.schedule_frame();
|
||||||
break;
|
break exec_loop;
|
||||||
case ExecutionResult.Halted:
|
case ExecutionResult.Halted:
|
||||||
asm_editor_store.unset_execution_location();
|
this._running.val = false;
|
||||||
|
asm_editor_store?.unset_execution_location();
|
||||||
break exec_loop;
|
break exec_loop;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this._paused.val = true;
|
||||||
};
|
};
|
||||||
|
|
||||||
private create_vm_io = (): VirtualMachineIO => {
|
private create_vm_io = (): VirtualMachineIO => {
|
||||||
return {
|
return {
|
||||||
window_msg: (msg: string): void => {
|
window_msg: (msg: string): void => {
|
||||||
logger.info(`window_msg "${msg}"`);
|
logger?.info(`window_msg "${msg}"`);
|
||||||
},
|
},
|
||||||
|
|
||||||
message: (msg: string): void => {
|
message: (msg: string): void => {
|
||||||
logger.info(`message "${msg}"`);
|
logger?.info(`message "${msg}"`);
|
||||||
},
|
},
|
||||||
|
|
||||||
add_msg: (msg: string): void => {
|
add_msg: (msg: string): void => {
|
||||||
logger.info(`add_msg "${msg}"`);
|
logger?.info(`add_msg "${msg}"`);
|
||||||
},
|
},
|
||||||
|
|
||||||
winend: (): void => {},
|
winend: (): void => {},
|
||||||
@ -171,15 +217,15 @@ export class QuestRunner {
|
|||||||
mesend: (): void => {},
|
mesend: (): void => {},
|
||||||
|
|
||||||
list: (list_items: string[]): void => {
|
list: (list_items: string[]): void => {
|
||||||
logger.info(`list "[${list_items}]"`);
|
logger?.info(`list "[${list_items}]"`);
|
||||||
},
|
},
|
||||||
|
|
||||||
warning: (msg: string, srcloc?: AsmToken): void => {
|
warning: (msg: string, srcloc?: AsmToken): void => {
|
||||||
logger.warning(msg, srcloc && srcloc_to_string(srcloc));
|
logger?.warning(msg, srcloc && srcloc_to_string(srcloc));
|
||||||
},
|
},
|
||||||
|
|
||||||
error: (err: Error, srcloc?: AsmToken): void => {
|
error: (err: Error, srcloc?: AsmToken): void => {
|
||||||
logger.error(err, srcloc && srcloc_to_string(srcloc));
|
logger?.error(err, srcloc && srcloc_to_string(srcloc));
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
@ -31,12 +31,20 @@ import Logger = require("js-logger");
|
|||||||
import { MessageLogStore, LogMessage, LogLevel, LogGroup } from "../../core/gui/MessageLog";
|
import { MessageLogStore, LogMessage, LogLevel, LogGroup } from "../../core/gui/MessageLog";
|
||||||
import { ListProperty } from "../../core/observable/property/list/ListProperty";
|
import { ListProperty } from "../../core/observable/property/list/ListProperty";
|
||||||
import { WritableProperty } from "../../core/observable/property/WritableProperty";
|
import { WritableProperty } from "../../core/observable/property/WritableProperty";
|
||||||
|
import { QuestRunner } from "../QuestRunner";
|
||||||
|
|
||||||
const logger = Logger.get("quest_editor/gui/QuestEditorStore");
|
const logger = Logger.get("quest_editor/gui/QuestEditorStore");
|
||||||
|
|
||||||
|
export interface Logger {
|
||||||
|
debug(...items: any[]): void;
|
||||||
|
info(...items: any[]): void;
|
||||||
|
warning(...items: any[]): void;
|
||||||
|
error(...items: any[]): void;
|
||||||
|
}
|
||||||
|
|
||||||
export class QuestEditorStore implements Disposable, MessageLogStore {
|
export class QuestEditorStore implements Disposable, MessageLogStore {
|
||||||
private readonly disposer = new Disposer();
|
private readonly disposer = new Disposer();
|
||||||
|
public readonly quest_runner: QuestRunner = new QuestRunner();
|
||||||
/**
|
/**
|
||||||
* Log levels as Record<name, LogLevel>
|
* Log levels as Record<name, LogLevel>
|
||||||
*/
|
*/
|
||||||
@ -309,15 +317,11 @@ export class QuestEditorStore implements Disposable, MessageLogStore {
|
|||||||
};
|
};
|
||||||
|
|
||||||
run_current_quest = (): void => {
|
run_current_quest = (): void => {
|
||||||
// workaround for circular dependency
|
const quest = this.current_quest.val;
|
||||||
import("../QuestRunner").then(({ QuestRunner }) => {
|
|
||||||
const quest = this.current_quest.val;
|
|
||||||
const quest_runner = new QuestRunner();
|
|
||||||
|
|
||||||
if (quest) {
|
if (quest) {
|
||||||
quest_runner.run(quest);
|
this.quest_runner.run(quest);
|
||||||
}
|
}
|
||||||
});
|
|
||||||
};
|
};
|
||||||
|
|
||||||
private log_message_predicate = (msg: LogMessage): boolean => {
|
private log_message_predicate = (msg: LogMessage): boolean => {
|
||||||
@ -372,7 +376,7 @@ export class QuestEditorStore implements Disposable, MessageLogStore {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public get_logger(group_name?: string) {
|
public get_logger(group_name?: string): Logger {
|
||||||
let group = this.default_log_group;
|
let group = this.default_log_group;
|
||||||
|
|
||||||
// find existing or create new
|
// find existing or create new
|
||||||
|
Loading…
Reference in New Issue
Block a user