mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
Ported several ASM editor features.
This commit is contained in:
parent
dc0615e1d2
commit
0133e82d3f
8
core/src/jsMain/kotlin/world/phantasmal/core/Js.kt
Normal file
8
core/src/jsMain/kotlin/world/phantasmal/core/Js.kt
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
package world.phantasmal.core
|
||||||
|
|
||||||
|
@Suppress("NOTHING_TO_INLINE")
|
||||||
|
inline fun <T> jsArrayOf(vararg elements: T): JsArray<T> =
|
||||||
|
elements.unsafeCast<JsArray<T>>()
|
||||||
|
|
||||||
|
inline fun <T> JsArray<T>.asArray(): Array<T> =
|
||||||
|
unsafeCast<Array<T>>()
|
@ -0,0 +1,5 @@
|
|||||||
|
package world.phantasmal.core
|
||||||
|
|
||||||
|
external interface JsArray<T> {
|
||||||
|
fun push(vararg elements: T): Int
|
||||||
|
}
|
@ -71,8 +71,8 @@ kotlin {
|
|||||||
val generateOpcodes = tasks.register("generateOpcodes") {
|
val generateOpcodes = tasks.register("generateOpcodes") {
|
||||||
group = "code generation"
|
group = "code generation"
|
||||||
|
|
||||||
val packageName = "world.phantasmal.lib.assembly"
|
val packageName = "world.phantasmal.lib.asm"
|
||||||
val opcodesFile = file("assetsGeneration/assembly/opcodes.yml")
|
val opcodesFile = file("assetsGeneration/asm/opcodes.yml")
|
||||||
val outputFile = file(
|
val outputFile = file(
|
||||||
"build/generated-src/commonMain/kotlin/${packageName.replace('.', '/')}/Opcodes.kt"
|
"build/generated-src/commonMain/kotlin/${packageName.replace('.', '/')}/Opcodes.kt"
|
||||||
)
|
)
|
||||||
@ -104,7 +104,9 @@ fun opcodeToCode(writer: PrintWriter, opcode: Map<String, Any>) {
|
|||||||
val code = (opcode["code"] as String).drop(2).toInt(16)
|
val code = (opcode["code"] as String).drop(2).toInt(16)
|
||||||
val codeStr = code.toString(16).toUpperCase().padStart(2, '0')
|
val codeStr = code.toString(16).toUpperCase().padStart(2, '0')
|
||||||
val mnemonic = opcode["mnemonic"] as String? ?: "unknown_${codeStr.toLowerCase()}"
|
val mnemonic = opcode["mnemonic"] as String? ?: "unknown_${codeStr.toLowerCase()}"
|
||||||
val description = opcode["description"] as String?
|
val doc = (opcode["doc"] as String?)?.let {
|
||||||
|
"\"${it.replace("\n", "\\n")}\""
|
||||||
|
}
|
||||||
val stack = opcode["stack"] as String?
|
val stack = opcode["stack"] as String?
|
||||||
|
|
||||||
val valName = "OP_" + mnemonic
|
val valName = "OP_" + mnemonic
|
||||||
@ -136,7 +138,7 @@ fun opcodeToCode(writer: PrintWriter, opcode: Map<String, Any>) {
|
|||||||
|val $valName = Opcode(
|
|val $valName = Opcode(
|
||||||
| 0x$codeStr,
|
| 0x$codeStr,
|
||||||
| "$mnemonic",
|
| "$mnemonic",
|
||||||
| ${description?.let { "\"$it\"" }},
|
| $doc,
|
||||||
| $params,
|
| $params,
|
||||||
| $stackInteraction,
|
| $stackInteraction,
|
||||||
|).also { ${array}[0x$indexStr] = it }""".trimMargin()
|
|).also { ${array}[0x$indexStr] = it }""".trimMargin()
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
package world.phantasmal.lib.assembly
|
package world.phantasmal.lib.asm
|
||||||
|
|
||||||
import world.phantasmal.core.isDigit
|
import world.phantasmal.core.isDigit
|
||||||
|
|
@ -1,4 +1,4 @@
|
|||||||
package world.phantasmal.lib.assembly
|
package world.phantasmal.lib.asm
|
||||||
|
|
||||||
import mu.KotlinLogging
|
import mu.KotlinLogging
|
||||||
import world.phantasmal.core.Problem
|
import world.phantasmal.core.Problem
|
@ -1,4 +1,4 @@
|
|||||||
package world.phantasmal.lib.assembly
|
package world.phantasmal.lib.asm
|
||||||
|
|
||||||
import mu.KotlinLogging
|
import mu.KotlinLogging
|
||||||
import world.phantasmal.core.reinterpretAsFloat
|
import world.phantasmal.core.reinterpretAsFloat
|
@ -1,4 +1,4 @@
|
|||||||
package world.phantasmal.lib.assembly
|
package world.phantasmal.lib.asm
|
||||||
|
|
||||||
import world.phantasmal.lib.buffer.Buffer
|
import world.phantasmal.lib.buffer.Buffer
|
||||||
import kotlin.math.min
|
import kotlin.math.min
|
@ -1,4 +1,4 @@
|
|||||||
package world.phantasmal.lib.assembly
|
package world.phantasmal.lib.asm
|
||||||
|
|
||||||
private val MNEMONIC_TO_OPCODES: MutableMap<String, Opcode> by lazy {
|
private val MNEMONIC_TO_OPCODES: MutableMap<String, Opcode> by lazy {
|
||||||
val map = mutableMapOf<String, Opcode>()
|
val map = mutableMapOf<String, Opcode>()
|
@ -1,6 +1,6 @@
|
|||||||
package world.phantasmal.lib.assembly.dataFlowAnalysis
|
package world.phantasmal.lib.asm.dataFlowAnalysis
|
||||||
|
|
||||||
import world.phantasmal.lib.assembly.*
|
import world.phantasmal.lib.asm.*
|
||||||
|
|
||||||
// See https://en.wikipedia.org/wiki/Control-flow_graph.
|
// See https://en.wikipedia.org/wiki/Control-flow_graph.
|
||||||
|
|
@ -1,10 +1,10 @@
|
|||||||
package world.phantasmal.lib.assembly.dataFlowAnalysis
|
package world.phantasmal.lib.asm.dataFlowAnalysis
|
||||||
|
|
||||||
import mu.KotlinLogging
|
import mu.KotlinLogging
|
||||||
import world.phantasmal.lib.assembly.InstructionSegment
|
import world.phantasmal.lib.asm.InstructionSegment
|
||||||
import world.phantasmal.lib.assembly.OP_BB_MAP_DESIGNATE
|
import world.phantasmal.lib.asm.OP_BB_MAP_DESIGNATE
|
||||||
import world.phantasmal.lib.assembly.OP_MAP_DESIGNATE
|
import world.phantasmal.lib.asm.OP_MAP_DESIGNATE
|
||||||
import world.phantasmal.lib.assembly.OP_MAP_DESIGNATE_EX
|
import world.phantasmal.lib.asm.OP_MAP_DESIGNATE_EX
|
||||||
|
|
||||||
private val logger = KotlinLogging.logger {}
|
private val logger = KotlinLogging.logger {}
|
||||||
|
|
@ -1,7 +1,7 @@
|
|||||||
package world.phantasmal.lib.assembly.dataFlowAnalysis
|
package world.phantasmal.lib.asm.dataFlowAnalysis
|
||||||
|
|
||||||
import mu.KotlinLogging
|
import mu.KotlinLogging
|
||||||
import world.phantasmal.lib.assembly.*
|
import world.phantasmal.lib.asm.*
|
||||||
import kotlin.math.max
|
import kotlin.math.max
|
||||||
import kotlin.math.min
|
import kotlin.math.min
|
||||||
|
|
@ -1,7 +1,7 @@
|
|||||||
package world.phantasmal.lib.assembly.dataFlowAnalysis
|
package world.phantasmal.lib.asm.dataFlowAnalysis
|
||||||
|
|
||||||
import mu.KotlinLogging
|
import mu.KotlinLogging
|
||||||
import world.phantasmal.lib.assembly.*
|
import world.phantasmal.lib.asm.*
|
||||||
|
|
||||||
private val logger = KotlinLogging.logger {}
|
private val logger = KotlinLogging.logger {}
|
||||||
|
|
@ -1,4 +1,4 @@
|
|||||||
package world.phantasmal.lib.assembly.dataFlowAnalysis
|
package world.phantasmal.lib.asm.dataFlowAnalysis
|
||||||
|
|
||||||
import kotlin.math.max
|
import kotlin.math.max
|
||||||
import kotlin.math.min
|
import kotlin.math.min
|
@ -3,10 +3,10 @@ package world.phantasmal.lib.fileFormats.quest
|
|||||||
import mu.KotlinLogging
|
import mu.KotlinLogging
|
||||||
import world.phantasmal.core.PwResult
|
import world.phantasmal.core.PwResult
|
||||||
import world.phantasmal.core.Severity
|
import world.phantasmal.core.Severity
|
||||||
import world.phantasmal.lib.assembly.*
|
import world.phantasmal.lib.asm.*
|
||||||
import world.phantasmal.lib.assembly.dataFlowAnalysis.ControlFlowGraph
|
import world.phantasmal.lib.asm.dataFlowAnalysis.ControlFlowGraph
|
||||||
import world.phantasmal.lib.assembly.dataFlowAnalysis.getRegisterValue
|
import world.phantasmal.lib.asm.dataFlowAnalysis.getRegisterValue
|
||||||
import world.phantasmal.lib.assembly.dataFlowAnalysis.getStackValue
|
import world.phantasmal.lib.asm.dataFlowAnalysis.getStackValue
|
||||||
import world.phantasmal.lib.buffer.Buffer
|
import world.phantasmal.lib.buffer.Buffer
|
||||||
import world.phantasmal.lib.cursor.BufferCursor
|
import world.phantasmal.lib.cursor.BufferCursor
|
||||||
import world.phantasmal.lib.cursor.Cursor
|
import world.phantasmal.lib.cursor.Cursor
|
||||||
|
@ -5,10 +5,10 @@ import world.phantasmal.core.PwResult
|
|||||||
import world.phantasmal.core.PwResultBuilder
|
import world.phantasmal.core.PwResultBuilder
|
||||||
import world.phantasmal.core.Severity
|
import world.phantasmal.core.Severity
|
||||||
import world.phantasmal.core.Success
|
import world.phantasmal.core.Success
|
||||||
import world.phantasmal.lib.assembly.InstructionSegment
|
import world.phantasmal.lib.asm.InstructionSegment
|
||||||
import world.phantasmal.lib.assembly.OP_SET_EPISODE
|
import world.phantasmal.lib.asm.OP_SET_EPISODE
|
||||||
import world.phantasmal.lib.assembly.Segment
|
import world.phantasmal.lib.asm.Segment
|
||||||
import world.phantasmal.lib.assembly.dataFlowAnalysis.getMapDesignations
|
import world.phantasmal.lib.asm.dataFlowAnalysis.getMapDesignations
|
||||||
import world.phantasmal.lib.compression.prs.prsDecompress
|
import world.phantasmal.lib.compression.prs.prsDecompress
|
||||||
import world.phantasmal.lib.cursor.Cursor
|
import world.phantasmal.lib.cursor.Cursor
|
||||||
import world.phantasmal.lib.cursor.cursor
|
import world.phantasmal.lib.cursor.cursor
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
package world.phantasmal.lib.assembly
|
package world.phantasmal.lib.asm
|
||||||
|
|
||||||
import world.phantasmal.lib.test.LibTestSuite
|
import world.phantasmal.lib.test.LibTestSuite
|
||||||
import world.phantasmal.testUtils.assertCloseTo
|
import world.phantasmal.testUtils.assertCloseTo
|
||||||
import kotlin.test.Test
|
import kotlin.test.Test
|
||||||
import kotlin.test.assertEquals
|
import kotlin.test.assertEquals
|
||||||
|
|
||||||
class AssemblyTokenizationTests : LibTestSuite() {
|
class AsmTokenizationTests : LibTestSuite() {
|
||||||
@Test
|
@Test
|
||||||
fun valid_floats_are_parsed_as_Float32_tokens() {
|
fun valid_floats_are_parsed_as_Float32_tokens() {
|
||||||
assertCloseTo(808.9f, (tokenizeLine("808.9")[0] as Token.Float32).value)
|
assertCloseTo(808.9f, (tokenizeLine("808.9")[0] as Token.Float32).value)
|
@ -1,4 +1,4 @@
|
|||||||
package world.phantasmal.lib.assembly
|
package world.phantasmal.lib.asm
|
||||||
|
|
||||||
import world.phantasmal.core.Success
|
import world.phantasmal.core.Success
|
||||||
import world.phantasmal.lib.test.LibTestSuite
|
import world.phantasmal.lib.test.LibTestSuite
|
@ -1,4 +1,4 @@
|
|||||||
package world.phantasmal.lib.assembly.dataFlowAnalysis
|
package world.phantasmal.lib.asm.dataFlowAnalysis
|
||||||
|
|
||||||
import world.phantasmal.lib.test.LibTestSuite
|
import world.phantasmal.lib.test.LibTestSuite
|
||||||
import world.phantasmal.lib.test.toInstructions
|
import world.phantasmal.lib.test.toInstructions
|
@ -1,6 +1,6 @@
|
|||||||
package world.phantasmal.lib.assembly.dataFlowAnalysis
|
package world.phantasmal.lib.asm.dataFlowAnalysis
|
||||||
|
|
||||||
import world.phantasmal.lib.assembly.*
|
import world.phantasmal.lib.asm.*
|
||||||
import world.phantasmal.lib.test.LibTestSuite
|
import world.phantasmal.lib.test.LibTestSuite
|
||||||
import world.phantasmal.lib.test.toInstructions
|
import world.phantasmal.lib.test.toInstructions
|
||||||
import kotlin.test.Test
|
import kotlin.test.Test
|
@ -1,4 +1,4 @@
|
|||||||
package world.phantasmal.lib.assembly.dataFlowAnalysis
|
package world.phantasmal.lib.asm.dataFlowAnalysis
|
||||||
|
|
||||||
import world.phantasmal.lib.test.LibTestSuite
|
import world.phantasmal.lib.test.LibTestSuite
|
||||||
import kotlin.test.Test
|
import kotlin.test.Test
|
@ -1,9 +1,9 @@
|
|||||||
package world.phantasmal.lib.fileFormats.quest
|
package world.phantasmal.lib.fileFormats.quest
|
||||||
|
|
||||||
import world.phantasmal.core.Success
|
import world.phantasmal.core.Success
|
||||||
import world.phantasmal.lib.assembly.InstructionSegment
|
import world.phantasmal.lib.asm.InstructionSegment
|
||||||
import world.phantasmal.lib.assembly.OP_BB_MAP_DESIGNATE
|
import world.phantasmal.lib.asm.OP_BB_MAP_DESIGNATE
|
||||||
import world.phantasmal.lib.assembly.OP_SET_EPISODE
|
import world.phantasmal.lib.asm.OP_SET_EPISODE
|
||||||
import world.phantasmal.lib.buffer.Buffer
|
import world.phantasmal.lib.buffer.Buffer
|
||||||
import world.phantasmal.lib.test.LibTestSuite
|
import world.phantasmal.lib.test.LibTestSuite
|
||||||
import kotlin.test.Test
|
import kotlin.test.Test
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
package world.phantasmal.lib.fileFormats.quest
|
package world.phantasmal.lib.fileFormats.quest
|
||||||
|
|
||||||
import world.phantasmal.core.Success
|
import world.phantasmal.core.Success
|
||||||
import world.phantasmal.lib.assembly.*
|
import world.phantasmal.lib.asm.*
|
||||||
import world.phantasmal.lib.test.LibTestSuite
|
import world.phantasmal.lib.test.LibTestSuite
|
||||||
import world.phantasmal.lib.test.readFile
|
import world.phantasmal.lib.test.readFile
|
||||||
import kotlin.test.Test
|
import kotlin.test.Test
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
package world.phantasmal.lib.test
|
package world.phantasmal.lib.test
|
||||||
|
|
||||||
import world.phantasmal.core.Success
|
import world.phantasmal.core.Success
|
||||||
import world.phantasmal.lib.assembly.InstructionSegment
|
import world.phantasmal.lib.asm.InstructionSegment
|
||||||
import world.phantasmal.lib.assembly.assemble
|
import world.phantasmal.lib.asm.assemble
|
||||||
import world.phantasmal.lib.cursor.Cursor
|
import world.phantasmal.lib.cursor.Cursor
|
||||||
import kotlin.test.assertTrue
|
import kotlin.test.assertTrue
|
||||||
|
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
@file:JsModule("monaco-editor")
|
@file:JsModule("monaco-editor")
|
||||||
@file:JsNonModule
|
@file:JsNonModule
|
||||||
@file:JsQualifier("languages")
|
@file:JsQualifier("languages")
|
||||||
|
@file:Suppress("unused")
|
||||||
|
|
||||||
package world.phantasmal.web.externals.monacoEditor
|
package world.phantasmal.web.externals.monacoEditor
|
||||||
|
|
||||||
@ -18,6 +19,27 @@ external fun setMonarchTokensProvider(
|
|||||||
languageDef: IMonarchLanguage,
|
languageDef: IMonarchLanguage,
|
||||||
): IDisposable
|
): IDisposable
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register a completion item provider (use by e.g. suggestions).
|
||||||
|
*/
|
||||||
|
external fun registerCompletionItemProvider(
|
||||||
|
languageId: String,
|
||||||
|
provider: CompletionItemProvider,
|
||||||
|
): IDisposable
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register a signature help provider (used by e.g. parameter hints).
|
||||||
|
*/
|
||||||
|
external fun registerSignatureHelpProvider(
|
||||||
|
languageId: String,
|
||||||
|
provider: SignatureHelpProvider,
|
||||||
|
): IDisposable
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register a hover provider (used by e.g. editor hover).
|
||||||
|
*/
|
||||||
|
external fun registerHoverProvider(languageId: String, provider: HoverProvider): IDisposable
|
||||||
|
|
||||||
external interface CommentRule {
|
external interface CommentRule {
|
||||||
var lineComment: String?
|
var lineComment: String?
|
||||||
get() = definedExternally
|
get() = definedExternally
|
||||||
@ -218,3 +240,384 @@ external interface IMonarchLanguageBracket {
|
|||||||
var close: String
|
var close: String
|
||||||
var token: String
|
var token: String
|
||||||
}
|
}
|
||||||
|
|
||||||
|
external interface CompletionItemLabel {
|
||||||
|
/**
|
||||||
|
* The function or variable. Rendered leftmost.
|
||||||
|
*/
|
||||||
|
var name: String
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The signature without the return type. Render after `name`.
|
||||||
|
*/
|
||||||
|
var signature: String?
|
||||||
|
get() = definedExternally
|
||||||
|
set(value) = definedExternally
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The fully qualified name, like package name or file path. Rendered after `signature`.
|
||||||
|
*/
|
||||||
|
var qualifier: String?
|
||||||
|
get() = definedExternally
|
||||||
|
set(value) = definedExternally
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The return-type of a function or type of a property/variable. Rendered rightmost.
|
||||||
|
*/
|
||||||
|
var type: String?
|
||||||
|
get() = definedExternally
|
||||||
|
set(value) = definedExternally
|
||||||
|
}
|
||||||
|
|
||||||
|
external interface CompletionItemRanges {
|
||||||
|
var insert: IRange
|
||||||
|
var replace: IRange
|
||||||
|
}
|
||||||
|
|
||||||
|
external enum class CompletionItemKind {
|
||||||
|
Method /* = 0 */,
|
||||||
|
Function /* = 1 */,
|
||||||
|
Constructor /* = 2 */,
|
||||||
|
Field /* = 3 */,
|
||||||
|
Variable /* = 4 */,
|
||||||
|
Class /* = 5 */,
|
||||||
|
Struct /* = 6 */,
|
||||||
|
Interface /* = 7 */,
|
||||||
|
Module /* = 8 */,
|
||||||
|
Property /* = 9 */,
|
||||||
|
Event /* = 10 */,
|
||||||
|
Operator /* = 11 */,
|
||||||
|
Unit /* = 12 */,
|
||||||
|
Value /* = 13 */,
|
||||||
|
Constant /* = 14 */,
|
||||||
|
Enum /* = 15 */,
|
||||||
|
EnumMember /* = 16 */,
|
||||||
|
Keyword /* = 17 */,
|
||||||
|
Text /* = 18 */,
|
||||||
|
Color /* = 19 */,
|
||||||
|
File /* = 20 */,
|
||||||
|
Reference /* = 21 */,
|
||||||
|
Customcolor /* = 22 */,
|
||||||
|
Folder /* = 23 */,
|
||||||
|
TypeParameter /* = 24 */,
|
||||||
|
Snippet /* = 25 */,
|
||||||
|
}
|
||||||
|
|
||||||
|
external enum class CompletionItemTag {
|
||||||
|
Deprecated /* = 1 */,
|
||||||
|
}
|
||||||
|
|
||||||
|
external enum class CompletionItemInsertTextRule {
|
||||||
|
/**
|
||||||
|
* Adjust whitespace/indentation of multiline insert texts to
|
||||||
|
* match the current line indentation.
|
||||||
|
*/
|
||||||
|
KeepWhitespace /* = 1 */,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* `insertText` is a snippet.
|
||||||
|
*/
|
||||||
|
InsertAsSnippet /* = 4 */,
|
||||||
|
}
|
||||||
|
|
||||||
|
external interface Command {
|
||||||
|
var id: String
|
||||||
|
var title: String
|
||||||
|
var tooltip: String
|
||||||
|
var arguments: Array<dynamic>
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A completion item represents a text snippet that is
|
||||||
|
* proposed to complete text that is being typed.
|
||||||
|
*/
|
||||||
|
external interface CompletionItem {
|
||||||
|
/**
|
||||||
|
* The label of this completion item. By default
|
||||||
|
* this is also the text that is inserted when selecting
|
||||||
|
* this completion.
|
||||||
|
*/
|
||||||
|
var label: CompletionItemLabel /* string | CompletionItemLabel */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The kind of this completion item. Based on the kind
|
||||||
|
* an icon is chosen by the editor.
|
||||||
|
*/
|
||||||
|
var kind: CompletionItemKind
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A modifier to the `kind` which affect how the item
|
||||||
|
* is rendered, e.g. Deprecated is rendered with a strikeout
|
||||||
|
*/
|
||||||
|
var tags: Array<CompletionItemTag>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A human-readable string with additional information
|
||||||
|
* about this item, like type or symbol information.
|
||||||
|
*/
|
||||||
|
var detail: String
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A human-readable string that represents a doc-comment.
|
||||||
|
*/
|
||||||
|
var documentation: String /* string | IMarkdownString */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A string that should be used when comparing this item
|
||||||
|
* with other items. When `falsy` the [label](#CompletionItem.label)
|
||||||
|
* is used.
|
||||||
|
*/
|
||||||
|
var sortText: String
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A string that should be used when filtering a set of
|
||||||
|
* completion items. When `falsy` the [label](#CompletionItem.label)
|
||||||
|
* is used.
|
||||||
|
*/
|
||||||
|
var filterText: String
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Select this item when showing. *Note* that only one completion item can be selected and
|
||||||
|
* that the editor decides which item that is. The rule is that the *first* item of those
|
||||||
|
* that match best is selected.
|
||||||
|
*/
|
||||||
|
var preselect: Boolean
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A string or snippet that should be inserted in a document when selecting
|
||||||
|
* this completion. When `falsy` the [label](#CompletionItem.label)
|
||||||
|
* is used.
|
||||||
|
*/
|
||||||
|
var insertText: String
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Addition rules (as bitmask) that should be applied when inserting
|
||||||
|
* this completion.
|
||||||
|
*/
|
||||||
|
var insertTextRules: CompletionItemInsertTextRule
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A range of text that should be replaced by this completion item.
|
||||||
|
*
|
||||||
|
* Defaults to a range from the start of the [current word](#TextDocument.getWordRangeAtPosition) to the
|
||||||
|
* current position.
|
||||||
|
*
|
||||||
|
* *Note:* The range must be a [single line](#Range.isSingleLine) and it must
|
||||||
|
* [contain](#Range.contains) the position at which completion has been [requested](#CompletionItemProvider.provideCompletionItems).
|
||||||
|
*/
|
||||||
|
var range: CompletionItemRanges
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An optional set of characters that when pressed while this completion is active will accept it first and
|
||||||
|
* then type that character. *Note* that all commit characters should have `length=1` and that superfluous
|
||||||
|
* characters will be ignored.
|
||||||
|
*/
|
||||||
|
var commitCharacters: Array<String>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An optional array of additional text edits that are applied when
|
||||||
|
* selecting this completion. Edits must not overlap with the main edit
|
||||||
|
* nor with themselves.
|
||||||
|
*/
|
||||||
|
var additionalTextEdits: Array<ISingleEditOperation>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A command that should be run upon acceptance of this item.
|
||||||
|
*/
|
||||||
|
var command: Command
|
||||||
|
}
|
||||||
|
|
||||||
|
external interface CompletionList {
|
||||||
|
var suggestions: Array<CompletionItem>
|
||||||
|
var incomplete: Boolean
|
||||||
|
|
||||||
|
fun dispose()
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* How a suggest provider was triggered.
|
||||||
|
*/
|
||||||
|
external enum class CompletionTriggerKind {
|
||||||
|
Invoke /* = 0 */,
|
||||||
|
TriggerCharacter /* = 1 */,
|
||||||
|
TriggerForIncompleteCompletions /* = 2 */,
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Contains additional information about the context in which
|
||||||
|
* [completion provider](#CompletionItemProvider.provideCompletionItems) is triggered.
|
||||||
|
*/
|
||||||
|
external interface CompletionContext {
|
||||||
|
/**
|
||||||
|
* How the completion was triggered.
|
||||||
|
*/
|
||||||
|
var triggerKind: CompletionTriggerKind
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Character that triggered the completion item provider.
|
||||||
|
*
|
||||||
|
* `undefined` if provider was not triggered by a character.
|
||||||
|
*/
|
||||||
|
var triggerCharacter: String
|
||||||
|
}
|
||||||
|
|
||||||
|
external interface CompletionItemProvider {
|
||||||
|
var triggerCharacters: Array<String>?
|
||||||
|
get() = definedExternally
|
||||||
|
set(value) = definedExternally
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provide completion items for the given position and document.
|
||||||
|
*/
|
||||||
|
fun provideCompletionItems(
|
||||||
|
model: ITextModel,
|
||||||
|
position: Position,
|
||||||
|
context: CompletionContext,
|
||||||
|
token: CancellationToken,
|
||||||
|
): CompletionList /* type ProviderResult<T> = T | undefined | null | Thenable<T | undefined | null> */
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents a parameter of a callable-signature. A parameter can
|
||||||
|
* have a label and a doc-comment.
|
||||||
|
*/
|
||||||
|
external interface ParameterInformation {
|
||||||
|
/**
|
||||||
|
* The label of this signature. Will be shown in
|
||||||
|
* the UI.
|
||||||
|
*/
|
||||||
|
var label: Array<Int> /* string | [number, number] */
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The human-readable doc-comment of this signature. Will be shown
|
||||||
|
* in the UI but can be omitted.
|
||||||
|
*
|
||||||
|
* This property is not nullable in TS. Do not assign null to it as null will be interpreted as
|
||||||
|
* IMarkdownString.
|
||||||
|
*/
|
||||||
|
var documentation: String? /* string | IMarkdownString */
|
||||||
|
get() = definedExternally
|
||||||
|
set(value) = definedExternally
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents the signature of something callable. A signature
|
||||||
|
* can have a label, like a function-name, a doc-comment, and
|
||||||
|
* a set of parameters.
|
||||||
|
*/
|
||||||
|
external interface SignatureInformation {
|
||||||
|
/**
|
||||||
|
* The label of this signature. Will be shown in
|
||||||
|
* the UI.
|
||||||
|
*/
|
||||||
|
var label: String
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The human-readable doc-comment of this signature. Will be shown
|
||||||
|
* in the UI but can be omitted.
|
||||||
|
*
|
||||||
|
* This property is not nullable in TS. Do not assign null to it as null will be interpreted as
|
||||||
|
* IMarkdownString.
|
||||||
|
*/
|
||||||
|
var documentation: String? /* string | IMarkdownString */
|
||||||
|
get() = definedExternally
|
||||||
|
set(value) = definedExternally
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The parameters of this signature.
|
||||||
|
*/
|
||||||
|
var parameters: Array<ParameterInformation>
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Signature help represents the signature of something
|
||||||
|
* callable. There can be multiple signatures but only one
|
||||||
|
* active and only one active parameter.
|
||||||
|
*/
|
||||||
|
external interface SignatureHelp {
|
||||||
|
/**
|
||||||
|
* One or more signatures.
|
||||||
|
*/
|
||||||
|
var signatures: Array<SignatureInformation>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The active signature.
|
||||||
|
*/
|
||||||
|
var activeSignature: Int
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The active parameter of the active signature.
|
||||||
|
*/
|
||||||
|
var activeParameter: Int
|
||||||
|
}
|
||||||
|
|
||||||
|
external enum class SignatureHelpTriggerKind {
|
||||||
|
Invoke /* = 1 */,
|
||||||
|
TriggerCharacter /* = 2 */,
|
||||||
|
ContentChange /* = 3 */,
|
||||||
|
}
|
||||||
|
|
||||||
|
external interface SignatureHelpContext {
|
||||||
|
val triggerKind: SignatureHelpTriggerKind
|
||||||
|
val triggerCharacter: String?
|
||||||
|
get() = definedExternally
|
||||||
|
val isRetrigger: Boolean
|
||||||
|
val activeSignatureHelp: SignatureHelp?
|
||||||
|
get() = definedExternally
|
||||||
|
}
|
||||||
|
|
||||||
|
external interface SignatureHelpResult : IDisposable {
|
||||||
|
var value: SignatureHelp
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The signature help provider interface defines the contract between extensions and
|
||||||
|
* the [parameter hints](https://code.visualstudio.com/docs/editor/intellisense)-feature.
|
||||||
|
*/
|
||||||
|
external interface SignatureHelpProvider {
|
||||||
|
val signatureHelpTriggerCharacters: Array<String>?
|
||||||
|
get() = definedExternally
|
||||||
|
val signatureHelpRetriggerCharacters: Array<String>?
|
||||||
|
get() = definedExternally
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provide help for the signature at the given position and document.
|
||||||
|
*/
|
||||||
|
fun provideSignatureHelp(
|
||||||
|
model: ITextModel,
|
||||||
|
position: Position,
|
||||||
|
token: CancellationToken,
|
||||||
|
context: SignatureHelpContext,
|
||||||
|
): SignatureHelpResult? /* type ProviderResult<T> = T | undefined | null | Thenable<T | undefined | null> */
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A hover represents additional information for a symbol or word. Hovers are
|
||||||
|
* rendered in a tooltip-like widget.
|
||||||
|
*/
|
||||||
|
external interface Hover {
|
||||||
|
/**
|
||||||
|
* The contents of this hover.
|
||||||
|
*/
|
||||||
|
var contents: Array<IMarkdownString>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The range to which this hover applies. When missing, the
|
||||||
|
* editor will use the range at the current position or the
|
||||||
|
* current position itself.
|
||||||
|
*/
|
||||||
|
var range: IRange
|
||||||
|
}
|
||||||
|
|
||||||
|
external interface HoverProvider {
|
||||||
|
/**
|
||||||
|
* Provide a hover for the given position and document. Multiple hovers at the same
|
||||||
|
* position will be merged by the editor. A hover can have a range which defaults
|
||||||
|
* to the word range at the position when omitted.
|
||||||
|
*/
|
||||||
|
fun provideHover(
|
||||||
|
model: ITextModel,
|
||||||
|
position: Position,
|
||||||
|
token: CancellationToken,
|
||||||
|
): Hover? /* type ProviderResult<T> = T | undefined | null | Thenable<T | undefined | null> */
|
||||||
|
}
|
||||||
|
@ -5,6 +5,16 @@ typealias IMonarchLanguageRule = IExpandedMonarchLanguageRule
|
|||||||
inline operator fun IMonarchLanguageTokenizer.get(name: String): Array<IMonarchLanguageRule> =
|
inline operator fun IMonarchLanguageTokenizer.get(name: String): Array<IMonarchLanguageRule> =
|
||||||
asDynamic()[name].unsafeCast<Array<IMonarchLanguageRule>>()
|
asDynamic()[name].unsafeCast<Array<IMonarchLanguageRule>>()
|
||||||
|
|
||||||
inline operator fun IMonarchLanguageTokenizer.set(name: String, value: Array<IMonarchLanguageRule>) {
|
inline operator fun IMonarchLanguageTokenizer.set(
|
||||||
|
name: String,
|
||||||
|
value: Array<IMonarchLanguageRule>,
|
||||||
|
) {
|
||||||
|
asDynamic()[name] = value
|
||||||
|
}
|
||||||
|
|
||||||
|
inline operator fun IMarkdownStringUris.get(name: String): UriComponents =
|
||||||
|
asDynamic()[name].unsafeCast<UriComponents>()
|
||||||
|
|
||||||
|
inline operator fun IMarkdownStringUris.set(name: String, value: UriComponents) {
|
||||||
asDynamic()[name] = value
|
asDynamic()[name] = value
|
||||||
}
|
}
|
||||||
|
@ -8,6 +8,19 @@ external interface IDisposable {
|
|||||||
fun dispose()
|
fun dispose()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
external interface CancellationToken {
|
||||||
|
val isCancellationRequested: Boolean
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An event emitted when cancellation is requested
|
||||||
|
* @event
|
||||||
|
*/
|
||||||
|
fun onCancellationRequested(
|
||||||
|
listener: (e: Any) -> Any,
|
||||||
|
thisArg: Any = definedExternally,
|
||||||
|
): IDisposable
|
||||||
|
}
|
||||||
|
|
||||||
external enum class MarkerTag {
|
external enum class MarkerTag {
|
||||||
Unnecessary /* = 1 */,
|
Unnecessary /* = 1 */,
|
||||||
Deprecated /* = 2 */
|
Deprecated /* = 2 */
|
||||||
@ -120,21 +133,21 @@ external enum class SelectionDirection {
|
|||||||
}
|
}
|
||||||
|
|
||||||
external interface IPosition {
|
external interface IPosition {
|
||||||
var lineNumber: Number
|
var lineNumber: Int
|
||||||
var column: Number
|
var column: Int
|
||||||
}
|
}
|
||||||
|
|
||||||
open external class Position(lineNumber: Number, column: Number) {
|
open external class Position(lineNumber: Int, column: Int) {
|
||||||
open var lineNumber: Number
|
open var lineNumber: Int
|
||||||
open var column: Number
|
open var column: Int
|
||||||
open fun with(
|
open fun with(
|
||||||
newLineNumber: Number = definedExternally,
|
newLineNumber: Int = definedExternally,
|
||||||
newColumn: Number = definedExternally,
|
newColumn: Int = definedExternally,
|
||||||
): Position
|
): Position
|
||||||
|
|
||||||
open fun delta(
|
open fun delta(
|
||||||
deltaLineNumber: Number = definedExternally,
|
deltaLineNumber: Int = definedExternally,
|
||||||
deltaColumn: Number = definedExternally,
|
deltaColumn: Int = definedExternally,
|
||||||
): Position
|
): Position
|
||||||
|
|
||||||
open fun equals(other: IPosition): Boolean
|
open fun equals(other: IPosition): Boolean
|
||||||
@ -147,7 +160,7 @@ open external class Position(lineNumber: Number, column: Number) {
|
|||||||
fun equals(a: IPosition?, b: IPosition?): Boolean
|
fun equals(a: IPosition?, b: IPosition?): Boolean
|
||||||
fun isBefore(a: IPosition, b: IPosition): Boolean
|
fun isBefore(a: IPosition, b: IPosition): Boolean
|
||||||
fun isBeforeOrEqual(a: IPosition, b: IPosition): Boolean
|
fun isBeforeOrEqual(a: IPosition, b: IPosition): Boolean
|
||||||
fun compare(a: IPosition, b: IPosition): Number
|
fun compare(a: IPosition, b: IPosition): Int
|
||||||
fun lift(pos: IPosition): Position
|
fun lift(pos: IPosition): Position
|
||||||
fun isIPosition(obj: Any): Boolean
|
fun isIPosition(obj: Any): Boolean
|
||||||
}
|
}
|
||||||
@ -182,6 +195,15 @@ open external class Uri : UriComponents {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
external interface IMarkdownStringUris
|
||||||
|
|
||||||
|
external interface IMarkdownString {
|
||||||
|
var value: String
|
||||||
|
var isTrusted: Boolean
|
||||||
|
var supportThemeIcons: Boolean
|
||||||
|
var uris: IMarkdownStringUris
|
||||||
|
}
|
||||||
|
|
||||||
external object KeyCode {
|
external object KeyCode {
|
||||||
/**
|
/**
|
||||||
* Placed first to cover the 0 value of the enum.
|
* Placed first to cover the 0 value of the enum.
|
||||||
|
@ -0,0 +1,70 @@
|
|||||||
|
package world.phantasmal.web.questEditor.asm
|
||||||
|
|
||||||
|
import world.phantasmal.lib.asm.OPCODES
|
||||||
|
import world.phantasmal.lib.asm.OPCODES_F8
|
||||||
|
import world.phantasmal.lib.asm.OPCODES_F9
|
||||||
|
import world.phantasmal.web.externals.monacoEditor.*
|
||||||
|
import world.phantasmal.webui.obj
|
||||||
|
|
||||||
|
object AsmCompletionItemProvider : CompletionItemProvider {
|
||||||
|
override fun provideCompletionItems(
|
||||||
|
model: ITextModel,
|
||||||
|
position: Position,
|
||||||
|
context: CompletionContext,
|
||||||
|
token: CancellationToken,
|
||||||
|
): CompletionList {
|
||||||
|
val text = model.getValueInRange(obj {
|
||||||
|
startLineNumber = position.lineNumber
|
||||||
|
endLineNumber = position.lineNumber
|
||||||
|
startColumn = 1
|
||||||
|
endColumn = position.column
|
||||||
|
})
|
||||||
|
|
||||||
|
val suggestions = when {
|
||||||
|
KEYWORD_REGEX.matches(text) -> KEYWORD_SUGGESTIONS
|
||||||
|
INSTRUCTION_REGEX.matches(text) -> INSTRUCTION_SUGGESTIONS
|
||||||
|
else -> emptyArray()
|
||||||
|
}
|
||||||
|
|
||||||
|
return obj {
|
||||||
|
this.suggestions = suggestions
|
||||||
|
incomplete = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private val KEYWORD_REGEX = Regex("""^\s*\.[a-z]+${'$'}""")
|
||||||
|
private val KEYWORD_SUGGESTIONS: Array<CompletionItem> =
|
||||||
|
arrayOf(
|
||||||
|
obj {
|
||||||
|
label = obj { name = ".code" }
|
||||||
|
kind = CompletionItemKind.Keyword
|
||||||
|
insertText = "code"
|
||||||
|
},
|
||||||
|
obj {
|
||||||
|
label = obj { name = ".data" }
|
||||||
|
kind = CompletionItemKind.Keyword
|
||||||
|
insertText = "data"
|
||||||
|
},
|
||||||
|
obj {
|
||||||
|
label = obj { name = ".string" }
|
||||||
|
kind = CompletionItemKind.Keyword
|
||||||
|
insertText = "string"
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
private val INSTRUCTION_REGEX = Regex("""^\s*([a-z][a-z0-9_=<>!]*)?${'$'}""")
|
||||||
|
private val INSTRUCTION_SUGGESTIONS: Array<CompletionItem> =
|
||||||
|
(OPCODES + OPCODES_F8 + OPCODES_F9)
|
||||||
|
.filterNotNull()
|
||||||
|
.map { opcode ->
|
||||||
|
obj<CompletionItem> {
|
||||||
|
label = obj {
|
||||||
|
name = opcode.mnemonic
|
||||||
|
// TODO: Add signature?
|
||||||
|
}
|
||||||
|
kind = CompletionItemKind.Function
|
||||||
|
insertText = opcode.mnemonic
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.toTypedArray()
|
||||||
|
}
|
@ -0,0 +1,60 @@
|
|||||||
|
package world.phantasmal.web.questEditor.asm
|
||||||
|
|
||||||
|
import world.phantasmal.core.asArray
|
||||||
|
import world.phantasmal.core.jsArrayOf
|
||||||
|
import world.phantasmal.web.externals.monacoEditor.*
|
||||||
|
import world.phantasmal.webui.obj
|
||||||
|
|
||||||
|
object AsmHoverProvider : HoverProvider {
|
||||||
|
override fun provideHover(
|
||||||
|
model: ITextModel,
|
||||||
|
position: Position,
|
||||||
|
token: CancellationToken,
|
||||||
|
): Hover? {
|
||||||
|
val help = AsmSignatureHelpProvider.getSignatureHelp(model, position)
|
||||||
|
?: return null
|
||||||
|
|
||||||
|
val sig = help.signatures[help.activeSignature]
|
||||||
|
val param = sig.parameters.getOrNull(help.activeParameter)
|
||||||
|
|
||||||
|
val contents = jsArrayOf<IMarkdownString>()
|
||||||
|
|
||||||
|
// Instruction signature. Parameter highlighted if possible.
|
||||||
|
contents.push(
|
||||||
|
obj {
|
||||||
|
value =
|
||||||
|
if (param == null) {
|
||||||
|
sig.label
|
||||||
|
} else {
|
||||||
|
// TODO: Figure out how to underline the active parameter in addition to
|
||||||
|
// bolding it to make it match the look of the signature help.
|
||||||
|
sig.label.substring(0, param.label[0]) +
|
||||||
|
"__" +
|
||||||
|
sig.label.substring(param.label[0], param.label[1]) +
|
||||||
|
"__" +
|
||||||
|
sig.label.substring(param.label[1])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// Put the parameter doc and the instruction doc in the same string to match the look of the
|
||||||
|
// signature help.
|
||||||
|
var doc = ""
|
||||||
|
|
||||||
|
// Parameter doc.
|
||||||
|
if (param?.documentation != null) {
|
||||||
|
doc += param.documentation
|
||||||
|
|
||||||
|
// TODO: Figure out how add an empty line here to make it match the look of the
|
||||||
|
// signature help.
|
||||||
|
doc += "\n\n"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Instruction doc.
|
||||||
|
sig.documentation?.let { doc += it }
|
||||||
|
|
||||||
|
contents.push(obj { value = doc })
|
||||||
|
|
||||||
|
return obj<Hover> { this.contents = contents.asArray() }
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,22 @@
|
|||||||
|
package world.phantasmal.web.questEditor.asm
|
||||||
|
|
||||||
|
import world.phantasmal.web.externals.monacoEditor.*
|
||||||
|
import world.phantasmal.webui.obj
|
||||||
|
import kotlin.js.RegExp
|
||||||
|
|
||||||
|
object AsmLanguageConfiguration : LanguageConfiguration {
|
||||||
|
override var indentationRules: IndentationRule? =
|
||||||
|
obj<IndentationRule> {
|
||||||
|
increaseIndentPattern = RegExp("""^\s*\d+:""")
|
||||||
|
decreaseIndentPattern = RegExp("""^\s*(\d+|\.)""")
|
||||||
|
}
|
||||||
|
|
||||||
|
override var autoClosingPairs: Array<IAutoClosingPairConditional>? =
|
||||||
|
arrayOf(obj { open = "\""; close = "\"" })
|
||||||
|
|
||||||
|
override var surroundingPairs: Array<IAutoClosingPair>? =
|
||||||
|
arrayOf(obj { open = "\""; close = "\"" })
|
||||||
|
|
||||||
|
override var comments: CommentRule? =
|
||||||
|
obj<CommentRule> { lineComment = "//" }
|
||||||
|
}
|
@ -0,0 +1,132 @@
|
|||||||
|
package world.phantasmal.web.questEditor.asm
|
||||||
|
|
||||||
|
import world.phantasmal.web.externals.monacoEditor.IMonarchLanguage
|
||||||
|
import world.phantasmal.web.externals.monacoEditor.IMonarchLanguageTokenizer
|
||||||
|
import world.phantasmal.web.externals.monacoEditor.set
|
||||||
|
import world.phantasmal.webui.obj
|
||||||
|
import kotlin.js.RegExp
|
||||||
|
|
||||||
|
object AsmMonarchLanguage : IMonarchLanguage {
|
||||||
|
override var defaultToken: String? = "invalid"
|
||||||
|
|
||||||
|
override var tokenizer: IMonarchLanguageTokenizer = obj {
|
||||||
|
this["root"] = arrayOf(
|
||||||
|
// Strings.
|
||||||
|
obj {
|
||||||
|
// Unterminated string.
|
||||||
|
regex = RegExp('"' + """([^"\\]|\.)*$""")
|
||||||
|
action = obj { token = "string.invalid" }
|
||||||
|
},
|
||||||
|
obj {
|
||||||
|
regex = RegExp("\"")
|
||||||
|
action = obj {
|
||||||
|
token = "string.quote"
|
||||||
|
bracket = "@open"
|
||||||
|
next = "@string"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// Registers.
|
||||||
|
obj {
|
||||||
|
regex = RegExp("""r\d+""")
|
||||||
|
action = obj { token = "predefined" }
|
||||||
|
},
|
||||||
|
|
||||||
|
// Labels.
|
||||||
|
obj {
|
||||||
|
regex = RegExp("""[^\s]+:""")
|
||||||
|
action = obj { token = "tag" }
|
||||||
|
},
|
||||||
|
|
||||||
|
// Numbers.
|
||||||
|
obj {
|
||||||
|
regex = RegExp("""0x[0-9a-fA-F]+""")
|
||||||
|
action = obj { token = "number.hex" }
|
||||||
|
},
|
||||||
|
obj {
|
||||||
|
regex = RegExp("""-?\d+(\.\d+)?(e-?\d+)?""")
|
||||||
|
action = obj { token = "number.float" }
|
||||||
|
},
|
||||||
|
obj {
|
||||||
|
regex = RegExp("""-?[0-9]+""")
|
||||||
|
action = obj { token = "number" }
|
||||||
|
},
|
||||||
|
|
||||||
|
// Section markers.
|
||||||
|
obj {
|
||||||
|
regex = RegExp("""\.[^\s]+""")
|
||||||
|
action = obj { token = "keyword" }
|
||||||
|
},
|
||||||
|
|
||||||
|
// Identifiers.
|
||||||
|
obj {
|
||||||
|
regex = RegExp("""[a-z][a-z0-9_=<>!]*""")
|
||||||
|
action = obj { token = "identifier" }
|
||||||
|
},
|
||||||
|
|
||||||
|
// Whitespace.
|
||||||
|
obj {
|
||||||
|
regex = RegExp("""[ \t\r\n]+""")
|
||||||
|
action = obj { token = "white" }
|
||||||
|
},
|
||||||
|
// obj {
|
||||||
|
// regex = RegExp("""\/\*""")
|
||||||
|
// action = obj { token = "comment"; next = "@comment" }
|
||||||
|
// },
|
||||||
|
obj {
|
||||||
|
regex = RegExp("\\/\\/.*$")
|
||||||
|
action = obj { token = "comment" }
|
||||||
|
},
|
||||||
|
|
||||||
|
// Delimiters.
|
||||||
|
obj {
|
||||||
|
regex = RegExp(",")
|
||||||
|
action = obj { token = "delimiter" }
|
||||||
|
},
|
||||||
|
)
|
||||||
|
|
||||||
|
// this["comment"] = arrayOf(
|
||||||
|
// obj {
|
||||||
|
// regex = RegExp("""[^/*]+""")
|
||||||
|
// action = obj { token = "comment" }
|
||||||
|
// },
|
||||||
|
// obj {
|
||||||
|
// // Nested comment.
|
||||||
|
// regex = RegExp("""\/\*""")
|
||||||
|
// action = obj { token = "comment"; next = "@push" }
|
||||||
|
// },
|
||||||
|
// obj {
|
||||||
|
// // Nested comment end.
|
||||||
|
// regex = RegExp("""\*/""")
|
||||||
|
// action = obj { token = "comment"; next = "@pop" }
|
||||||
|
// },
|
||||||
|
// obj {
|
||||||
|
// regex = RegExp("""[/*]""")
|
||||||
|
// action = obj { token = "comment" }
|
||||||
|
// },
|
||||||
|
// )
|
||||||
|
|
||||||
|
this["string"] = arrayOf(
|
||||||
|
obj {
|
||||||
|
regex = RegExp("""[^\\"]+""")
|
||||||
|
action = obj { token = "string" }
|
||||||
|
},
|
||||||
|
obj {
|
||||||
|
regex = RegExp("""\\(?:[n\\"])""")
|
||||||
|
action = obj { token = "string.escape" }
|
||||||
|
},
|
||||||
|
obj {
|
||||||
|
regex = RegExp("""\\.""")
|
||||||
|
action = obj { token = "string.escape.invalid" }
|
||||||
|
},
|
||||||
|
obj {
|
||||||
|
regex = RegExp("\"")
|
||||||
|
action = obj {
|
||||||
|
token = "string.quote"
|
||||||
|
bracket = "@close"
|
||||||
|
next = "@pop"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,113 @@
|
|||||||
|
package world.phantasmal.web.questEditor.asm
|
||||||
|
|
||||||
|
import world.phantasmal.core.asArray
|
||||||
|
import world.phantasmal.core.jsArrayOf
|
||||||
|
import world.phantasmal.lib.asm.*
|
||||||
|
import world.phantasmal.web.externals.monacoEditor.*
|
||||||
|
import world.phantasmal.webui.obj
|
||||||
|
|
||||||
|
object AsmSignatureHelpProvider : SignatureHelpProvider {
|
||||||
|
override val signatureHelpTriggerCharacters: Array<String> =
|
||||||
|
arrayOf(" ", ",")
|
||||||
|
|
||||||
|
override val signatureHelpRetriggerCharacters: Array<String> =
|
||||||
|
arrayOf(", ")
|
||||||
|
|
||||||
|
override fun provideSignatureHelp(
|
||||||
|
model: ITextModel,
|
||||||
|
position: Position,
|
||||||
|
token: CancellationToken,
|
||||||
|
context: SignatureHelpContext,
|
||||||
|
): SignatureHelpResult? =
|
||||||
|
getSignatureHelp(model, position)?.let { signatureHelp ->
|
||||||
|
object : SignatureHelpResult {
|
||||||
|
override var value: SignatureHelp = signatureHelp
|
||||||
|
|
||||||
|
override fun dispose() {
|
||||||
|
// Nothing to dispose.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun getSignatureHelp(model: ITextModel, position: Position): SignatureHelp? {
|
||||||
|
// Hacky way of providing parameter hints.
|
||||||
|
// We just tokenize the current line and look for the first identifier and check whether
|
||||||
|
// it's a valid opcode.
|
||||||
|
var signatureInfo: SignatureInformation? = null
|
||||||
|
var activeParam = -1
|
||||||
|
val line = model.getLineContent(position.lineNumber)
|
||||||
|
|
||||||
|
val tokens = tokenizeLine(line)
|
||||||
|
|
||||||
|
tokens.find { it is Token.Ident }?.let { ident ->
|
||||||
|
ident as Token.Ident
|
||||||
|
|
||||||
|
mnemonicToOpcode(ident.value)?.let { opcode ->
|
||||||
|
signatureInfo = getSignatureInformation(opcode)
|
||||||
|
|
||||||
|
for (tkn in tokens) {
|
||||||
|
if (tkn.col + tkn.len > position.column) {
|
||||||
|
break
|
||||||
|
} else if (tkn is Token.Ident && activeParam == -1) {
|
||||||
|
activeParam = 0
|
||||||
|
} else if (tkn is Token.ArgSeparator) {
|
||||||
|
activeParam++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return signatureInfo?.let { sigInfo ->
|
||||||
|
obj<SignatureHelp> {
|
||||||
|
signatures = arrayOf(sigInfo)
|
||||||
|
activeSignature = 0
|
||||||
|
activeParameter = activeParam
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getSignatureInformation(opcode: Opcode): SignatureInformation {
|
||||||
|
var signature = opcode.mnemonic + " "
|
||||||
|
val params = jsArrayOf<ParameterInformation>()
|
||||||
|
var first = true
|
||||||
|
|
||||||
|
for (param in opcode.params) {
|
||||||
|
if (first) {
|
||||||
|
first = false
|
||||||
|
} else {
|
||||||
|
signature += ", "
|
||||||
|
}
|
||||||
|
|
||||||
|
val paramTypeStr = when (param.type) {
|
||||||
|
ByteType -> "Byte"
|
||||||
|
ShortType -> "Short"
|
||||||
|
IntType -> "Int"
|
||||||
|
FloatType -> "Float"
|
||||||
|
ILabelType -> "&Function"
|
||||||
|
DLabelType -> "&Data"
|
||||||
|
SLabelType -> "&String"
|
||||||
|
ILabelVarType -> "...&Function"
|
||||||
|
StringType -> "String"
|
||||||
|
RegRefType, is RegTupRefType -> "Register"
|
||||||
|
RegRefVarType -> "...Register"
|
||||||
|
PointerType -> "Pointer"
|
||||||
|
else -> "Any"
|
||||||
|
}
|
||||||
|
|
||||||
|
params.push(
|
||||||
|
obj {
|
||||||
|
label = arrayOf(signature.length, signature.length + paramTypeStr.length)
|
||||||
|
param.doc?.let { documentation = it }
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
signature += paramTypeStr
|
||||||
|
}
|
||||||
|
|
||||||
|
return obj {
|
||||||
|
label = signature
|
||||||
|
opcode.doc?.let { documentation = it }
|
||||||
|
parameters = params.asArray()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,6 +1,6 @@
|
|||||||
package world.phantasmal.web.questEditor.models
|
package world.phantasmal.web.questEditor.models
|
||||||
|
|
||||||
import world.phantasmal.lib.assembly.Segment
|
import world.phantasmal.lib.asm.Segment
|
||||||
import world.phantasmal.lib.fileFormats.quest.Episode
|
import world.phantasmal.lib.fileFormats.quest.Episode
|
||||||
import world.phantasmal.observable.value.Val
|
import world.phantasmal.observable.value.Val
|
||||||
import world.phantasmal.observable.value.list.ListVal
|
import world.phantasmal.observable.value.list.ListVal
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
package world.phantasmal.web.questEditor.stores
|
package world.phantasmal.web.questEditor.stores
|
||||||
|
|
||||||
import world.phantasmal.lib.assembly.disassemble
|
import world.phantasmal.lib.asm.disassemble
|
||||||
import world.phantasmal.observable.ChangeEvent
|
import world.phantasmal.observable.ChangeEvent
|
||||||
import world.phantasmal.observable.Observable
|
import world.phantasmal.observable.Observable
|
||||||
import world.phantasmal.observable.emitter
|
import world.phantasmal.observable.emitter
|
||||||
@ -10,10 +10,10 @@ import world.phantasmal.observable.value.trueVal
|
|||||||
import world.phantasmal.web.core.undo.SimpleUndo
|
import world.phantasmal.web.core.undo.SimpleUndo
|
||||||
import world.phantasmal.web.core.undo.UndoManager
|
import world.phantasmal.web.core.undo.UndoManager
|
||||||
import world.phantasmal.web.externals.monacoEditor.*
|
import world.phantasmal.web.externals.monacoEditor.*
|
||||||
|
import world.phantasmal.web.questEditor.asm.*
|
||||||
import world.phantasmal.web.questEditor.models.QuestModel
|
import world.phantasmal.web.questEditor.models.QuestModel
|
||||||
import world.phantasmal.webui.obj
|
import world.phantasmal.webui.obj
|
||||||
import world.phantasmal.webui.stores.Store
|
import world.phantasmal.webui.stores.Store
|
||||||
import kotlin.js.RegExp
|
|
||||||
|
|
||||||
class AsmStore(
|
class AsmStore(
|
||||||
questEditorStore: QuestEditorStore,
|
questEditorStore: QuestEditorStore,
|
||||||
@ -103,141 +103,11 @@ class AsmStore(
|
|||||||
|
|
||||||
init {
|
init {
|
||||||
register(obj { id = ASM_LANG_ID })
|
register(obj { id = ASM_LANG_ID })
|
||||||
|
setMonarchTokensProvider(ASM_LANG_ID, AsmMonarchLanguage)
|
||||||
setMonarchTokensProvider(ASM_LANG_ID, obj {
|
setLanguageConfiguration(ASM_LANG_ID, AsmLanguageConfiguration)
|
||||||
defaultToken = "invalid"
|
registerCompletionItemProvider(ASM_LANG_ID, AsmCompletionItemProvider)
|
||||||
|
registerSignatureHelpProvider(ASM_LANG_ID, AsmSignatureHelpProvider)
|
||||||
tokenizer = obj {
|
registerHoverProvider(ASM_LANG_ID, AsmHoverProvider)
|
||||||
this["root"] = arrayOf(
|
|
||||||
// Strings.
|
|
||||||
obj {
|
|
||||||
// Unterminated string.
|
|
||||||
regex = RegExp('"' + """([^"\\]|\.)*$""")
|
|
||||||
action = obj { token = "string.invalid" }
|
|
||||||
},
|
|
||||||
obj {
|
|
||||||
regex = RegExp("\"")
|
|
||||||
action = obj {
|
|
||||||
token = "string.quote"
|
|
||||||
bracket = "@open"
|
|
||||||
next = "@string"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// Registers.
|
|
||||||
obj {
|
|
||||||
regex = RegExp("""r\d+""")
|
|
||||||
action = obj { token = "predefined" }
|
|
||||||
},
|
|
||||||
|
|
||||||
// Labels.
|
|
||||||
obj {
|
|
||||||
regex = RegExp("""[^\s]+:""")
|
|
||||||
action = obj { token = "tag" }
|
|
||||||
},
|
|
||||||
|
|
||||||
// Numbers.
|
|
||||||
obj {
|
|
||||||
regex = RegExp("""0x[0-9a-fA-F]+""")
|
|
||||||
action = obj { token = "number.hex" }
|
|
||||||
},
|
|
||||||
obj {
|
|
||||||
regex = RegExp("""-?\d+(\.\d+)?(e-?\d+)?""")
|
|
||||||
action = obj { token = "number.float" }
|
|
||||||
},
|
|
||||||
obj {
|
|
||||||
regex = RegExp("""-?[0-9]+""")
|
|
||||||
action = obj { token = "number" }
|
|
||||||
},
|
|
||||||
|
|
||||||
// Section markers.
|
|
||||||
obj {
|
|
||||||
regex = RegExp("""\.[^\s]+""")
|
|
||||||
action = obj { token = "keyword" }
|
|
||||||
},
|
|
||||||
|
|
||||||
// Identifiers.
|
|
||||||
obj {
|
|
||||||
regex = RegExp("""[a-z][a-z0-9_=<>!]*""")
|
|
||||||
action = obj { token = "identifier" }
|
|
||||||
},
|
|
||||||
|
|
||||||
// Whitespace.
|
|
||||||
obj {
|
|
||||||
regex = RegExp("""[ \t\r\n]+""")
|
|
||||||
action = obj { token = "white" }
|
|
||||||
},
|
|
||||||
// obj {
|
|
||||||
// regex = RegExp("""\/\*""")
|
|
||||||
// action = obj { token = "comment"; next = "@comment" }
|
|
||||||
// },
|
|
||||||
obj {
|
|
||||||
regex = RegExp("\\/\\/.*$")
|
|
||||||
action = obj { token = "comment" }
|
|
||||||
},
|
|
||||||
|
|
||||||
// Delimiters.
|
|
||||||
obj {
|
|
||||||
regex = RegExp(",")
|
|
||||||
action = obj { token = "delimiter" }
|
|
||||||
},
|
|
||||||
)
|
|
||||||
|
|
||||||
// this["comment"] = arrayOf(
|
|
||||||
// obj {
|
|
||||||
// regex = RegExp("""[^/*]+""")
|
|
||||||
// action = obj { token = "comment" }
|
|
||||||
// },
|
|
||||||
// obj {
|
|
||||||
// // Nested comment.
|
|
||||||
// regex = RegExp("""\/\*""")
|
|
||||||
// action = obj { token = "comment"; next = "@push" }
|
|
||||||
// },
|
|
||||||
// obj {
|
|
||||||
// // Nested comment end.
|
|
||||||
// regex = RegExp("""\*/""")
|
|
||||||
// action = obj { token = "comment"; next = "@pop" }
|
|
||||||
// },
|
|
||||||
// obj {
|
|
||||||
// regex = RegExp("""[/*]""")
|
|
||||||
// action = obj { token = "comment" }
|
|
||||||
// },
|
|
||||||
// )
|
|
||||||
|
|
||||||
this["string"] = arrayOf(
|
|
||||||
obj {
|
|
||||||
regex = RegExp("""[^\\"]+""")
|
|
||||||
action = obj { token = "string" }
|
|
||||||
},
|
|
||||||
obj {
|
|
||||||
regex = RegExp("""\\(?:[n\\"])""")
|
|
||||||
action = obj { token = "string.escape" }
|
|
||||||
},
|
|
||||||
obj {
|
|
||||||
regex = RegExp("""\\.""")
|
|
||||||
action = obj { token = "string.escape.invalid" }
|
|
||||||
},
|
|
||||||
obj {
|
|
||||||
regex = RegExp("\"")
|
|
||||||
action = obj {
|
|
||||||
token = "string.quote"
|
|
||||||
bracket = "@close"
|
|
||||||
next = "@pop"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
setLanguageConfiguration(ASM_LANG_ID, obj {
|
|
||||||
indentationRules = obj<IndentationRule> {
|
|
||||||
increaseIndentPattern = RegExp("^\\s*\\d+:")
|
|
||||||
decreaseIndentPattern = RegExp("^\\s*(\\d+|\\.)")
|
|
||||||
}
|
|
||||||
autoClosingPairs = arrayOf(obj { open = "\""; close = "\"" })
|
|
||||||
surroundingPairs = arrayOf(obj { open = "\""; close = "\"" })
|
|
||||||
comments = obj<CommentRule> { lineComment = "//" }
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
package world.phantasmal.web.test
|
package world.phantasmal.web.test
|
||||||
|
|
||||||
import world.phantasmal.lib.assembly.Segment
|
import world.phantasmal.lib.asm.Segment
|
||||||
import world.phantasmal.lib.fileFormats.quest.Episode
|
import world.phantasmal.lib.fileFormats.quest.Episode
|
||||||
import world.phantasmal.lib.fileFormats.quest.NpcType
|
import world.phantasmal.lib.fileFormats.quest.NpcType
|
||||||
import world.phantasmal.lib.fileFormats.quest.QuestNpc
|
import world.phantasmal.lib.fileFormats.quest.QuestNpc
|
||||||
|
Loading…
Reference in New Issue
Block a user