mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
Added type definitions and documentation to completion suggestions.
This commit is contained in:
parent
5210792a3e
commit
9494a70591
@ -54,7 +54,7 @@ class QstTests : LibTestSuite {
|
|||||||
"/ep2/shop/gallon.qst",
|
"/ep2/shop/gallon.qst",
|
||||||
"/princ/ep1/",
|
"/princ/ep1/",
|
||||||
"/princ/ep4/",
|
"/princ/ep4/",
|
||||||
"/solo/ep1/04.qst", // Skip because it contains every chuck twice.
|
"/solo/ep1/04.qst", // Skip because it contains every chunk twice.
|
||||||
"/fragmentofmemoryen.qst",
|
"/fragmentofmemoryen.qst",
|
||||||
"/lost havoc vulcan.qst",
|
"/lost havoc vulcan.qst",
|
||||||
"/goodluck.qst",
|
"/goodluck.qst",
|
||||||
|
@ -316,88 +316,6 @@ class AssemblyWorker(private val sendMessage: (ServerMessage) -> Unit) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun getSignature(opcode: Opcode): Signature {
|
|
||||||
val signature = StringBuilder(opcode.mnemonic).append(" ")
|
|
||||||
val params = mutableListOf<Parameter>()
|
|
||||||
var first = true
|
|
||||||
|
|
||||||
for (param in opcode.params) {
|
|
||||||
if (first) {
|
|
||||||
first = false
|
|
||||||
} else {
|
|
||||||
signature.append(", ")
|
|
||||||
}
|
|
||||||
|
|
||||||
val labelStart = signature.length
|
|
||||||
|
|
||||||
signature.appendParam(param)
|
|
||||||
|
|
||||||
params.add(
|
|
||||||
Parameter(
|
|
||||||
labelStart,
|
|
||||||
labelEnd = signature.length,
|
|
||||||
documentation = param.doc,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
return Signature(
|
|
||||||
label = signature.toString(),
|
|
||||||
documentation = opcode.doc,
|
|
||||||
parameters = params,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun StringBuilder.appendParam(param: Param) {
|
|
||||||
if (param.read || param.write) {
|
|
||||||
if (param.read) append("in")
|
|
||||||
if (param.write) append("out")
|
|
||||||
append(" ")
|
|
||||||
}
|
|
||||||
|
|
||||||
when (val type = param.type) {
|
|
||||||
AnyType.Instance -> append("Any")
|
|
||||||
ByteType -> append("Byte")
|
|
||||||
ShortType -> append("Short")
|
|
||||||
IntType -> append("Int")
|
|
||||||
FloatType -> append("Float")
|
|
||||||
LabelType.Instance -> append("Label")
|
|
||||||
ILabelType -> append("ILabel")
|
|
||||||
DLabelType -> append("DLabel")
|
|
||||||
SLabelType -> append("SLabel")
|
|
||||||
ILabelVarType -> append("...ILabel")
|
|
||||||
StringType -> append("String")
|
|
||||||
is RegType -> {
|
|
||||||
append("Reg")
|
|
||||||
|
|
||||||
type.registers?.let { registers ->
|
|
||||||
append("<")
|
|
||||||
|
|
||||||
var first = true
|
|
||||||
|
|
||||||
for (register in registers) {
|
|
||||||
if (first) {
|
|
||||||
first = false
|
|
||||||
} else {
|
|
||||||
append(", ")
|
|
||||||
}
|
|
||||||
|
|
||||||
appendParam(register)
|
|
||||||
}
|
|
||||||
|
|
||||||
append(">")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
RegVarType -> append("...Reg")
|
|
||||||
PointerType -> append("Pointer")
|
|
||||||
}
|
|
||||||
|
|
||||||
param.name?.let {
|
|
||||||
append(" ")
|
|
||||||
append(param.name)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun getHover(requestId: Int, lineNo: Int, col: Int) {
|
private fun getHover(requestId: Int, lineNo: Int, col: Int) {
|
||||||
val hover = signatureHelp(lineNo, col)?.let { help ->
|
val hover = signatureHelp(lineNo, col)?.let { help ->
|
||||||
val sig = help.signature
|
val sig = help.signature
|
||||||
@ -559,16 +477,22 @@ class AssemblyWorker(private val sendMessage: (ServerMessage) -> Unit) {
|
|||||||
CompletionItem(
|
CompletionItem(
|
||||||
label = ".code",
|
label = ".code",
|
||||||
type = CompletionItemType.Keyword,
|
type = CompletionItemType.Keyword,
|
||||||
|
detail = null,
|
||||||
|
documentation = "Start of a code segment",
|
||||||
insertText = "code",
|
insertText = "code",
|
||||||
),
|
),
|
||||||
CompletionItem(
|
CompletionItem(
|
||||||
label = ".data",
|
label = ".data",
|
||||||
type = CompletionItemType.Keyword,
|
type = CompletionItemType.Keyword,
|
||||||
|
detail = null,
|
||||||
|
documentation = "Start of a data segment",
|
||||||
insertText = "data",
|
insertText = "data",
|
||||||
),
|
),
|
||||||
CompletionItem(
|
CompletionItem(
|
||||||
label = ".string",
|
label = ".string",
|
||||||
type = CompletionItemType.Keyword,
|
type = CompletionItemType.Keyword,
|
||||||
|
detail = null,
|
||||||
|
documentation = "Start of a string data segment",
|
||||||
insertText = "string",
|
insertText = "string",
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
@ -578,14 +502,98 @@ class AssemblyWorker(private val sendMessage: (ServerMessage) -> Unit) {
|
|||||||
(OPCODES.asSequence() + OPCODES_F8.asSequence() + OPCODES_F9.asSequence())
|
(OPCODES.asSequence() + OPCODES_F8.asSequence() + OPCODES_F9.asSequence())
|
||||||
.filterNotNull()
|
.filterNotNull()
|
||||||
.map { opcode ->
|
.map { opcode ->
|
||||||
|
val sig = getSignature(opcode)
|
||||||
CompletionItem(
|
CompletionItem(
|
||||||
label = opcode.mnemonic,
|
label = opcode.mnemonic,
|
||||||
// TODO: Add signature?
|
|
||||||
type = CompletionItemType.Opcode,
|
type = CompletionItemType.Opcode,
|
||||||
insertText = opcode.mnemonic,
|
detail = sig.label,
|
||||||
|
documentation = sig.documentation,
|
||||||
|
insertText = "${opcode.mnemonic} ",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
.sortedBy { it.label }
|
.sortedBy { it.label }
|
||||||
.toList()
|
.toList()
|
||||||
|
|
||||||
|
private fun getSignature(opcode: Opcode): Signature {
|
||||||
|
val signature = StringBuilder(opcode.mnemonic).append(" ")
|
||||||
|
val params = mutableListOf<Parameter>()
|
||||||
|
var first = true
|
||||||
|
|
||||||
|
for (param in opcode.params) {
|
||||||
|
if (first) {
|
||||||
|
first = false
|
||||||
|
} else {
|
||||||
|
signature.append(", ")
|
||||||
|
}
|
||||||
|
|
||||||
|
val labelStart = signature.length
|
||||||
|
|
||||||
|
signature.appendParam(param)
|
||||||
|
|
||||||
|
params.add(
|
||||||
|
Parameter(
|
||||||
|
labelStart,
|
||||||
|
labelEnd = signature.length,
|
||||||
|
documentation = param.doc,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
return Signature(
|
||||||
|
label = signature.toString(),
|
||||||
|
documentation = opcode.doc,
|
||||||
|
parameters = params,
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun StringBuilder.appendParam(param: Param) {
|
||||||
|
if (param.read || param.write) {
|
||||||
|
if (param.read) append("in")
|
||||||
|
if (param.write) append("out")
|
||||||
|
append(" ")
|
||||||
|
}
|
||||||
|
|
||||||
|
when (val type = param.type) {
|
||||||
|
AnyType.Instance -> append("Any")
|
||||||
|
ByteType -> append("Byte")
|
||||||
|
ShortType -> append("Short")
|
||||||
|
IntType -> append("Int")
|
||||||
|
FloatType -> append("Float")
|
||||||
|
LabelType.Instance -> append("Label")
|
||||||
|
ILabelType -> append("ILabel")
|
||||||
|
DLabelType -> append("DLabel")
|
||||||
|
SLabelType -> append("SLabel")
|
||||||
|
ILabelVarType -> append("...ILabel")
|
||||||
|
StringType -> append("String")
|
||||||
|
is RegType -> {
|
||||||
|
append("Reg")
|
||||||
|
|
||||||
|
type.registers?.let { registers ->
|
||||||
|
append("<")
|
||||||
|
|
||||||
|
var first = true
|
||||||
|
|
||||||
|
for (register in registers) {
|
||||||
|
if (first) {
|
||||||
|
first = false
|
||||||
|
} else {
|
||||||
|
append(", ")
|
||||||
|
}
|
||||||
|
|
||||||
|
appendParam(register)
|
||||||
|
}
|
||||||
|
|
||||||
|
append(">")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
RegVarType -> append("...Reg")
|
||||||
|
PointerType -> append("Pointer")
|
||||||
|
}
|
||||||
|
|
||||||
|
param.name?.let {
|
||||||
|
append(" ")
|
||||||
|
append(param.name)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -113,7 +113,13 @@ enum class CompletionItemType {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Serializable
|
@Serializable
|
||||||
class CompletionItem(val label: String, val type: CompletionItemType, val insertText: String)
|
class CompletionItem(
|
||||||
|
val label: String,
|
||||||
|
val type: CompletionItemType,
|
||||||
|
val detail: String?,
|
||||||
|
val documentation: String?,
|
||||||
|
val insertText: String,
|
||||||
|
)
|
||||||
|
|
||||||
@Serializable
|
@Serializable
|
||||||
class SignatureHelp(val signature: Signature, val activeParameter: Int)
|
class SignatureHelp(val signature: Signature, val activeParameter: Int)
|
||||||
|
@ -63,7 +63,7 @@ class QuestEditor(
|
|||||||
val questInfoController = addDisposable(QuestInfoController(questEditorStore))
|
val questInfoController = addDisposable(QuestInfoController(questEditorStore))
|
||||||
val npcCountsController = addDisposable(NpcCountsController(questEditorStore))
|
val npcCountsController = addDisposable(NpcCountsController(questEditorStore))
|
||||||
val entityInfoController = addDisposable(EntityInfoController(areaStore, questEditorStore))
|
val entityInfoController = addDisposable(EntityInfoController(areaStore, questEditorStore))
|
||||||
val asmController = addDisposable(AsmController(asmStore))
|
val asmController = addDisposable(AsmEditorController(asmStore))
|
||||||
val npcListController = addDisposable(EntityListController(questEditorStore, npcs = true))
|
val npcListController = addDisposable(EntityListController(questEditorStore, npcs = true))
|
||||||
val objectListController =
|
val objectListController =
|
||||||
addDisposable(EntityListController(questEditorStore, npcs = false))
|
addDisposable(EntityListController(questEditorStore, npcs = false))
|
||||||
|
@ -7,6 +7,7 @@ import kotlinx.coroutines.suspendCancellableCoroutine
|
|||||||
import kotlinx.coroutines.withTimeout
|
import kotlinx.coroutines.withTimeout
|
||||||
import kotlinx.serialization.decodeFromString
|
import kotlinx.serialization.decodeFromString
|
||||||
import kotlinx.serialization.encodeToString
|
import kotlinx.serialization.encodeToString
|
||||||
|
import mu.KotlinLogging
|
||||||
import org.w3c.dom.Worker
|
import org.w3c.dom.Worker
|
||||||
import world.phantasmal.observable.ChangeEvent
|
import world.phantasmal.observable.ChangeEvent
|
||||||
import world.phantasmal.observable.Observable
|
import world.phantasmal.observable.Observable
|
||||||
@ -18,6 +19,8 @@ import world.phantasmal.web.shared.messages.*
|
|||||||
import kotlin.coroutines.Continuation
|
import kotlin.coroutines.Continuation
|
||||||
import kotlin.coroutines.resume
|
import kotlin.coroutines.resume
|
||||||
|
|
||||||
|
private val logger = KotlinLogging.logger {}
|
||||||
|
|
||||||
class AsmAnalyser {
|
class AsmAnalyser {
|
||||||
private var inlineStackArgs: Boolean = true
|
private var inlineStackArgs: Boolean = true
|
||||||
private var _mapDesignations = emitter<Map<Int, Int>>()
|
private var _mapDesignations = emitter<Map<Int, Int>>()
|
||||||
@ -95,8 +98,16 @@ class AsmAnalyser {
|
|||||||
}
|
}
|
||||||
|
|
||||||
is Response<*> -> {
|
is Response<*> -> {
|
||||||
val continuation = inFlightRequests[message.id].unsafeCast<Continuation<Any?>?>()
|
val continuation = inFlightRequests.remove(message.id)
|
||||||
continuation?.resume(message.result.unsafeCast<Any?>())
|
|
||||||
|
if (continuation == null) {
|
||||||
|
logger.warn {
|
||||||
|
"No continuation for ${message::class.simpleName} ${message.id}, possibly due to timeout."
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
continuation.unsafeCast<Continuation<Any>>()
|
||||||
|
.resume(message.result.unsafeCast<Any>())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -32,6 +32,8 @@ class AsmCompletionItemProvider(private val analyser: AsmAnalyser) : CompletionI
|
|||||||
CompletionItemType.Opcode -> CompletionItemKind.Function
|
CompletionItemType.Opcode -> CompletionItemKind.Function
|
||||||
}
|
}
|
||||||
insertText = completion.insertText
|
insertText = completion.insertText
|
||||||
|
completion.detail?.let { detail = it }
|
||||||
|
completion.documentation?.let { documentation = it }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
incomplete = false
|
incomplete = false
|
||||||
|
@ -10,7 +10,7 @@ import world.phantasmal.web.externals.monacoEditor.createModel
|
|||||||
import world.phantasmal.web.questEditor.stores.AsmStore
|
import world.phantasmal.web.questEditor.stores.AsmStore
|
||||||
import world.phantasmal.webui.controllers.Controller
|
import world.phantasmal.webui.controllers.Controller
|
||||||
|
|
||||||
class AsmController(private val store: AsmStore) : Controller() {
|
class AsmEditorController(private val store: AsmStore) : Controller() {
|
||||||
val enabled: Val<Boolean> = store.editingEnabled
|
val enabled: Val<Boolean> = store.editingEnabled
|
||||||
val readOnly: Val<Boolean> = !enabled or store.textModel.isNull()
|
val readOnly: Val<Boolean> = !enabled or store.textModel.isNull()
|
||||||
|
|
@ -4,12 +4,12 @@ import org.w3c.dom.Node
|
|||||||
import world.phantasmal.core.disposable.disposable
|
import world.phantasmal.core.disposable.disposable
|
||||||
import world.phantasmal.web.externals.monacoEditor.*
|
import world.phantasmal.web.externals.monacoEditor.*
|
||||||
import world.phantasmal.web.questEditor.asm.monaco.EditorHistory
|
import world.phantasmal.web.questEditor.asm.monaco.EditorHistory
|
||||||
import world.phantasmal.web.questEditor.controllers.AsmController
|
import world.phantasmal.web.questEditor.controllers.AsmEditorController
|
||||||
import world.phantasmal.webui.dom.div
|
import world.phantasmal.webui.dom.div
|
||||||
import world.phantasmal.webui.obj
|
import world.phantasmal.webui.obj
|
||||||
import world.phantasmal.webui.widgets.Widget
|
import world.phantasmal.webui.widgets.Widget
|
||||||
|
|
||||||
class AsmEditorWidget(private val ctrl: AsmController) : Widget() {
|
class AsmEditorWidget(private val ctrl: AsmEditorController) : Widget() {
|
||||||
private lateinit var editor: IStandaloneCodeEditor
|
private lateinit var editor: IStandaloneCodeEditor
|
||||||
|
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
|
@ -1,18 +1,19 @@
|
|||||||
package world.phantasmal.web.questEditor.widgets
|
package world.phantasmal.web.questEditor.widgets
|
||||||
|
|
||||||
import org.w3c.dom.Node
|
import org.w3c.dom.Node
|
||||||
import world.phantasmal.web.questEditor.controllers.AsmController
|
import world.phantasmal.web.questEditor.controllers.AsmEditorController
|
||||||
import world.phantasmal.webui.dom.div
|
import world.phantasmal.webui.dom.div
|
||||||
import world.phantasmal.webui.widgets.Checkbox
|
import world.phantasmal.webui.widgets.Checkbox
|
||||||
import world.phantasmal.webui.widgets.Toolbar
|
import world.phantasmal.webui.widgets.Toolbar
|
||||||
import world.phantasmal.webui.widgets.Widget
|
import world.phantasmal.webui.widgets.Widget
|
||||||
|
|
||||||
class AsmToolbarWidget(private val ctrl: AsmController) : Widget() {
|
class AsmToolbarWidget(private val ctrl: AsmEditorController) : Widget() {
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
div {
|
div {
|
||||||
className = "pw-quest-editor-asm-toolbar"
|
className = "pw-quest-editor-asm-toolbar"
|
||||||
|
|
||||||
addChild(Toolbar(
|
addChild(
|
||||||
|
Toolbar(
|
||||||
enabled = ctrl.enabled,
|
enabled = ctrl.enabled,
|
||||||
children = listOf(
|
children = listOf(
|
||||||
Checkbox(
|
Checkbox(
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
package world.phantasmal.web.questEditor.widgets
|
package world.phantasmal.web.questEditor.widgets
|
||||||
|
|
||||||
import org.w3c.dom.Node
|
import org.w3c.dom.Node
|
||||||
import world.phantasmal.web.questEditor.controllers.AsmController
|
import world.phantasmal.web.questEditor.controllers.AsmEditorController
|
||||||
import world.phantasmal.webui.dom.div
|
import world.phantasmal.webui.dom.div
|
||||||
import world.phantasmal.webui.widgets.Widget
|
import world.phantasmal.webui.widgets.Widget
|
||||||
|
|
||||||
class AsmWidget(private val ctrl: AsmController) : Widget() {
|
class AsmWidget(private val ctrl: AsmEditorController) : Widget() {
|
||||||
private lateinit var editorWidget: AsmEditorWidget
|
private lateinit var editorWidget: AsmEditorWidget
|
||||||
|
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
|
Loading…
Reference in New Issue
Block a user