From 05bfe7122f8f4b734fde388ef08f7cb59f15b9ac Mon Sep 17 00:00:00 2001 From: Danielle Church Date: Wed, 28 Feb 2024 19:45:20 -0500 Subject: [PATCH] Start work on parsing the RelaxNG schema to drive the editor --- data/schema/actionList.rng | 17 +- editor.html | 1 + editor.js | 1 - helpers.js | 53 ++++- schema.js | 436 +++++++++++++++++++++++++++++++++++++ town.js | 1 - types.d.ts | 13 ++ 7 files changed, 511 insertions(+), 11 deletions(-) create mode 100644 schema.js diff --git a/data/schema/actionList.rng b/data/schema/actionList.rng index 68e1db65..93f7e6bb 100644 --- a/data/schema/actionList.rng +++ b/data/schema/actionList.rng @@ -1,10 +1,13 @@ + Definitions @@ -54,14 +57,12 @@ - - - - - - - - + + + + + + diff --git a/editor.html b/editor.html index 275ab84e..a7615b86 100644 --- a/editor.html +++ b/editor.html @@ -178,6 +178,7 @@

Actions

+ diff --git a/editor.js b/editor.js index f2418462..fc9a049f 100644 --- a/editor.js +++ b/editor.js @@ -768,7 +768,6 @@ class ActionListEditor { if (!this.windowBound) { this.windowBound = true; - addEventListener("input", setValueAttribute, {capture: true, passive: true}); this.bindDataList("skills", skills, s => s.label); this.bindDataList("buffs", buffs, b => !(b.name in prestigeBases) && b.label); diff --git a/helpers.js b/helpers.js index b3c12d47..c9165b64 100644 --- a/helpers.js +++ b/helpers.js @@ -731,4 +731,55 @@ function defineLazyGetter(object, name, getter) { const typedKeys = /** @type {(object: Partial>) => K[]} */(Object.keys); /** Strongly-typed version of Object.keys */ -const typedEntries = /** @type {(object: Partial>) => [K, V][]} */(Object.entries); \ No newline at end of file +const typedEntries = /** @type {(object: Partial>) => [K, V][]} */(Object.entries); + +const devtoolsHeader = Symbol.for("devtoolsHeader"); +const devtoolsHasBody = Symbol.for("devtoolsHasBody"); +const devtoolsBody = Symbol.for("devtoolsBody"); + +/** + * Convenience class for defining devtools formatting + * @template {*} DTConfig + */ +class DevtoolsFormattable { + /** @param {DTConfig} config @returns {DTJHTML | null} */ + dtHeader(config) { return null; } + /** @param {DTConfig} config */ + dtHasBody(config) { return false; } + /** @param {DTConfig} config @returns {DTJHTML | null} */ + dtBody(config) { return null; } + + [devtoolsHeader](config) { + return this.dtHeader(config); + } + [devtoolsHasBody](config) { + return this.dtHasBody(config); + } + [devtoolsBody](config) { + return this.dtBody(config); + } + + constructor() { + new.target.addFormatter(); + } + + /** @type {DTFormatter} */ + static formatter = { + header(object, config) { + return object?.[devtoolsHeader]?.(config) ?? null; + }, + hasBody(object, config) { + return object?.[devtoolsHasBody]?.(config) ?? false; + }, + body(object, config) { + return object?.[devtoolsBody]?.(config) ?? null; + } + } + + static addFormatter() { + self.devtoolsFormatters ??= []; + if (!self.devtoolsFormatters.includes(this.formatter)) { + self.devtoolsFormatters.push(this.formatter); + } + } +} \ No newline at end of file diff --git a/schema.js b/schema.js new file mode 100644 index 00000000..bdc29af1 --- /dev/null +++ b/schema.js @@ -0,0 +1,436 @@ +class Schema { + // constants + /** @readonly */ static XMLNS_RELAXNG = "http://relaxng.org/ns/structure/1.0"; + /** @readonly */ static XMLNS_RELAXNG_A = "http://relaxng.org/ns/compatibility/annotations/1.0"; + /** @readonly */ static XMLNS_IL = "http://dmchurch.github.io/omsi-loops/schema/1.0"; + + // well-known instances, should never be reassigned at runtime + /** @readonly */ static actionList = new Schema("data/schema/actionList.rng"); + + /** @param {string} url */ + static async fetchXML(url) { + const res = await fetch(url); + const xmlText = await res.text(); + const parser = new DOMParser(); + const doc = parser.parseFromString(xmlText, "text/xml"); + console.log("Schema document", doc, doc.documentElement); + return doc; + }; + + /** @type {XMLDocument} */ + schemaDocument; + /** @type {Promise} */ + schemaPromise; + /** @type {string} */ + schemaUrl; + + /** @type {SchemaGrammarNode} */ + grammar; + + /** @param {string} url */ + constructor(url) { + this.schemaUrl = url; + } + + async fetchSchema() { + if (this.schemaDocument) return this; + + this.schemaPromise ??= Schema.fetchXML(this.schemaUrl).then(d => this.schemaDocument = d); + await this.schemaPromise; + + const grammar = this.schemaDocument.documentElement; + if (grammar.getAttribute("xmlns") !== Schema.XMLNS_RELAXNG + || grammar.getAttribute("xmlns:a") !== Schema.XMLNS_RELAXNG_A + || grammar.getAttribute("xmlns:il") !== Schema.XMLNS_IL + || grammar.nodeName !== "grammar") { + throw new Error("Bad schema document"); + } + + this.grammar = new SchemaGrammarNode(grammar, null); + + return this; + } +} + +class SchemaNode extends DevtoolsFormattable { + // this can't be a static field assignment because the child classes aren't defined yet + static get tagMapping() { + /** @type {Record SchemaNode>} */ + const mapping = { + element: SchemaElementNode, + attribute: SchemaAttributeNode, + zeroOrMore: SchemaRepetitionNode, + oneOrMore: SchemaRepetitionNode, + optional: SchemaRepetitionNode, + text: SchemaDataValueNode, + data: SchemaDataValueNode, + value: SchemaDataValueNode, + name: SchemaDataValueNode, + choice: SchemaAlternativeNode, + interleave: SchemaAlternativeNode, + group: SchemaGroupNode, + empty: SchemaEmptyNode, + define: SchemaDefineNode, + ref: SchemaRefNode, + }; + SchemaNode.prototype.memoize.call(SchemaNode, "tagMapping", mapping); + return mapping; + } + + /** @template {SchemaNode} T @param {new(...args:any) => T} nodeType @returns {(node: SchemaNode) => node is T} */ + static is(nodeType) { + /** @param {SchemaNode} node @returns {node is T} */ + return function nodeIsOfType(node) { return node instanceof nodeType; } + } + + get schemaChildren() { + /** @type {SchemaNode[]} */ + const children = []; + /** @type {Element[]} */ + const elements = []; + + for (const child of this.xmlElement.children) { + if (child.namespaceURI === Schema.XMLNS_RELAXNG) { + children.push(new SchemaNode(child, this)); + } else { + elements.push(child); + } + } + + this.memoize("nonSchemaChildren", elements); + return this.memoize("schemaChildren", children); + } + + /** @type {Element[]} */ + get nonSchemaChildren() { + return (this.schemaChildren, this.nonSchemaChildren); + } + + get treeChildren() { + return this.schemaChildren; + } + + get tagName() { + return this.xmlElement?.tagName; + } + + /** @type {SchemaNode} */ + parent; + /** @type {Element} */ + xmlElement; + /** @param {Element} xmlElement @param {SchemaNode} parent */ + constructor(xmlElement, parent) { + super(); + if (Object.hasOwn(new.target, "tagMapping") && xmlElement.tagName in new.target.tagMapping) { + return new new.target.tagMapping[xmlElement.tagName](xmlElement, parent); + } + this.parent = parent; + this.xmlElement = xmlElement; + } + + get [Symbol.toStringTag]() { + return this.constructor.name; + } + + get stringRepArgs() { + const args = [this.tagName]; + const text = this.xmlElement.childElementCount === 0 ? this.xmlElement.textContent.trim() : ""; + if (text) args.push(JSON.stringify(text)); + return args; + } + + toString() { + return `${this.constructor.name}<${this.stringRepArgs.join(", ")}>`; + } + + /** @returns {DTJHTML | null} */ + dtHeader(cfg) { + return ["div", + {style: ""}, + ["span", + {style: "font-weight:bold;"}, + this.toString(), + ], + ...this.dtAttributes(cfg), + ["object", {object: this.xmlElement}], + ]; + } + + /** @returns {DTJML[]} */ + dtAttributes(cfg) { + return []; + } + + dtHasBody(cfg) { + return true || this.treeChildren.length > 0 && false; + } + + /** @returns {DTJHTML | null} */ + dtBody(cfg) { + return ["ol", + ...this.treeChildren.map(/** @returns {DTJHTML} */n => + ["li", + ["object", {object: n}] + ]), + ]; + } + + /** + * @template {keyof this} K + * @template {*} V + * @param {K} property + * @param {V} value + */ + memoize(property, value, writable = false) { + if (Object.hasOwn(this, property)) { + delete this[property]; + } + Object.defineProperty(this, property, {value, writable, configurable: true, enumerable: true}); + return value; + } + + /** + * @template {string} V + * @param {keyof this} property + */ + memoizeAttribute(property, {writable = false, attribute = String(property), allowedValues = /** @type {V[]} */(undefined)} = {}) { + const value = /** @type {V} */(this.xmlElement.getAttribute(attribute)); + if (allowedValues && !allowedValues.includes(value)) { + throw new Error(`Bad value for ${attribute}: "${value}" (expected one of: "${allowedValues?.join('", "')}")`); + } + return this.memoize(property, value, writable); + } + + /** @param {string} name @returns {SchemaDefineNode} */ + getDefinition(name) { + let node = this.parent; + while (node) { + if (node instanceof SchemaGrammarNode) { + return node.getDefinition(name); + } + node = node.parent; + } + } + + /** @returns {Generator} */ + *walkTree() { + yield this; + for (const child of this.treeChildren) { + yield *child.walkTree(); + } + } + + /** @param {(node: SchemaNode, parent: SchemaNode) => void} callback @param {SchemaNode} [parent] */ + visitTree(callback, parent) { + callback(this, parent); + for (const child of this.treeChildren) { + child.visitTree(callback, this); + } + return this; + } + + debugTree(parent, level=0, childIndex=0, childLength=1, prefix=" ") { + if (level > 100) throw new Error("too much recursion"); + const leader = level === 0 ? "" + : childIndex === childLength - 1 ? "\\-- " + : "+-- "; + const nextPrefix = `${prefix}${leader.replace(/[-\\]/g, " ").replace("+", "|")}`; + if (this.treeChildren.length > 1) { + console.group(`${prefix}${leader}${this}`, this.xmlElement, this); + for (const [index, child] of this.treeChildren.entries()) { + child.debugTree(this, level + 1, index, this.treeChildren.length, nextPrefix.slice(2)); + } + console.groupEnd(); + } else { + console.log(`${prefix}${leader}${this}`, this.xmlElement, this); + for (const [index, child] of this.treeChildren.entries()) { + child.debugTree(this, level + 1, index, this.treeChildren.length, nextPrefix); + } + } + } +} + +class SchemaGrammarNode extends SchemaNode { + /** @type {SchemaElementNode} */ + get startElement() { + const element = this.xmlElement.querySelector(":scope > start > element"); + if (!element) { + throw new Error("Could not find start element"); + } + return this.memoize("startElement", new SchemaElementNode(element, this)); + } + + get defines() { + /** @type {Record} */ + const defs = {__proto__: null}; + for (const defNode of this.schemaChildren.filter(SchemaNode.is(SchemaDefineNode))) { + const { name, alternatives } = defNode; + if (!name) { + console.warn("Skipping define without name", defNode); + continue; + } + if (defs[name]) { + if (!defs[name].combine && defNode.combine) { + defs[name].combine = defNode.combine; + } + defs[name].alternatives.push(...alternatives); + } else { + defs[name] = defNode; + } + } + return this.memoize("defines", defs); + } + + /** @param {string} name */ + getDefinition(name) { + return this.defines[name] ?? super.getDefinition(name); + } + + get treeChildren() { + return this.memoize("treeChildren", [this.startElement, ...Object.values(this.defines)]); + } +} + +class SchemaAlternativeNode extends SchemaNode { + /** @returns {"interleave" | "choice" | null} */ + get alternativeType() { + if (this.tagName === "interleave") return "interleave"; + if (this.tagName === "choice") return "choice"; + throw new Error(`Unexpected SchemaAlternativeNode tagName: ${this.tagName}`); + } + + get alternatives() { + /** @type {SchemaNode[][]} */ + const alternatives = []; + + for (const child of this.schemaChildren) { + if (child instanceof SchemaAlternativeNode && (child.tagName === this.tagName || child instanceof SchemaGroupNode)) { + alternatives.push(...child.alternatives); + } else { + alternatives.push([child]); + } + } + + return this.memoize("alternatives", alternatives); + } + + get treeChildren() { + const children = []; + for (const alt of this.alternatives) { + if (alt.length === 1) { + children.push(alt[0]); + } else { + const groupElement = this.xmlElement.ownerDocument.createElement("group"); + const group = new SchemaGroupNode(groupElement, this); + group.alternatives[0] = alt; + children.push(group); + } + } + return this.memoize("treeChildren", children); + } + + get stringRepArgs() { + const args = super.stringRepArgs; + if (this.alternatives.length > 1) { + args.push(JSON.stringify(this.alternativeType)); + } + return args; + } + +} + +class SchemaGroupNode extends SchemaAlternativeNode { + get alternativeType() { + return null; + } + + get alternatives() { + return this.memoize("alternatives", [this.schemaChildren]); + } + + get treeChildren() { + return this.alternatives[0]; + } +} + +class SchemaEmptyNode extends SchemaGroupNode { +} + +class SchemaDefineNode extends SchemaAlternativeNode { + get name() {return this.memoizeAttribute("name");} + + get combine() {return (this.memoizeAttribute("combine", {writable: true, allowedValues: ["choice", "interleave", null]}));} + set combine(v) {this.memoize("combine", v, true);} + get alternativeType() { return this.combine; } + get alternatives() { + if (this.schemaChildren.length === 1 && this.schemaChildren[0] instanceof SchemaAlternativeNode && this.schemaChildren[0].tagName === this.combine) { + return this.memoize("alternatives", this.schemaChildren[0].alternatives); + } else { + return this.memoize("alternatives", [this.schemaChildren]); + } + } + + get stringRepArgs() { + const args = super.stringRepArgs; + args.splice(1, 0, JSON.stringify(this.name)); + return args; + } +} + +class SchemaRefNode extends SchemaAlternativeNode { + get name() {return this.memoizeAttribute("name");} + + get definition() { + return this.memoize("definition", this.getDefinition(this.name)); + } + + get alternativeType() {return this.definition.alternativeType;} + get alternatives() {return this.definition.alternatives;} + + get treeChildren() { + return []; + } +} + +class SchemaXMLComponent extends SchemaNode { + get name() { + if (this.xmlElement.hasAttribute("name")) { + const nameElement = this.xmlElement.ownerDocument.createElement("name"); + nameElement.textContent = this.xmlElement.getAttribute("name"); + return this.memoize("name", new SchemaNode(nameElement, this)); + } else { + return this.memoize("name", this.schemaChildren.shift()); + } + } + + get schemaChildren() { + super.schemaChildren; + this.name; + return this.schemaChildren; + } + + get treeChildren() { + return [this.name, ...this.schemaChildren]; + } +} + +class SchemaElementNode extends SchemaXMLComponent { +} + +class SchemaAttributeNode extends SchemaXMLComponent { +} + +class SchemaDataValueNode extends SchemaNode { + get type() { + const {tagName} = this.xmlElement; + return this.memoize("type", tagName === "data" ? this.xmlElement.getAttribute("type") ?? "string" + : tagName === "value" ? this.xmlElement.getAttribute("type") ?? "token" + : tagName === "name" ? "token" + : tagName); + } +} + +class SchemaRepetitionNode extends SchemaNode { + get minOccur() { return this.memoize("minOccur", this.tagName === "oneOrMore" ? 1 : 0); } + get maxOccur() { return this.memoize("maxOccur", this.tagName === "optional" ? 1 : Infinity); } +} \ No newline at end of file diff --git a/town.js b/town.js index a4a490dc..7346bbb3 100644 --- a/town.js +++ b/town.js @@ -193,7 +193,6 @@ class Town { } if (!inLateGameActions && lateGameActionCount > 0 && isTravel(action.name)) { // shift late-game actions to end of action button list - console.log(`moving ${lateGameActionCount} actions ${this.totalActionList.slice(0, lateGameActionCount).map(a=>a.name).join(", ")} to end of list for town ${index}`); this.totalActionList.push(...this.totalActionList.splice(0, lateGameActionCount)); lateGameActionCount = 0; } diff --git a/types.d.ts b/types.d.ts index e60fa149..e3212ec8 100644 --- a/types.d.ts +++ b/types.d.ts @@ -47,5 +47,18 @@ declare interface AssassinAction { } +type DTJHTMLTag = "span" | "div" | "ol" | "ul" | "li" | "table" | "tr" | "td"; +type DTJHTML = [DTJHTMLTag, {style?: string}, ...DTJML[]] | [DTJHTMLTag, ...DTJML[]]; +type DTJML = DTJHTML | ["object", {object: O, config: C} | {object: any}] | string; +interface DTFormatter { + header(object: unknown, config?: C): DTJHTML | null; + hasBody?: (object: O, config?: C) => boolean; + body?: (object: O, config?: C) => DTJHTML | null; +} + +declare interface Window { + devtoolsFormatters: DTFormatter[]; +} + declare const LZString = await import("lz-string"); declare const Mousetrap = await import("mousetrap"); \ No newline at end of file