mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-04 06:28:28 +08:00
Improved LoadingStatusCell and its usage in Table, TableController and HuntMethodStore. Added a unit test for MethodsForEpisodeController.
This commit is contained in:
parent
9807418435
commit
f10a4ebe6c
@ -3,6 +3,11 @@ package world.phantasmal.web.core.models
|
|||||||
/**
|
/**
|
||||||
* Represents a PSO private server.
|
* Represents a PSO private server.
|
||||||
*/
|
*/
|
||||||
enum class Server(val uiName: String, val slug: String) {
|
enum class Server(
|
||||||
Ephinea("Ephinea", "ephinea")
|
/** Display name shown to the user. */
|
||||||
|
val uiName: String,
|
||||||
|
/** Used in URLs, do not change these. */
|
||||||
|
val slug: String,
|
||||||
|
) {
|
||||||
|
Ephinea(uiName = "Ephinea", slug = "ephinea")
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
package world.phantasmal.web.core.persistence
|
package world.phantasmal.web.core.persistence
|
||||||
|
|
||||||
|
import kotlinx.coroutines.Dispatchers
|
||||||
|
import kotlinx.coroutines.withContext
|
||||||
import kotlinx.serialization.KSerializer
|
import kotlinx.serialization.KSerializer
|
||||||
import kotlinx.serialization.json.Json
|
import kotlinx.serialization.json.Json
|
||||||
import kotlinx.serialization.serializer
|
import kotlinx.serialization.serializer
|
||||||
@ -21,10 +23,12 @@ abstract class Persister(private val store: KeyValueStore) {
|
|||||||
// Method suspends so we can use async storage in the future.
|
// Method suspends so we can use async storage in the future.
|
||||||
@Suppress("RedundantSuspendModifier")
|
@Suppress("RedundantSuspendModifier")
|
||||||
protected suspend fun <T> persist(key: String, data: T, serializer: KSerializer<T>) {
|
protected suspend fun <T> persist(key: String, data: T, serializer: KSerializer<T>) {
|
||||||
try {
|
withContext(Dispatchers.Default) {
|
||||||
store.put(key, format.encodeToString(serializer, data))
|
try {
|
||||||
} catch (e: Throwable) {
|
store.put(key, format.encodeToString(serializer, data))
|
||||||
logger.error(e) { "Couldn't persist ${key}." }
|
} catch (e: Throwable) {
|
||||||
|
logger.error(e) { "Couldn't persist ${key}." }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -42,12 +46,14 @@ abstract class Persister(private val store: KeyValueStore) {
|
|||||||
// Method suspends so we can use async storage in the future.
|
// Method suspends so we can use async storage in the future.
|
||||||
@Suppress("RedundantSuspendModifier")
|
@Suppress("RedundantSuspendModifier")
|
||||||
protected suspend fun <T> load(key: String, serializer: KSerializer<T>): T? =
|
protected suspend fun <T> load(key: String, serializer: KSerializer<T>): T? =
|
||||||
try {
|
withContext(Dispatchers.Default) {
|
||||||
val json = store.get(key)
|
try {
|
||||||
json?.let { format.decodeFromString(serializer, it) }
|
val json = store.get(key)
|
||||||
} catch (e: Throwable) {
|
json?.let { format.decodeFromString(serializer, it) }
|
||||||
logger.error(e) { "Couldn't load ${key}." }
|
} catch (e: Throwable) {
|
||||||
null
|
logger.error(e) { "Couldn't load ${key}." }
|
||||||
|
null
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected suspend inline fun <reified T> loadForServer(server: Server, key: String): T? =
|
protected suspend inline fun <reified T> loadForServer(server: Server, key: String): T? =
|
||||||
|
@ -21,11 +21,47 @@ class MethodsForEpisodeController(
|
|||||||
private val methods = mutableListCell<HuntMethodModel>()
|
private val methods = mutableListCell<HuntMethodModel>()
|
||||||
private val enemies: List<NpcType> = NpcType.VALUES.filter { it.enemy && it.episode == episode }
|
private val enemies: List<NpcType> = NpcType.VALUES.filter { it.enemy && it.episode == episode }
|
||||||
|
|
||||||
|
private var sortColumns: List<SortColumn<HuntMethodModel>> = emptyList()
|
||||||
|
|
||||||
|
private val comparator: Comparator<HuntMethodModel> =
|
||||||
|
Comparator { a, b ->
|
||||||
|
for (sortColumn in sortColumns) {
|
||||||
|
val cmp = when (sortColumn.column.key) {
|
||||||
|
METHOD_COL_KEY ->
|
||||||
|
a.name.asDynamic().localeCompare(b.name).unsafeCast<Int>()
|
||||||
|
|
||||||
|
TIME_COL_KEY -> a.time.value.compareTo(b.time.value)
|
||||||
|
|
||||||
|
else -> {
|
||||||
|
val type = NpcType.valueOf(sortColumn.column.key)
|
||||||
|
(a.enemyCounts[type] ?: 0) - (b.enemyCounts[type] ?: 0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (cmp != 0) {
|
||||||
|
return@Comparator if (sortColumn.direction == SortDirection.Asc) cmp else -cmp
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
0
|
||||||
|
}
|
||||||
|
|
||||||
override val fixedColumns = 2
|
override val fixedColumns = 2
|
||||||
|
|
||||||
override val values: ListCell<HuntMethodModel> = methods
|
override val values: ListCell<HuntMethodModel> by lazy {
|
||||||
|
// TODO: Use ListCell.sortedWith when this is available.
|
||||||
|
observe(huntMethodStore.methods) { allMethods ->
|
||||||
|
methods.value = allMethods
|
||||||
|
.asSequence()
|
||||||
|
.filter { it.episode == episode }
|
||||||
|
.sortedWith(comparator)
|
||||||
|
.toList()
|
||||||
|
}
|
||||||
|
|
||||||
override val valuesStatus: LoadingStatusCell = huntMethodStore.methodsStatus
|
methods
|
||||||
|
}
|
||||||
|
|
||||||
|
override val loadingStatus: LoadingStatusCell = huntMethodStore.methodsStatus
|
||||||
|
|
||||||
override val columns: ListCell<Column<HuntMethodModel>> = listCell(
|
override val columns: ListCell<Column<HuntMethodModel>> = listCell(
|
||||||
Column(
|
Column(
|
||||||
@ -60,42 +96,6 @@ class MethodsForEpisodeController(
|
|||||||
}.toTypedArray()
|
}.toTypedArray()
|
||||||
)
|
)
|
||||||
|
|
||||||
private var sortColumns: List<SortColumn<HuntMethodModel>> = emptyList()
|
|
||||||
|
|
||||||
private val comparator: Comparator<HuntMethodModel> =
|
|
||||||
Comparator { a, b ->
|
|
||||||
for (sortColumn in sortColumns) {
|
|
||||||
val cmp = when (sortColumn.column.key) {
|
|
||||||
METHOD_COL_KEY ->
|
|
||||||
a.name.asDynamic().localeCompare(b.name).unsafeCast<Int>()
|
|
||||||
|
|
||||||
TIME_COL_KEY -> a.time.value.compareTo(b.time.value)
|
|
||||||
|
|
||||||
else -> {
|
|
||||||
val type = NpcType.valueOf(sortColumn.column.key)
|
|
||||||
(a.enemyCounts[type] ?: 0) - (b.enemyCounts[type] ?: 0)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cmp != 0) {
|
|
||||||
return@Comparator if (sortColumn.direction == SortDirection.Asc) cmp else -cmp
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
0
|
|
||||||
}
|
|
||||||
|
|
||||||
init {
|
|
||||||
// TODO: Use ListCell.sortedWith when this is available.
|
|
||||||
observe(huntMethodStore.methods) { allMethods ->
|
|
||||||
methods.value = allMethods
|
|
||||||
.asSequence()
|
|
||||||
.filter { it.episode == episode }
|
|
||||||
.sortedWith(comparator)
|
|
||||||
.toList()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun sort(sortColumns: List<SortColumn<HuntMethodModel>>) {
|
override fun sort(sortColumns: List<SortColumn<HuntMethodModel>>) {
|
||||||
this.sortColumns = sortColumns
|
this.sortColumns = sortColumns
|
||||||
methods.sortWith(comparator)
|
methods.sortWith(comparator)
|
||||||
|
@ -7,8 +7,8 @@ import world.phantasmal.observable.cell.list.mutableListCell
|
|||||||
import world.phantasmal.psolib.Episode
|
import world.phantasmal.psolib.Episode
|
||||||
import world.phantasmal.psolib.fileFormats.quest.NpcType
|
import world.phantasmal.psolib.fileFormats.quest.NpcType
|
||||||
import world.phantasmal.web.core.loading.AssetLoader
|
import world.phantasmal.web.core.loading.AssetLoader
|
||||||
import world.phantasmal.web.core.models.Server
|
|
||||||
import world.phantasmal.web.core.stores.UiStore
|
import world.phantasmal.web.core.stores.UiStore
|
||||||
|
import world.phantasmal.web.huntOptimizer.HuntOptimizerUrls.methods
|
||||||
import world.phantasmal.web.huntOptimizer.models.HuntMethodModel
|
import world.phantasmal.web.huntOptimizer.models.HuntMethodModel
|
||||||
import world.phantasmal.web.huntOptimizer.models.SimpleQuestModel
|
import world.phantasmal.web.huntOptimizer.models.SimpleQuestModel
|
||||||
import world.phantasmal.web.huntOptimizer.persistence.HuntMethodPersister
|
import world.phantasmal.web.huntOptimizer.persistence.HuntMethodPersister
|
||||||
@ -27,22 +27,27 @@ class HuntMethodStore(
|
|||||||
private val huntMethodPersister: HuntMethodPersister,
|
private val huntMethodPersister: HuntMethodPersister,
|
||||||
) : Store() {
|
) : Store() {
|
||||||
private val _methods = mutableListCell<HuntMethodModel> { arrayOf(it.time) }
|
private val _methods = mutableListCell<HuntMethodModel> { arrayOf(it.time) }
|
||||||
|
private val _methodsStatus = LoadingStatusCellImpl(scope, "methods", ::loadMethods)
|
||||||
|
|
||||||
|
/** Hunting methods supported by the current server. */
|
||||||
val methods: ListCell<HuntMethodModel> by lazy {
|
val methods: ListCell<HuntMethodModel> by lazy {
|
||||||
observe(uiStore.server) { loadMethods(it) }
|
observe(uiStore.server) { _methodsStatus.load() }
|
||||||
_methods
|
_methods
|
||||||
}
|
}
|
||||||
|
|
||||||
private val _methodsStatus = LoadingStatusCellImpl("methods")
|
/** Loading status of [methods]. */
|
||||||
val methodsStatus: LoadingStatusCell = _methodsStatus
|
val methodsStatus: LoadingStatusCell = _methodsStatus
|
||||||
|
|
||||||
suspend fun setMethodTime(method: HuntMethodModel, time: Duration) {
|
suspend fun setMethodTime(method: HuntMethodModel, time: Duration) {
|
||||||
method.setUserTime(time)
|
method.setUserTime(time)
|
||||||
|
|
||||||
huntMethodPersister.persistMethodUserTimes(methods.value, uiStore.server.value)
|
huntMethodPersister.persistMethodUserTimes(methods.value, uiStore.server.value)
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun loadMethods(server: Server) {
|
private suspend fun loadMethods() {
|
||||||
_methodsStatus.load(scope) {
|
val server = uiStore.server.value
|
||||||
|
|
||||||
|
withContext(Dispatchers.Default) {
|
||||||
val quests = assetLoader.load<List<QuestDto>>("/quests.${server.slug}.json")
|
val quests = assetLoader.load<List<QuestDto>>("/quests.${server.slug}.json")
|
||||||
|
|
||||||
val methods = quests
|
val methods = quests
|
||||||
|
@ -0,0 +1,39 @@
|
|||||||
|
package world.phantasmal.web.huntOptimizer.controllers
|
||||||
|
|
||||||
|
import world.phantasmal.psolib.Episode
|
||||||
|
import world.phantasmal.web.huntOptimizer.stores.HuntMethodStore
|
||||||
|
import world.phantasmal.web.test.WebTestSuite
|
||||||
|
import world.phantasmal.webui.LoadingStatus
|
||||||
|
import kotlin.test.Test
|
||||||
|
import kotlin.test.assertEquals
|
||||||
|
import kotlin.test.assertTrue
|
||||||
|
|
||||||
|
class MethodsForEpisodeControllerTests : WebTestSuite {
|
||||||
|
@Test
|
||||||
|
fun methods_for_the_given_episode_are_loaded_when_necessary() = testAsync {
|
||||||
|
for (episode in Episode.values()) {
|
||||||
|
val ctrl = disposer.add(
|
||||||
|
MethodsForEpisodeController(
|
||||||
|
// Create our own store each time to ensure methods is uninitialized.
|
||||||
|
disposer.add(HuntMethodStore(
|
||||||
|
components.uiStore,
|
||||||
|
components.assetLoader,
|
||||||
|
components.huntMethodPersister,
|
||||||
|
)),
|
||||||
|
episode,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
assertEquals(LoadingStatus.Uninitialized, ctrl.loadingStatus.value)
|
||||||
|
|
||||||
|
// Start loading methods by accessing values.
|
||||||
|
ctrl.values
|
||||||
|
|
||||||
|
ctrl.loadingStatus.await()
|
||||||
|
|
||||||
|
assertEquals(LoadingStatus.Ok, ctrl.loadingStatus.value)
|
||||||
|
|
||||||
|
assertTrue(ctrl.values.value.all { it.episode == episode })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -10,15 +10,20 @@ import world.phantasmal.core.disposable.Disposable
|
|||||||
import world.phantasmal.core.disposable.disposable
|
import world.phantasmal.core.disposable.disposable
|
||||||
import world.phantasmal.testUtils.TestContext
|
import world.phantasmal.testUtils.TestContext
|
||||||
import world.phantasmal.web.core.loading.AssetLoader
|
import world.phantasmal.web.core.loading.AssetLoader
|
||||||
|
import world.phantasmal.web.core.persistence.KeyValueStore
|
||||||
|
import world.phantasmal.web.core.persistence.MemoryKeyValueStore
|
||||||
import world.phantasmal.web.core.rendering.DisposableThreeRenderer
|
import world.phantasmal.web.core.rendering.DisposableThreeRenderer
|
||||||
import world.phantasmal.web.core.stores.ApplicationUrl
|
import world.phantasmal.web.core.stores.ApplicationUrl
|
||||||
import world.phantasmal.web.core.stores.UiStore
|
import world.phantasmal.web.core.stores.UiStore
|
||||||
import world.phantasmal.web.core.undo.UndoManager
|
import world.phantasmal.web.core.undo.UndoManager
|
||||||
import world.phantasmal.web.externals.three.WebGLRenderer
|
import world.phantasmal.web.externals.three.WebGLRenderer
|
||||||
|
import world.phantasmal.web.huntOptimizer.persistence.HuntMethodPersister
|
||||||
|
import world.phantasmal.web.huntOptimizer.stores.HuntMethodStore
|
||||||
import world.phantasmal.web.questEditor.loading.AreaAssetLoader
|
import world.phantasmal.web.questEditor.loading.AreaAssetLoader
|
||||||
import world.phantasmal.web.questEditor.loading.QuestLoader
|
import world.phantasmal.web.questEditor.loading.QuestLoader
|
||||||
import world.phantasmal.web.questEditor.stores.AreaStore
|
import world.phantasmal.web.questEditor.stores.AreaStore
|
||||||
import world.phantasmal.web.questEditor.stores.QuestEditorStore
|
import world.phantasmal.web.questEditor.stores.QuestEditorStore
|
||||||
|
import kotlin.properties.ReadWriteProperty
|
||||||
import kotlin.reflect.KProperty
|
import kotlin.reflect.KProperty
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -52,6 +57,12 @@ class TestComponents(private val ctx: TestContext) {
|
|||||||
|
|
||||||
var questLoader: QuestLoader by default { QuestLoader(assetLoader) }
|
var questLoader: QuestLoader by default { QuestLoader(assetLoader) }
|
||||||
|
|
||||||
|
// Persistence
|
||||||
|
|
||||||
|
var keyValueStore: KeyValueStore by default { MemoryKeyValueStore() }
|
||||||
|
|
||||||
|
var huntMethodPersister: HuntMethodPersister by default { HuntMethodPersister(keyValueStore) }
|
||||||
|
|
||||||
// Undo
|
// Undo
|
||||||
|
|
||||||
var undoManager: UndoManager by default { UndoManager() }
|
var undoManager: UndoManager by default { UndoManager() }
|
||||||
@ -62,6 +73,10 @@ class TestComponents(private val ctx: TestContext) {
|
|||||||
|
|
||||||
var areaStore: AreaStore by default { AreaStore(areaAssetLoader) }
|
var areaStore: AreaStore by default { AreaStore(areaAssetLoader) }
|
||||||
|
|
||||||
|
var huntMethodStore: HuntMethodStore by default {
|
||||||
|
HuntMethodStore(uiStore, assetLoader, huntMethodPersister)
|
||||||
|
}
|
||||||
|
|
||||||
var questEditorStore: QuestEditorStore by default {
|
var questEditorStore: QuestEditorStore by default {
|
||||||
QuestEditorStore(questLoader, uiStore, areaStore, undoManager, initializeNewQuest = false)
|
QuestEditorStore(questLoader, uiStore, areaStore, undoManager, initializeNewQuest = false)
|
||||||
}
|
}
|
||||||
@ -78,30 +93,30 @@ class TestComponents(private val ctx: TestContext) {
|
|||||||
|
|
||||||
private fun <T> default(defaultValue: () -> T) = LazyDefault(defaultValue)
|
private fun <T> default(defaultValue: () -> T) = LazyDefault(defaultValue)
|
||||||
|
|
||||||
private inner class LazyDefault<T>(private val defaultValue: () -> T) {
|
private inner class LazyDefault<T>(
|
||||||
|
private val defaultValue: () -> T,
|
||||||
|
) : ReadWriteProperty<Any?, T> {
|
||||||
|
|
||||||
private var initialized = false
|
private var initialized = false
|
||||||
private var value: T? = null
|
private var value: T? = null
|
||||||
|
|
||||||
operator fun getValue(thisRef: Any?, prop: KProperty<*>): T {
|
override operator fun getValue(thisRef: Any?, property: KProperty<*>): T {
|
||||||
if (!initialized) {
|
if (!initialized) {
|
||||||
val value = defaultValue()
|
setValue(defaultValue())
|
||||||
|
|
||||||
if (value is Disposable) {
|
|
||||||
ctx.disposer.add(value)
|
|
||||||
}
|
|
||||||
|
|
||||||
this.value = value
|
|
||||||
initialized = true
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return value.unsafeCast<T>()
|
return value.unsafeCast<T>()
|
||||||
}
|
}
|
||||||
|
|
||||||
operator fun setValue(thisRef: Any?, prop: KProperty<*>, value: T) {
|
override operator fun setValue(thisRef: Any?, property: KProperty<*>, value: T) {
|
||||||
require(!initialized) {
|
require(!initialized) {
|
||||||
"Property ${prop.name} is already initialized."
|
"Property ${property.name} is already initialized."
|
||||||
}
|
}
|
||||||
|
|
||||||
|
setValue(value)
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun setValue(value: T) {
|
||||||
if (value is Disposable) {
|
if (value is Disposable) {
|
||||||
ctx.disposer.add(value)
|
ctx.disposer.add(value)
|
||||||
}
|
}
|
||||||
|
@ -3,7 +3,6 @@ package world.phantasmal.webui
|
|||||||
import kotlinx.coroutines.*
|
import kotlinx.coroutines.*
|
||||||
import mu.KotlinLogging
|
import mu.KotlinLogging
|
||||||
import world.phantasmal.observable.cell.Cell
|
import world.phantasmal.observable.cell.Cell
|
||||||
import world.phantasmal.observable.cell.ImmutableCell
|
|
||||||
import world.phantasmal.observable.cell.SimpleCell
|
import world.phantasmal.observable.cell.SimpleCell
|
||||||
import kotlin.time.measureTime
|
import kotlin.time.measureTime
|
||||||
|
|
||||||
@ -18,58 +17,65 @@ enum class LoadingStatus {
|
|||||||
}
|
}
|
||||||
|
|
||||||
interface LoadingStatusCell : Cell<LoadingStatus> {
|
interface LoadingStatusCell : Cell<LoadingStatus> {
|
||||||
suspend fun awaitLoad()
|
/** Await the current load, if a load in ongoing. */
|
||||||
|
suspend fun await()
|
||||||
}
|
}
|
||||||
|
|
||||||
class ImmutableLoadingStatusCell(status: LoadingStatus) :
|
class LoadingStatusCellImpl private constructor(
|
||||||
LoadingStatusCell,
|
private val scope: CoroutineScope,
|
||||||
Cell<LoadingStatus> by ImmutableCell(status) {
|
|
||||||
|
|
||||||
override suspend fun awaitLoad() {
|
|
||||||
// Nothing to await.
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class LoadingStatusCellImpl(
|
|
||||||
private val cellDelegate: SimpleCell<LoadingStatus>,
|
|
||||||
private val dataName: String,
|
private val dataName: String,
|
||||||
|
/** Will be called with [Dispatchers.Main] context. */
|
||||||
|
private val loadData: suspend () -> Unit,
|
||||||
|
private val cellDelegate: SimpleCell<LoadingStatus>,
|
||||||
) : LoadingStatusCell, Cell<LoadingStatus> by cellDelegate {
|
) : LoadingStatusCell, Cell<LoadingStatus> by cellDelegate {
|
||||||
|
|
||||||
constructor(dataName: String) : this(SimpleCell(LoadingStatus.Uninitialized), dataName)
|
constructor(
|
||||||
|
scope: CoroutineScope,
|
||||||
|
dataName: String,
|
||||||
|
loadData: suspend () -> Unit,
|
||||||
|
) : this(scope, dataName, loadData, SimpleCell(LoadingStatus.Uninitialized))
|
||||||
|
|
||||||
private var job: Job? = null
|
private var currentJob: Job? = null
|
||||||
|
|
||||||
fun load(scope: CoroutineScope, loadData: suspend () -> Unit) {
|
fun load() {
|
||||||
logger.trace { "Loading $dataName." }
|
logger.trace { "Loading $dataName." }
|
||||||
|
|
||||||
cellDelegate.value =
|
cellDelegate.value =
|
||||||
if (value == LoadingStatus.Uninitialized) LoadingStatus.InitialLoad
|
if (value == LoadingStatus.Uninitialized) LoadingStatus.InitialLoad
|
||||||
else LoadingStatus.Loading
|
else LoadingStatus.Loading
|
||||||
|
|
||||||
job = scope.launch {
|
currentJob?.cancel("New load started.")
|
||||||
|
|
||||||
|
currentJob = scope.launch(Dispatchers.Main) {
|
||||||
var success = false
|
var success = false
|
||||||
|
|
||||||
try {
|
try {
|
||||||
val duration = measureTime {
|
val duration = measureTime {
|
||||||
withContext(Dispatchers.Default) {
|
loadData()
|
||||||
loadData()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.trace { "Loaded $dataName in ${duration.inWholeMilliseconds}ms." }
|
logger.trace { "Loaded $dataName in ${duration.inWholeMilliseconds}ms." }
|
||||||
|
|
||||||
success = true
|
success = true
|
||||||
|
} catch (e: CancellationException) {
|
||||||
|
logger.trace(e) { "Loading $dataName was cancelled." }
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
logger.error(e) { "Error while loading $dataName." }
|
logger.error(e) { "Error while loading $dataName." }
|
||||||
} finally {
|
}
|
||||||
job = null
|
|
||||||
|
|
||||||
|
// Only reset job and set value when a new job hasn't been started in the meantime.
|
||||||
|
if (coroutineContext.job == currentJob) {
|
||||||
|
currentJob = null
|
||||||
cellDelegate.value = if (success) LoadingStatus.Ok else LoadingStatus.Error
|
cellDelegate.value = if (success) LoadingStatus.Ok else LoadingStatus.Error
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override suspend fun awaitLoad() {
|
override suspend fun await() {
|
||||||
job?.join()
|
currentJob?.let {
|
||||||
|
if (!it.isCompleted) {
|
||||||
|
it.join()
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3,8 +3,6 @@ package world.phantasmal.webui.controllers
|
|||||||
import world.phantasmal.observable.cell.Cell
|
import world.phantasmal.observable.cell.Cell
|
||||||
import world.phantasmal.observable.cell.list.ListCell
|
import world.phantasmal.observable.cell.list.ListCell
|
||||||
import world.phantasmal.observable.cell.nullCell
|
import world.phantasmal.observable.cell.nullCell
|
||||||
import world.phantasmal.webui.ImmutableLoadingStatusCell
|
|
||||||
import world.phantasmal.webui.LoadingStatus
|
|
||||||
import world.phantasmal.webui.LoadingStatusCell
|
import world.phantasmal.webui.LoadingStatusCell
|
||||||
|
|
||||||
class Column<T>(
|
class Column<T>(
|
||||||
@ -44,9 +42,7 @@ abstract class TableController<T> : Controller() {
|
|||||||
/** Each value is represented by a row in the table. */
|
/** Each value is represented by a row in the table. */
|
||||||
abstract val values: ListCell<T>
|
abstract val values: ListCell<T>
|
||||||
|
|
||||||
open val valuesStatus: LoadingStatusCell =
|
open val loadingStatus: LoadingStatusCell? = null
|
||||||
// Assume values are already loaded by default.
|
|
||||||
ImmutableLoadingStatusCell(LoadingStatus.Ok)
|
|
||||||
|
|
||||||
abstract val columns: ListCell<Column<T>>
|
abstract val columns: ListCell<Column<T>>
|
||||||
|
|
||||||
|
@ -29,18 +29,22 @@ class Table<T>(
|
|||||||
div {
|
div {
|
||||||
className = "pw-table-notification"
|
className = "pw-table-notification"
|
||||||
|
|
||||||
observe(ctrl.valuesStatus) {
|
ctrl.loadingStatus?.let { loadingStatus ->
|
||||||
when (it) {
|
observe(loadingStatus) { status ->
|
||||||
LoadingStatus.InitialLoad -> {
|
when (status) {
|
||||||
hidden = false
|
LoadingStatus.Uninitialized,
|
||||||
innerText = "Loading..."
|
LoadingStatus.InitialLoad,
|
||||||
}
|
-> {
|
||||||
LoadingStatus.Error -> {
|
hidden = false
|
||||||
hidden = false
|
innerText = "Loading..."
|
||||||
innerText = "An error occurred while loading this table."
|
}
|
||||||
}
|
LoadingStatus.Error -> {
|
||||||
else -> {
|
hidden = false
|
||||||
hidden = true
|
innerText = "An error occurred while loading this table."
|
||||||
|
}
|
||||||
|
else -> {
|
||||||
|
hidden = true
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user