mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 15:28:29 +08:00
The file system access API is now used for opening files when supported by the browser.
This commit is contained in:
parent
d6751b0151
commit
329e067a17
@ -0,0 +1,9 @@
|
|||||||
|
package world.phantasmal.web.core.files
|
||||||
|
|
||||||
|
import world.phantasmal.lib.Endianness
|
||||||
|
import world.phantasmal.lib.cursor.Cursor
|
||||||
|
import world.phantasmal.lib.cursor.cursor
|
||||||
|
import world.phantasmal.webui.files.FileHandle
|
||||||
|
|
||||||
|
suspend fun FileHandle.cursor(endianness: Endianness): Cursor =
|
||||||
|
arrayBuffer().cursor(endianness)
|
@ -5,17 +5,16 @@ import mu.KotlinLogging
|
|||||||
import org.w3c.dom.HTMLAnchorElement
|
import org.w3c.dom.HTMLAnchorElement
|
||||||
import org.w3c.dom.url.URL
|
import org.w3c.dom.url.URL
|
||||||
import org.w3c.files.Blob
|
import org.w3c.files.Blob
|
||||||
import org.w3c.files.File
|
|
||||||
import world.phantasmal.core.*
|
import world.phantasmal.core.*
|
||||||
import world.phantasmal.lib.Endianness
|
import world.phantasmal.lib.Endianness
|
||||||
import world.phantasmal.lib.Episode
|
import world.phantasmal.lib.Episode
|
||||||
import world.phantasmal.lib.cursor.cursor
|
|
||||||
import world.phantasmal.lib.fileFormats.quest.*
|
import world.phantasmal.lib.fileFormats.quest.*
|
||||||
import world.phantasmal.observable.value.Val
|
import world.phantasmal.observable.value.Val
|
||||||
import world.phantasmal.observable.value.map
|
import world.phantasmal.observable.value.map
|
||||||
import world.phantasmal.observable.value.mutableVal
|
import world.phantasmal.observable.value.mutableVal
|
||||||
import world.phantasmal.observable.value.value
|
import world.phantasmal.observable.value.value
|
||||||
import world.phantasmal.web.core.PwToolType
|
import world.phantasmal.web.core.PwToolType
|
||||||
|
import world.phantasmal.web.core.files.cursor
|
||||||
import world.phantasmal.web.core.stores.UiStore
|
import world.phantasmal.web.core.stores.UiStore
|
||||||
import world.phantasmal.web.questEditor.models.AreaModel
|
import world.phantasmal.web.questEditor.models.AreaModel
|
||||||
import world.phantasmal.web.questEditor.stores.AreaStore
|
import world.phantasmal.web.questEditor.stores.AreaStore
|
||||||
@ -23,9 +22,10 @@ import world.phantasmal.web.questEditor.stores.QuestEditorStore
|
|||||||
import world.phantasmal.web.questEditor.stores.convertQuestFromModel
|
import world.phantasmal.web.questEditor.stores.convertQuestFromModel
|
||||||
import world.phantasmal.web.questEditor.stores.convertQuestToModel
|
import world.phantasmal.web.questEditor.stores.convertQuestToModel
|
||||||
import world.phantasmal.webui.controllers.Controller
|
import world.phantasmal.webui.controllers.Controller
|
||||||
|
import world.phantasmal.webui.files.FileHandle
|
||||||
|
import world.phantasmal.webui.files.FileType
|
||||||
|
import world.phantasmal.webui.files.showFilePicker
|
||||||
import world.phantasmal.webui.obj
|
import world.phantasmal.webui.obj
|
||||||
import world.phantasmal.webui.readFile
|
|
||||||
import world.phantasmal.webui.selectFiles
|
|
||||||
|
|
||||||
private val logger = KotlinLogging.logger {}
|
private val logger = KotlinLogging.logger {}
|
||||||
|
|
||||||
@ -47,7 +47,12 @@ class QuestEditorToolbarController(
|
|||||||
val resultDialogVisible: Val<Boolean> = _resultDialogVisible
|
val resultDialogVisible: Val<Boolean> = _resultDialogVisible
|
||||||
val result: Val<PwResult<*>?> = _result
|
val result: Val<PwResult<*>?> = _result
|
||||||
|
|
||||||
val openFileAccept = ".bin, .dat, .qst"
|
val supportedFileTypes = listOf(
|
||||||
|
FileType(
|
||||||
|
description = "Quests",
|
||||||
|
accept = mapOf("application/pw-quest" to setOf(".qst", ".bin", ".dat")),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
// Save as
|
// Save as
|
||||||
|
|
||||||
@ -101,7 +106,7 @@ class QuestEditorToolbarController(
|
|||||||
init {
|
init {
|
||||||
addDisposables(
|
addDisposables(
|
||||||
uiStore.onGlobalKeyDown(PwToolType.QuestEditor, "Ctrl-O") {
|
uiStore.onGlobalKeyDown(PwToolType.QuestEditor, "Ctrl-O") {
|
||||||
openFiles(selectFiles(accept = openFileAccept, multiple = true))
|
openFiles(showFilePicker(supportedFileTypes, multiple = true))
|
||||||
},
|
},
|
||||||
|
|
||||||
uiStore.onGlobalKeyDown(PwToolType.QuestEditor, "Ctrl-Shift-S") {
|
uiStore.onGlobalKeyDown(PwToolType.QuestEditor, "Ctrl-Shift-S") {
|
||||||
@ -128,26 +133,26 @@ class QuestEditorToolbarController(
|
|||||||
questEditorStore.setDefaultQuest(episode)
|
questEditorStore.setDefaultQuest(episode)
|
||||||
}
|
}
|
||||||
|
|
||||||
suspend fun openFiles(files: List<File>) {
|
suspend fun openFiles(files: List<FileHandle>?) {
|
||||||
try {
|
try {
|
||||||
if (files.isEmpty()) return
|
if (files.isNullOrEmpty()) return
|
||||||
|
|
||||||
val qst = files.find { it.name.endsWith(".qst", ignoreCase = true) }
|
val qstFile = files.find { it.extension().equals("qst", ignoreCase = true) }
|
||||||
|
|
||||||
if (qst != null) {
|
if (qstFile != null) {
|
||||||
val parseResult = parseQstToQuest(readFile(qst).cursor(Endianness.Little))
|
val parseResult = parseQstToQuest(qstFile.cursor(Endianness.Little))
|
||||||
setResult(parseResult)
|
setResult(parseResult)
|
||||||
|
|
||||||
if (parseResult is Success) {
|
if (parseResult is Success) {
|
||||||
setFilename(filenameBase(qst.name) ?: qst.name)
|
setFilename(filenameBase(qstFile.name) ?: qstFile.name)
|
||||||
setVersion(parseResult.value.version)
|
setVersion(parseResult.value.version)
|
||||||
setCurrentQuest(parseResult.value.quest)
|
setCurrentQuest(parseResult.value.quest)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
val bin = files.find { it.name.endsWith(".bin", ignoreCase = true) }
|
val binFile = files.find { it.extension().equals("bin", ignoreCase = true) }
|
||||||
val dat = files.find { it.name.endsWith(".dat", ignoreCase = true) }
|
val datFile = files.find { it.extension().equals("dat", ignoreCase = true) }
|
||||||
|
|
||||||
if (bin == null || dat == null) {
|
if (binFile == null || datFile == null) {
|
||||||
setResult(Failure(listOf(Problem(
|
setResult(Failure(listOf(Problem(
|
||||||
Severity.Error,
|
Severity.Error,
|
||||||
"Please select a .qst file or one .bin and one .dat file."
|
"Please select a .qst file or one .bin and one .dat file."
|
||||||
@ -156,13 +161,13 @@ class QuestEditorToolbarController(
|
|||||||
}
|
}
|
||||||
|
|
||||||
val parseResult = parseBinDatToQuest(
|
val parseResult = parseBinDatToQuest(
|
||||||
readFile(bin).cursor(Endianness.Little),
|
binFile.cursor(Endianness.Little),
|
||||||
readFile(dat).cursor(Endianness.Little),
|
datFile.cursor(Endianness.Little),
|
||||||
)
|
)
|
||||||
setResult(parseResult)
|
setResult(parseResult)
|
||||||
|
|
||||||
if (parseResult is Success) {
|
if (parseResult is Success) {
|
||||||
setFilename(filenameBase(bin.name) ?: filenameBase(dat.name) ?: bin.name)
|
setFilename(binFile.basename() ?: datFile.basename() ?: binFile.name)
|
||||||
setVersion(Version.BB)
|
setVersion(Version.BB)
|
||||||
setCurrentQuest(parseResult.value)
|
setCurrentQuest(parseResult.value)
|
||||||
}
|
}
|
||||||
|
@ -30,7 +30,7 @@ class QuestEditorToolbarWidget(private val ctrl: QuestEditorToolbarController) :
|
|||||||
text = "Open file...",
|
text = "Open file...",
|
||||||
tooltip = value("Open a quest file (Ctrl-O)"),
|
tooltip = value("Open a quest file (Ctrl-O)"),
|
||||||
iconLeft = Icon.File,
|
iconLeft = Icon.File,
|
||||||
accept = ctrl.openFileAccept,
|
types = ctrl.supportedFileTypes,
|
||||||
multiple = true,
|
multiple = true,
|
||||||
filesSelected = { files -> scope.launch { ctrl.openFiles(files) } },
|
filesSelected = { files -> scope.launch { ctrl.openFiles(files) } },
|
||||||
),
|
),
|
||||||
|
@ -16,7 +16,7 @@ import world.phantasmal.web.viewer.rendering.MeshRenderer
|
|||||||
import world.phantasmal.web.viewer.rendering.TextureRenderer
|
import world.phantasmal.web.viewer.rendering.TextureRenderer
|
||||||
import world.phantasmal.web.viewer.stores.ViewerStore
|
import world.phantasmal.web.viewer.stores.ViewerStore
|
||||||
import world.phantasmal.web.viewer.widgets.CharacterClassOptionsWidget
|
import world.phantasmal.web.viewer.widgets.CharacterClassOptionsWidget
|
||||||
import world.phantasmal.web.viewer.widgets.ViewerToolbar
|
import world.phantasmal.web.viewer.widgets.ViewerToolbarWidget
|
||||||
import world.phantasmal.web.viewer.widgets.ViewerWidget
|
import world.phantasmal.web.viewer.widgets.ViewerWidget
|
||||||
import world.phantasmal.webui.DisposableContainer
|
import world.phantasmal.webui.DisposableContainer
|
||||||
import world.phantasmal.webui.widgets.Widget
|
import world.phantasmal.webui.widgets.Widget
|
||||||
@ -54,7 +54,7 @@ class Viewer(
|
|||||||
// Main Widget
|
// Main Widget
|
||||||
return ViewerWidget(
|
return ViewerWidget(
|
||||||
viewerController,
|
viewerController,
|
||||||
{ ViewerToolbar(viewerToolbarController) },
|
{ ViewerToolbarWidget(viewerToolbarController) },
|
||||||
{ CharacterClassOptionsWidget(characterClassOptionsController) },
|
{ CharacterClassOptionsWidget(characterClassOptionsController) },
|
||||||
{ RendererWidget(meshRenderer) },
|
{ RendererWidget(meshRenderer) },
|
||||||
{ RendererWidget(textureRenderer) },
|
{ RendererWidget(textureRenderer) },
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
package world.phantasmal.web.viewer.controllers
|
package world.phantasmal.web.viewer.controllers
|
||||||
|
|
||||||
import mu.KotlinLogging
|
import mu.KotlinLogging
|
||||||
import org.w3c.files.File
|
|
||||||
import world.phantasmal.core.Failure
|
import world.phantasmal.core.Failure
|
||||||
import world.phantasmal.core.PwResult
|
import world.phantasmal.core.PwResult
|
||||||
import world.phantasmal.core.Severity
|
import world.phantasmal.core.Severity
|
||||||
@ -16,11 +15,11 @@ import world.phantasmal.lib.fileFormats.parseAreaCollisionGeometry
|
|||||||
import world.phantasmal.lib.fileFormats.parseAreaRenderGeometry
|
import world.phantasmal.lib.fileFormats.parseAreaRenderGeometry
|
||||||
import world.phantasmal.observable.value.Val
|
import world.phantasmal.observable.value.Val
|
||||||
import world.phantasmal.observable.value.mutableVal
|
import world.phantasmal.observable.value.mutableVal
|
||||||
|
import world.phantasmal.web.core.files.cursor
|
||||||
import world.phantasmal.web.viewer.stores.NinjaGeometry
|
import world.phantasmal.web.viewer.stores.NinjaGeometry
|
||||||
import world.phantasmal.web.viewer.stores.ViewerStore
|
import world.phantasmal.web.viewer.stores.ViewerStore
|
||||||
import world.phantasmal.webui.controllers.Controller
|
import world.phantasmal.webui.controllers.Controller
|
||||||
import world.phantasmal.webui.extension
|
import world.phantasmal.webui.files.FileHandle
|
||||||
import world.phantasmal.webui.readFile
|
|
||||||
|
|
||||||
private val logger = KotlinLogging.logger {}
|
private val logger = KotlinLogging.logger {}
|
||||||
|
|
||||||
@ -70,7 +69,9 @@ class ViewerToolbarController(private val store: ViewerStore) : Controller() {
|
|||||||
store.setCurrentAnimation(null)
|
store.setCurrentAnimation(null)
|
||||||
}
|
}
|
||||||
|
|
||||||
suspend fun openFiles(files: List<File>) {
|
suspend fun openFiles(files: List<FileHandle>?) {
|
||||||
|
files ?: return
|
||||||
|
|
||||||
val result = PwResult.build<Unit>(logger)
|
val result = PwResult.build<Unit>(logger)
|
||||||
var success = false
|
var success = false
|
||||||
|
|
||||||
@ -82,7 +83,7 @@ class ViewerToolbarController(private val store: ViewerStore) : Controller() {
|
|||||||
for (file in files) {
|
for (file in files) {
|
||||||
val extension = file.extension()?.toLowerCase()
|
val extension = file.extension()?.toLowerCase()
|
||||||
|
|
||||||
val cursor = readFile(file).cursor(Endianness.Little)
|
val cursor = file.cursor(Endianness.Little)
|
||||||
var fileResult: PwResult<*>
|
var fileResult: PwResult<*>
|
||||||
|
|
||||||
when (extension) {
|
when (extension) {
|
||||||
@ -105,6 +106,7 @@ class ViewerToolbarController(private val store: ViewerStore) : Controller() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
"rel" -> {
|
"rel" -> {
|
||||||
|
// TODO: Detect .rel type instead of relying on filename.
|
||||||
if (file.name.endsWith("c.rel")) {
|
if (file.name.endsWith("c.rel")) {
|
||||||
val collisionGeometry = parseAreaCollisionGeometry(cursor)
|
val collisionGeometry = parseAreaCollisionGeometry(cursor)
|
||||||
fileResult = Success(collisionGeometry)
|
fileResult = Success(collisionGeometry)
|
||||||
|
@ -5,9 +5,10 @@ import org.w3c.dom.Node
|
|||||||
import world.phantasmal.web.viewer.controllers.ViewerToolbarController
|
import world.phantasmal.web.viewer.controllers.ViewerToolbarController
|
||||||
import world.phantasmal.webui.dom.Icon
|
import world.phantasmal.webui.dom.Icon
|
||||||
import world.phantasmal.webui.dom.div
|
import world.phantasmal.webui.dom.div
|
||||||
|
import world.phantasmal.webui.files.FileType
|
||||||
import world.phantasmal.webui.widgets.*
|
import world.phantasmal.webui.widgets.*
|
||||||
|
|
||||||
class ViewerToolbar(private val ctrl: ViewerToolbarController) : Widget() {
|
class ViewerToolbarWidget(private val ctrl: ViewerToolbarController) : Widget() {
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
div {
|
div {
|
||||||
className = "pw-viewer-toolbar"
|
className = "pw-viewer-toolbar"
|
||||||
@ -17,7 +18,16 @@ class ViewerToolbar(private val ctrl: ViewerToolbarController) : Widget() {
|
|||||||
FileButton(
|
FileButton(
|
||||||
text = "Open file...",
|
text = "Open file...",
|
||||||
iconLeft = Icon.File,
|
iconLeft = Icon.File,
|
||||||
accept = ".afs, .nj, .njm, .rel, .xj, .xvm",
|
types = listOf(
|
||||||
|
FileType(
|
||||||
|
description = "Models, textures, animations",
|
||||||
|
accept = mapOf(
|
||||||
|
"application/pw-viewer-file" to setOf(
|
||||||
|
".afs", ".nj", ".njm", ".rel", ".xj", ".xvm"
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
multiple = true,
|
multiple = true,
|
||||||
filesSelected = { files -> scope.launch { ctrl.openFiles(files) } },
|
filesSelected = { files -> scope.launch { ctrl.openFiles(files) } },
|
||||||
),
|
),
|
@ -9,6 +9,7 @@ import world.phantasmal.web.core.actions.Action
|
|||||||
import world.phantasmal.web.test.WebTestSuite
|
import world.phantasmal.web.test.WebTestSuite
|
||||||
import world.phantasmal.web.test.createQuestModel
|
import world.phantasmal.web.test.createQuestModel
|
||||||
import world.phantasmal.web.test.createQuestNpcModel
|
import world.phantasmal.web.test.createQuestNpcModel
|
||||||
|
import world.phantasmal.webui.files.FileHandle
|
||||||
import kotlin.test.*
|
import kotlin.test.*
|
||||||
|
|
||||||
class QuestEditorToolbarControllerTests : WebTestSuite {
|
class QuestEditorToolbarControllerTests : WebTestSuite {
|
||||||
@ -35,7 +36,7 @@ class QuestEditorToolbarControllerTests : WebTestSuite {
|
|||||||
|
|
||||||
assertNull(ctrl.result.value)
|
assertNull(ctrl.result.value)
|
||||||
|
|
||||||
ctrl.openFiles(listOf(File(arrayOf(), "unknown.extension")))
|
ctrl.openFiles(listOf(FileHandle(File(arrayOf(), "unknown.extension"))))
|
||||||
|
|
||||||
val result = ctrl.result.value
|
val result = ctrl.result.value
|
||||||
|
|
||||||
|
@ -1,38 +0,0 @@
|
|||||||
package world.phantasmal.webui
|
|
||||||
|
|
||||||
import kotlinx.browser.document
|
|
||||||
import kotlinx.coroutines.ExperimentalCoroutinesApi
|
|
||||||
import kotlinx.coroutines.suspendCancellableCoroutine
|
|
||||||
import org.khronos.webgl.ArrayBuffer
|
|
||||||
import org.w3c.dom.HTMLInputElement
|
|
||||||
import org.w3c.dom.asList
|
|
||||||
import org.w3c.files.File
|
|
||||||
import org.w3c.files.FileReader
|
|
||||||
|
|
||||||
@OptIn(ExperimentalCoroutinesApi::class)
|
|
||||||
suspend fun selectFiles(accept: String = "", multiple: Boolean = false): List<File> =
|
|
||||||
suspendCancellableCoroutine { cont ->
|
|
||||||
val el = document.createElement("input") as HTMLInputElement
|
|
||||||
el.type = "file"
|
|
||||||
el.accept = accept
|
|
||||||
el.multiple = multiple
|
|
||||||
|
|
||||||
el.onchange = {
|
|
||||||
cont.resume(el.files?.asList() ?: emptyList()) {}
|
|
||||||
}
|
|
||||||
|
|
||||||
el.click()
|
|
||||||
}
|
|
||||||
|
|
||||||
@OptIn(ExperimentalCoroutinesApi::class)
|
|
||||||
suspend fun readFile(file: File): ArrayBuffer = suspendCancellableCoroutine { cont ->
|
|
||||||
val reader = FileReader()
|
|
||||||
reader.onloadend = {
|
|
||||||
if (reader.result is ArrayBuffer) {
|
|
||||||
cont.resume(reader.result.unsafeCast<ArrayBuffer>()) {}
|
|
||||||
} else {
|
|
||||||
cont.cancel(Exception(reader.error.message.unsafeCast<String>()))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
reader.readAsArrayBuffer(file)
|
|
||||||
}
|
|
@ -1,8 +1,13 @@
|
|||||||
package world.phantasmal.webui
|
package world.phantasmal.webui
|
||||||
|
|
||||||
|
import kotlinx.browser.window
|
||||||
import org.w3c.files.File
|
import org.w3c.files.File
|
||||||
import world.phantasmal.core.filenameExtension
|
import world.phantasmal.core.filenameExtension
|
||||||
|
|
||||||
|
object BrowserFeatures {
|
||||||
|
val fileSystemApi: Boolean = window.asDynamic().showOpenFilePicker != null
|
||||||
|
}
|
||||||
|
|
||||||
inline fun <T> obj(block: T.() -> Unit): T =
|
inline fun <T> obj(block: T.() -> Unit): T =
|
||||||
js("{}").unsafeCast<T>().apply(block)
|
js("{}").unsafeCast<T>().apply(block)
|
||||||
|
|
||||||
|
35
webui/src/main/kotlin/world/phantasmal/webui/externals/browser/Browser.kt
vendored
Normal file
35
webui/src/main/kotlin/world/phantasmal/webui/externals/browser/Browser.kt
vendored
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
package world.phantasmal.webui.externals.browser
|
||||||
|
|
||||||
|
import org.khronos.webgl.ArrayBuffer
|
||||||
|
import org.w3c.dom.Window
|
||||||
|
import org.w3c.files.Blob
|
||||||
|
import org.w3c.files.File
|
||||||
|
import kotlin.js.Promise
|
||||||
|
|
||||||
|
fun Blob.arrayBuffer(): Promise<ArrayBuffer> =
|
||||||
|
asDynamic().arrayBuffer().unsafeCast<Promise<ArrayBuffer>>()
|
||||||
|
|
||||||
|
open external class FileSystemHandle {
|
||||||
|
val kind: String /* "file" | "directory" */
|
||||||
|
val name: String
|
||||||
|
}
|
||||||
|
|
||||||
|
external class FileSystemFileHandle : FileSystemHandle {
|
||||||
|
fun getFile(): Promise<File>
|
||||||
|
}
|
||||||
|
|
||||||
|
external interface ShowOpenFilePickerOptionsType {
|
||||||
|
var description: String
|
||||||
|
var accept: dynamic
|
||||||
|
}
|
||||||
|
|
||||||
|
external interface ShowOpenFilePickerOptions {
|
||||||
|
var multiple: Boolean
|
||||||
|
var excludeAcceptAllOption: Boolean
|
||||||
|
var types: Array<ShowOpenFilePickerOptionsType>
|
||||||
|
}
|
||||||
|
|
||||||
|
fun Window.showOpenFilePicker(
|
||||||
|
options: ShowOpenFilePickerOptions,
|
||||||
|
): Promise<Array<FileSystemFileHandle>> =
|
||||||
|
asDynamic().showOpenFilePicker(options).unsafeCast<Promise<Array<FileSystemFileHandle>>>()
|
108
webui/src/main/kotlin/world/phantasmal/webui/files/Files.kt
Normal file
108
webui/src/main/kotlin/world/phantasmal/webui/files/Files.kt
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
package world.phantasmal.webui.files
|
||||||
|
|
||||||
|
import kotlinx.browser.document
|
||||||
|
import kotlinx.browser.window
|
||||||
|
import kotlinx.coroutines.ExperimentalCoroutinesApi
|
||||||
|
import kotlinx.coroutines.suspendCancellableCoroutine
|
||||||
|
import org.khronos.webgl.ArrayBuffer
|
||||||
|
import org.w3c.dom.HTMLInputElement
|
||||||
|
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.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.js.Promise
|
||||||
|
|
||||||
|
@OptIn(ExperimentalCoroutinesApi::class)
|
||||||
|
class FileHandle private constructor(
|
||||||
|
private val handle: FileSystemFileHandle?,
|
||||||
|
private val file: File?,
|
||||||
|
) {
|
||||||
|
constructor(file: File) : this(null, file)
|
||||||
|
constructor(handle: FileSystemFileHandle) : this(handle, null)
|
||||||
|
|
||||||
|
val name: String = handle?.name ?: file!!.name
|
||||||
|
|
||||||
|
init {
|
||||||
|
require((handle == null) xor (file == null))
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the filename without extension if there is one.
|
||||||
|
*/
|
||||||
|
fun basename(): String? = filenameBase(name)
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the filename extension if there is one.
|
||||||
|
*/
|
||||||
|
fun extension(): String? = filenameExtension(name)
|
||||||
|
|
||||||
|
suspend fun arrayBuffer(): ArrayBuffer = suspendCancellableCoroutine { cont ->
|
||||||
|
getFile()
|
||||||
|
.then { it.arrayBuffer() }
|
||||||
|
.then({ cont.resume(it) {} }, cont::cancel)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getFile(): Promise<File> =
|
||||||
|
handle?.getFile() ?: Promise.resolve(file!!)
|
||||||
|
}
|
||||||
|
|
||||||
|
class FileType(
|
||||||
|
val description: String,
|
||||||
|
/**
|
||||||
|
* Map of MIME types to file extensions
|
||||||
|
*/
|
||||||
|
val accept: Map<String, Set<String>>,
|
||||||
|
)
|
||||||
|
|
||||||
|
@OptIn(ExperimentalCoroutinesApi::class)
|
||||||
|
suspend fun showFilePicker(types: List<FileType>, multiple: Boolean = false): List<FileHandle>? =
|
||||||
|
suspendCancellableCoroutine { cont ->
|
||||||
|
if (BrowserFeatures.fileSystemApi) {
|
||||||
|
window.showOpenFilePicker(obj {
|
||||||
|
this.multiple = multiple
|
||||||
|
this.types = types.map {
|
||||||
|
obj<ShowOpenFilePickerOptionsType> {
|
||||||
|
description = it.description
|
||||||
|
accept = obj {
|
||||||
|
for ((mimeType, extensions) in it.accept) {
|
||||||
|
this[mimeType] = extensions.toTypedArray()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}.toTypedArray()
|
||||||
|
}).then({ cont.resume(it.map(::FileHandle)) {} }, { cont.resume(null) {} })
|
||||||
|
} else {
|
||||||
|
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()) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Suppress("JoinDeclarationAndAssignment")
|
||||||
|
lateinit var focusListener: Disposable
|
||||||
|
|
||||||
|
focusListener = window.disposableListener<Event>("focus", {
|
||||||
|
focusListener.dispose()
|
||||||
|
|
||||||
|
window.setTimeout({
|
||||||
|
if (cont.isActive) {
|
||||||
|
cont.resume(null) {}
|
||||||
|
}
|
||||||
|
}, 500)
|
||||||
|
})
|
||||||
|
|
||||||
|
el.click()
|
||||||
|
}
|
||||||
|
}
|
@ -16,6 +16,7 @@ import world.phantasmal.webui.dom.dom
|
|||||||
import world.phantasmal.webui.dom.h1
|
import world.phantasmal.webui.dom.h1
|
||||||
import world.phantasmal.webui.dom.section
|
import world.phantasmal.webui.dom.section
|
||||||
|
|
||||||
|
// TODO: Use HTML dialog element.
|
||||||
open class Dialog(
|
open class Dialog(
|
||||||
visible: Val<Boolean> = trueVal(),
|
visible: Val<Boolean> = trueVal(),
|
||||||
enabled: Val<Boolean> = trueVal(),
|
enabled: Val<Boolean> = trueVal(),
|
||||||
|
@ -2,12 +2,13 @@ package world.phantasmal.webui.widgets
|
|||||||
|
|
||||||
import kotlinx.coroutines.launch
|
import kotlinx.coroutines.launch
|
||||||
import org.w3c.dom.HTMLElement
|
import org.w3c.dom.HTMLElement
|
||||||
import org.w3c.files.File
|
|
||||||
import world.phantasmal.observable.value.Val
|
import world.phantasmal.observable.value.Val
|
||||||
import world.phantasmal.observable.value.nullVal
|
import world.phantasmal.observable.value.nullVal
|
||||||
import world.phantasmal.observable.value.trueVal
|
import world.phantasmal.observable.value.trueVal
|
||||||
import world.phantasmal.webui.dom.Icon
|
import world.phantasmal.webui.dom.Icon
|
||||||
import world.phantasmal.webui.selectFiles
|
import world.phantasmal.webui.files.FileHandle
|
||||||
|
import world.phantasmal.webui.files.FileType
|
||||||
|
import world.phantasmal.webui.files.showFilePicker
|
||||||
|
|
||||||
class FileButton(
|
class FileButton(
|
||||||
visible: Val<Boolean> = trueVal(),
|
visible: Val<Boolean> = trueVal(),
|
||||||
@ -18,9 +19,9 @@ class FileButton(
|
|||||||
textVal: Val<String>? = null,
|
textVal: Val<String>? = null,
|
||||||
iconLeft: Icon? = null,
|
iconLeft: Icon? = null,
|
||||||
iconRight: Icon? = null,
|
iconRight: Icon? = null,
|
||||||
private val accept: String = "",
|
private val types: List<FileType> = emptyList(),
|
||||||
private val multiple: Boolean = false,
|
private val multiple: Boolean = false,
|
||||||
private val filesSelected: ((List<File>) -> Unit)? = null,
|
private val filesSelected: ((List<FileHandle>?) -> Unit)? = null,
|
||||||
) : Button(visible, enabled, tooltip, className, text, textVal, iconLeft, iconRight) {
|
) : Button(visible, enabled, tooltip, className, text, textVal, iconLeft, iconRight) {
|
||||||
override fun interceptElement(element: HTMLElement) {
|
override fun interceptElement(element: HTMLElement) {
|
||||||
element.classList.add("pw-file-button")
|
element.classList.add("pw-file-button")
|
||||||
@ -28,7 +29,7 @@ class FileButton(
|
|||||||
if (filesSelected != null) {
|
if (filesSelected != null) {
|
||||||
element.onclick = {
|
element.onclick = {
|
||||||
scope.launch {
|
scope.launch {
|
||||||
filesSelected.invoke(selectFiles(accept, multiple))
|
filesSelected.invoke(showFilePicker(types, multiple))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user