From 4b8241ba80facd68f5b3aecd38402e63d7818ea2 Mon Sep 17 00:00:00 2001 From: Daan Vanden Bosch Date: Thu, 15 Apr 2021 16:16:00 +0200 Subject: [PATCH] Existing files can now be saved directly with the "Save" button or Ctrl-S. --- .../world/phantasmal/core/JsExtensions.kt | 11 +++ .../core}/externals/browser/Browser.kt | 29 +++++- .../phantasmal/lib/fileFormats/quest/Quest.kt | 13 ++- .../observable/value/ValExtensions.kt | 3 + .../observable/value/list/AbstractListVal.kt | 5 + .../observable/value/list/ListVal.kt | 4 + .../observable/value/list/StaticListVal.kt | 6 +- .../QuestEditorToolbarController.kt | 99 +++++++++++++++---- .../widgets/QuestEditorToolbarWidget.kt | 7 ++ .../world/phantasmal/webui/files/Files.kt | 39 +++++--- 10 files changed, 180 insertions(+), 36 deletions(-) create mode 100644 core/src/jsMain/kotlin/world/phantasmal/core/JsExtensions.kt rename {webui/src/main/kotlin/world/phantasmal/webui => core/src/jsMain/kotlin/world/phantasmal/core}/externals/browser/Browser.kt (55%) diff --git a/core/src/jsMain/kotlin/world/phantasmal/core/JsExtensions.kt b/core/src/jsMain/kotlin/world/phantasmal/core/JsExtensions.kt new file mode 100644 index 00000000..b9fd8d09 --- /dev/null +++ b/core/src/jsMain/kotlin/world/phantasmal/core/JsExtensions.kt @@ -0,0 +1,11 @@ +package world.phantasmal.core + +import world.phantasmal.core.externals.browser.WritableStream + +inline fun S.use(block: (S) -> R): R { + try { + return block(this) + } finally { + close() + } +} diff --git a/webui/src/main/kotlin/world/phantasmal/webui/externals/browser/Browser.kt b/core/src/jsMain/kotlin/world/phantasmal/core/externals/browser/Browser.kt similarity index 55% rename from webui/src/main/kotlin/world/phantasmal/webui/externals/browser/Browser.kt rename to core/src/jsMain/kotlin/world/phantasmal/core/externals/browser/Browser.kt index 47b67e2c..680e3f29 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/externals/browser/Browser.kt +++ b/core/src/jsMain/kotlin/world/phantasmal/core/externals/browser/Browser.kt @@ -1,6 +1,9 @@ -package world.phantasmal.webui.externals.browser +@file:Suppress("unused") + +package world.phantasmal.core.externals.browser import org.khronos.webgl.ArrayBuffer +import org.khronos.webgl.BufferDataSource import org.w3c.dom.Window import org.w3c.files.Blob import org.w3c.files.File @@ -16,6 +19,30 @@ open external class FileSystemHandle { external class FileSystemFileHandle : FileSystemHandle { fun getFile(): Promise + + fun createWritable(): Promise +} + +open external class WritableStream { + val locked: Boolean + + fun abort(reason: Any): Promise + + fun close(): Promise +} + +external interface FileSystemWritableFileStreamData { + var type: String /* "write" | "seek" | "truncate" */ + var data: dynamic /* BufferDataSource | Blob | String */ + var position: Int + var size: Int +} + +external class FileSystemWritableFileStream : WritableStream { + fun write(data: BufferDataSource): Promise + fun write(data: FileSystemWritableFileStreamData): Promise + + fun seek(position: Int): Promise } external interface ShowOpenFilePickerOptionsType { diff --git a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/Quest.kt b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/Quest.kt index e9275428..3ae97175 100644 --- a/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/Quest.kt +++ b/lib/src/commonMain/kotlin/world/phantasmal/lib/fileFormats/quest/Quest.kt @@ -240,9 +240,9 @@ private fun extractScriptEntryPoints( } /** - * Creates a .qst file from [quest]. + * Returns a .bin and .dat file in that order. */ -fun writeQuestToQst(quest: Quest, filename: String, version: Version, online: Boolean): Buffer { +fun writeQuestToBinDat(quest: Quest, version: Version): Pair { val dat = writeDat(DatFile( objs = quest.objects.mapTo(mutableListOf()) { DatEntity(it.areaId, it.data) }, npcs = quest.npcs.mapTo(mutableListOf()) { DatEntity(it.areaId, it.data) }, @@ -270,6 +270,15 @@ fun writeQuestToQst(quest: Quest, filename: String, version: Version, online: Bo quest.shopItems, )) + return Pair(bin, dat) +} + +/** + * Creates a .qst file from [quest]. + */ +fun writeQuestToQst(quest: Quest, filename: String, version: Version, online: Boolean): Buffer { + val (bin, dat) = writeQuestToBinDat(quest, version) + val baseFilename = (filenameBase(filename) ?: filename).take(11) val questName = quest.name.take(if (version == Version.BB) 23 else 31) diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/value/ValExtensions.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/value/ValExtensions.kt index f75675ba..0388d48f 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/value/ValExtensions.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/value/ValExtensions.kt @@ -30,6 +30,9 @@ infix fun > Val.lt(value: Val): Val = infix fun Val.and(other: Val): Val = map(this, other) { a, b -> a && b } +infix fun Val.and(other: Boolean): Val = + if (other) this else falseVal() + infix fun Val.or(other: Val): Val = map(this, other) { a, b -> a || b } diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/AbstractListVal.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/AbstractListVal.kt index 8c66f427..55672bd3 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/AbstractListVal.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/AbstractListVal.kt @@ -9,6 +9,7 @@ import world.phantasmal.observable.Observer import world.phantasmal.observable.value.AbstractVal import world.phantasmal.observable.value.DependentVal import world.phantasmal.observable.value.Val +import world.phantasmal.observable.value.not abstract class AbstractListVal( private val extractObservables: ObservablesExtractor?, @@ -24,6 +25,10 @@ abstract class AbstractListVal( */ protected val listObservers = mutableListOf>() + override val empty: Val by lazy { size.map { it == 0 } } + + override val notEmpty: Val by lazy { !empty } + override fun get(index: Int): E = value[index] diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/ListVal.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/ListVal.kt index b2c526f6..5f2e4074 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/ListVal.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/ListVal.kt @@ -12,6 +12,10 @@ interface ListVal : Val> { val size: Val + val empty: Val + + val notEmpty: Val + operator fun get(index: Int): E fun observeList(callNow: Boolean = false, observer: ListValObserver): Disposable diff --git a/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/StaticListVal.kt b/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/StaticListVal.kt index 1619ba59..d8bc887b 100644 --- a/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/StaticListVal.kt +++ b/observable/src/commonMain/kotlin/world/phantasmal/observable/value/list/StaticListVal.kt @@ -4,14 +4,14 @@ import world.phantasmal.core.disposable.Disposable import world.phantasmal.core.disposable.stubDisposable import world.phantasmal.observable.ChangeEvent import world.phantasmal.observable.Observer -import world.phantasmal.observable.value.StaticVal -import world.phantasmal.observable.value.Val -import world.phantasmal.observable.value.value +import world.phantasmal.observable.value.* class StaticListVal(private val elements: List) : ListVal { private val firstOrNull = StaticVal(elements.firstOrNull()) override val size: Val = value(elements.size) + override val empty: Val = if (elements.isEmpty()) trueVal() else falseVal() + override val notEmpty: Val = if (elements.isNotEmpty()) trueVal() else falseVal() override val value: List = elements diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/QuestEditorToolbarController.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/QuestEditorToolbarController.kt index 814ccb50..cac5cccf 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/QuestEditorToolbarController.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/controllers/QuestEditorToolbarController.kt @@ -1,6 +1,7 @@ package world.phantasmal.web.questEditor.controllers import kotlinx.browser.document +import kotlinx.coroutines.await import mu.KotlinLogging import org.w3c.dom.HTMLAnchorElement import org.w3c.dom.url.URL @@ -9,10 +10,9 @@ import world.phantasmal.core.* import world.phantasmal.lib.Endianness import world.phantasmal.lib.Episode import world.phantasmal.lib.fileFormats.quest.* -import world.phantasmal.observable.value.Val -import world.phantasmal.observable.value.map -import world.phantasmal.observable.value.mutableVal -import world.phantasmal.observable.value.value +import world.phantasmal.observable.value.* +import world.phantasmal.observable.value.list.MutableListVal +import world.phantasmal.observable.value.list.mutableListVal import world.phantasmal.web.core.PwToolType import world.phantasmal.web.core.files.cursor import world.phantasmal.web.core.stores.UiStore @@ -21,6 +21,7 @@ import world.phantasmal.web.questEditor.stores.AreaStore import world.phantasmal.web.questEditor.stores.QuestEditorStore import world.phantasmal.web.questEditor.stores.convertQuestFromModel import world.phantasmal.web.questEditor.stores.convertQuestToModel +import world.phantasmal.webui.BrowserFeatures import world.phantasmal.webui.controllers.Controller import world.phantasmal.webui.files.FileHandle import world.phantasmal.webui.files.FileType @@ -36,9 +37,11 @@ class QuestEditorToolbarController( private val areaStore: AreaStore, private val questEditorStore: QuestEditorStore, ) : Controller() { + private val questLoaded = questEditorStore.currentQuest.isNotNull() private val _resultDialogVisible = mutableVal(false) private val _result = mutableVal?>(null) private val _saveAsDialogVisible = mutableVal(false) + private val files: MutableListVal = mutableListVal() private val _filename = mutableVal("") private val _version = mutableVal(Version.BB) @@ -56,7 +59,12 @@ class QuestEditorToolbarController( // Save as - val saveAsEnabled: Val = questEditorStore.currentQuest.isNotNull() + val saveEnabled: Val = questLoaded and files.notEmpty and BrowserFeatures.fileSystemApi + val saveTooltip: Val = value( + if (BrowserFeatures.fileSystemApi) "Save changes (Ctrl-S)" + else "This browser doesn't support saving to an existing file" + ) + val saveAsEnabled: Val = questLoaded val saveAsDialogVisible: Val = _saveAsDialogVisible val filename: Val = _filename val version: Val = _version @@ -109,6 +117,10 @@ class QuestEditorToolbarController( openFiles(showFilePicker(supportedFileTypes, multiple = true)) }, + uiStore.onGlobalKeyDown(PwToolType.QuestEditor, "Ctrl-S") { + save() + }, + uiStore.onGlobalKeyDown(PwToolType.QuestEditor, "Ctrl-Shift-S") { saveAs() }, @@ -133,11 +145,13 @@ class QuestEditorToolbarController( questEditorStore.setDefaultQuest(episode) } - suspend fun openFiles(files: List?) { + suspend fun openFiles(newFiles: List?) { try { - if (files.isNullOrEmpty()) return + files.clear() - val qstFile = files.find { it.extension().equals("qst", ignoreCase = true) } + if (newFiles.isNullOrEmpty()) return + + val qstFile = newFiles.find { it.extension().equals("qst", ignoreCase = true) } if (qstFile != null) { val parseResult = parseQstToQuest(qstFile.cursor(Endianness.Little)) @@ -147,10 +161,11 @@ class QuestEditorToolbarController( setFilename(filenameBase(qstFile.name) ?: qstFile.name) setVersion(parseResult.value.version) setCurrentQuest(parseResult.value.quest) + files.replaceAll(listOf(qstFile)) } } else { - val binFile = files.find { it.extension().equals("bin", ignoreCase = true) } - val datFile = files.find { it.extension().equals("dat", ignoreCase = true) } + val binFile = newFiles.find { it.extension().equals("bin", ignoreCase = true) } + val datFile = newFiles.find { it.extension().equals("dat", ignoreCase = true) } if (binFile == null || datFile == null) { setResult(Failure(listOf(Problem( @@ -170,9 +185,10 @@ class QuestEditorToolbarController( setFilename(binFile.basename() ?: datFile.basename() ?: binFile.name) setVersion(Version.BB) setCurrentQuest(parseResult.value) + files.replaceAll(listOf(binFile, datFile)) } } - } catch (e: Exception) { + } catch (e: Throwable) { setResult( PwResult.build(logger) .addProblem(Severity.Error, "Couldn't parse file.", cause = e) @@ -181,6 +197,48 @@ class QuestEditorToolbarController( } } + suspend fun save() { + if (saveEnabled.value) { + try { + val quest = questEditorStore.currentQuest.value ?: return + val headerFilename = filename.value.trim() + + files.value.find { it.extension().equals("qst", ignoreCase = true) } + ?.let { qstFile -> + val buffer = writeQuestToQst( + convertQuestFromModel(quest), + headerFilename, + version.value, + online = true, + ) + + qstFile.writableStream()!!.use { + it.write(buffer.arrayBuffer).await() + } + } + + val binFile = files.value.find { it.extension().equals("bin", ignoreCase = true) } + val datFile = files.value.find { it.extension().equals("dat", ignoreCase = true) } + + if (binFile != null && datFile != null) { + val (bin, dat) = writeQuestToBinDat( + convertQuestFromModel(quest), + version.value, + ) + + binFile.writableStream()!!.use { it.write(bin.arrayBuffer).await() } + datFile.writableStream()!!.use { it.write(dat.arrayBuffer).await() } + } + } catch (e: Throwable) { + setResult( + PwResult.build(logger) + .addProblem(Severity.Error, "Couldn't save file.", cause = e) + .failure() + ) + } + } + } + fun saveAs() { if (saveAsEnabled.value) { _saveAsDialogVisible.value = true @@ -197,19 +255,18 @@ class QuestEditorToolbarController( fun saveAsDialogSave() { val quest = questEditorStore.currentQuest.value ?: return - var filename = filename.value.trim() + val headerFilename = filename.value.trim() + val filename = + if (headerFilename.endsWith(".qst")) headerFilename + else "$headerFilename.qst" val buffer = writeQuestToQst( convertQuestFromModel(quest), - filename, + headerFilename, version.value, online = true, ) - if (!filename.endsWith(".qst")) { - filename += ".qst" - } - val a = document.createElement("a") as HTMLAnchorElement val url = URL.createObjectURL( Blob( @@ -223,8 +280,12 @@ class QuestEditorToolbarController( a.download = filename document.body?.appendChild(a) a.click() - } catch (e: Exception) { - logger.error(e) { """Couldn't save file "$filename".""" } + } catch (e: Throwable) { + setResult( + PwResult.build(logger) + .addProblem(Severity.Error, "Couldn't save file.", cause = e) + .failure() + ) } finally { URL.revokeObjectURL(url) document.body?.removeChild(a) diff --git a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestEditorToolbarWidget.kt b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestEditorToolbarWidget.kt index 3b9f77b2..d7605d60 100644 --- a/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestEditorToolbarWidget.kt +++ b/web/src/main/kotlin/world/phantasmal/web/questEditor/widgets/QuestEditorToolbarWidget.kt @@ -34,6 +34,13 @@ class QuestEditorToolbarWidget(private val ctrl: QuestEditorToolbarController) : multiple = true, filesSelected = { files -> scope.launch { ctrl.openFiles(files) } }, ), + Button( + text = "Save", + iconLeft = Icon.Save, + enabled = ctrl.saveEnabled, + tooltip = ctrl.saveTooltip, + onClick = { scope.launch { ctrl.save() } }, + ), Button( text = "Save as...", iconLeft = Icon.Save, diff --git a/webui/src/main/kotlin/world/phantasmal/webui/files/Files.kt b/webui/src/main/kotlin/world/phantasmal/webui/files/Files.kt index 3e95986a..a22f7b78 100644 --- a/webui/src/main/kotlin/world/phantasmal/webui/files/Files.kt +++ b/webui/src/main/kotlin/world/phantasmal/webui/files/Files.kt @@ -3,6 +3,7 @@ package world.phantasmal.webui.files import kotlinx.browser.document import kotlinx.browser.window import kotlinx.coroutines.ExperimentalCoroutinesApi +import kotlinx.coroutines.await import kotlinx.coroutines.suspendCancellableCoroutine import org.khronos.webgl.ArrayBuffer import org.w3c.dom.HTMLInputElement @@ -10,15 +11,13 @@ import org.w3c.dom.asList import org.w3c.dom.events.Event import org.w3c.files.File import world.phantasmal.core.disposable.Disposable +import world.phantasmal.core.externals.browser.* import world.phantasmal.core.filenameBase import world.phantasmal.core.filenameExtension import world.phantasmal.webui.BrowserFeatures import world.phantasmal.webui.dom.disposableListener -import world.phantasmal.webui.externals.browser.FileSystemFileHandle -import world.phantasmal.webui.externals.browser.ShowOpenFilePickerOptionsType -import world.phantasmal.webui.externals.browser.arrayBuffer -import world.phantasmal.webui.externals.browser.showOpenFilePicker import world.phantasmal.webui.obj +import kotlin.coroutines.resume import kotlin.js.Promise @OptIn(ExperimentalCoroutinesApi::class) @@ -45,6 +44,12 @@ class FileHandle private constructor( */ fun extension(): String? = filenameExtension(name) + /** + * Returns a writable stream if this [FileHandle] represents a [FileSystemFileHandle]. + */ + suspend fun writableStream(): FileSystemWritableFileStream? = + handle?.createWritable()?.await() + suspend fun arrayBuffer(): ArrayBuffer = suspendCancellableCoroutine { cont -> getFile() .then { it.arrayBuffer() } @@ -65,9 +70,9 @@ class FileType( @OptIn(ExperimentalCoroutinesApi::class) suspend fun showFilePicker(types: List, multiple: Boolean = false): List? = - suspendCancellableCoroutine { cont -> - if (BrowserFeatures.fileSystemApi) { - window.showOpenFilePicker(obj { + if (BrowserFeatures.fileSystemApi) { + try { + val fileHandles = window.showOpenFilePicker(obj { this.multiple = multiple this.types = types.map { obj { @@ -79,17 +84,29 @@ suspend fun showFilePicker(types: List, multiple: Boolean = false): Li } } }.toTypedArray() - }).then({ cont.resume(it.map(::FileHandle)) {} }, { cont.resume(null) {} }) - } else { + }).await() + + fileHandles.map(::FileHandle) + } catch (e: Throwable) { + // Ensure we return null when the user cancels. + if (e.asDynamic().name == "AbortError") { + null + } else { + throw e + } + } + } else { + suspendCancellableCoroutine { cont -> val el = document.createElement("input") as HTMLInputElement el.type = "file" el.accept = types.flatMap { it.accept.values.flatten() }.joinToString() el.multiple = multiple el.onchange = { - cont.resume(el.files?.asList()?.map(::FileHandle) ?: emptyList()) {} + cont.resume(el.files!!.asList().map(::FileHandle)) } + // Ensure we return null when the user cancels. @Suppress("JoinDeclarationAndAssignment") lateinit var focusListener: Disposable @@ -98,7 +115,7 @@ suspend fun showFilePicker(types: List, multiple: Boolean = false): Li window.setTimeout({ if (cont.isActive) { - cont.resume(null) {} + cont.resume(null) } }, 500) })