mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-04 06:28:28 +08:00
Added LoadingStatusCell, a cell that shows the status of some loadable data. Table and TableController make use of it to show a notification on the first load and on errors.
This commit is contained in:
parent
6374a3f054
commit
3e17865346
@ -1,12 +1,13 @@
|
|||||||
package world.phantasmal.web.huntOptimizer.controllers
|
package world.phantasmal.web.huntOptimizer.controllers
|
||||||
|
|
||||||
import world.phantasmal.psolib.Episode
|
|
||||||
import world.phantasmal.psolib.fileFormats.quest.NpcType
|
|
||||||
import world.phantasmal.observable.cell.list.ListCell
|
import world.phantasmal.observable.cell.list.ListCell
|
||||||
import world.phantasmal.observable.cell.list.listCell
|
import world.phantasmal.observable.cell.list.listCell
|
||||||
import world.phantasmal.observable.cell.list.mutableListCell
|
import world.phantasmal.observable.cell.list.mutableListCell
|
||||||
|
import world.phantasmal.psolib.Episode
|
||||||
|
import world.phantasmal.psolib.fileFormats.quest.NpcType
|
||||||
import world.phantasmal.web.huntOptimizer.models.HuntMethodModel
|
import world.phantasmal.web.huntOptimizer.models.HuntMethodModel
|
||||||
import world.phantasmal.web.huntOptimizer.stores.HuntMethodStore
|
import world.phantasmal.web.huntOptimizer.stores.HuntMethodStore
|
||||||
|
import world.phantasmal.webui.LoadingStatusCell
|
||||||
import world.phantasmal.webui.controllers.Column
|
import world.phantasmal.webui.controllers.Column
|
||||||
import world.phantasmal.webui.controllers.SortColumn
|
import world.phantasmal.webui.controllers.SortColumn
|
||||||
import world.phantasmal.webui.controllers.SortDirection
|
import world.phantasmal.webui.controllers.SortDirection
|
||||||
@ -24,6 +25,8 @@ class MethodsForEpisodeController(
|
|||||||
|
|
||||||
override val values: ListCell<HuntMethodModel> = methods
|
override val values: ListCell<HuntMethodModel> = methods
|
||||||
|
|
||||||
|
override val valuesStatus: LoadingStatusCell = huntMethodStore.methodsStatus
|
||||||
|
|
||||||
override val columns: ListCell<Column<HuntMethodModel>> = listCell(
|
override val columns: ListCell<Column<HuntMethodModel>> = listCell(
|
||||||
Column(
|
Column(
|
||||||
key = METHOD_COL_KEY,
|
key = METHOD_COL_KEY,
|
||||||
|
@ -1,12 +1,11 @@
|
|||||||
package world.phantasmal.web.huntOptimizer.stores
|
package world.phantasmal.web.huntOptimizer.stores
|
||||||
|
|
||||||
import kotlinx.coroutines.Dispatchers
|
import kotlinx.coroutines.Dispatchers
|
||||||
import kotlinx.coroutines.launch
|
|
||||||
import kotlinx.coroutines.withContext
|
import kotlinx.coroutines.withContext
|
||||||
import world.phantasmal.psolib.Episode
|
|
||||||
import world.phantasmal.psolib.fileFormats.quest.NpcType
|
|
||||||
import world.phantasmal.observable.cell.list.ListCell
|
import world.phantasmal.observable.cell.list.ListCell
|
||||||
import world.phantasmal.observable.cell.list.mutableListCell
|
import world.phantasmal.observable.cell.list.mutableListCell
|
||||||
|
import world.phantasmal.psolib.Episode
|
||||||
|
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.models.Server
|
||||||
import world.phantasmal.web.core.stores.UiStore
|
import world.phantasmal.web.core.stores.UiStore
|
||||||
@ -14,6 +13,8 @@ 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
|
||||||
import world.phantasmal.web.shared.dto.QuestDto
|
import world.phantasmal.web.shared.dto.QuestDto
|
||||||
|
import world.phantasmal.webui.LoadingStatusCell
|
||||||
|
import world.phantasmal.webui.LoadingStatusCellImpl
|
||||||
import world.phantasmal.webui.stores.Store
|
import world.phantasmal.webui.stores.Store
|
||||||
import kotlin.collections.component1
|
import kotlin.collections.component1
|
||||||
import kotlin.collections.component2
|
import kotlin.collections.component2
|
||||||
@ -32,13 +33,16 @@ class HuntMethodStore(
|
|||||||
_methods
|
_methods
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private val _methodsStatus = LoadingStatusCellImpl("methods")
|
||||||
|
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 fun loadMethods(server: Server) {
|
||||||
scope.launch(Dispatchers.Default) {
|
_methodsStatus.load(scope) {
|
||||||
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,75 @@
|
|||||||
|
package world.phantasmal.webui
|
||||||
|
|
||||||
|
import kotlinx.coroutines.*
|
||||||
|
import mu.KotlinLogging
|
||||||
|
import world.phantasmal.observable.cell.Cell
|
||||||
|
import world.phantasmal.observable.cell.ImmutableCell
|
||||||
|
import world.phantasmal.observable.cell.SimpleCell
|
||||||
|
import kotlin.time.measureTime
|
||||||
|
|
||||||
|
private val logger = KotlinLogging.logger {}
|
||||||
|
|
||||||
|
enum class LoadingStatus {
|
||||||
|
Uninitialized,
|
||||||
|
InitialLoad,
|
||||||
|
Loading,
|
||||||
|
Ok,
|
||||||
|
Error,
|
||||||
|
}
|
||||||
|
|
||||||
|
interface LoadingStatusCell : Cell<LoadingStatus> {
|
||||||
|
suspend fun awaitLoad()
|
||||||
|
}
|
||||||
|
|
||||||
|
class ImmutableLoadingStatusCell(status: LoadingStatus) :
|
||||||
|
LoadingStatusCell,
|
||||||
|
Cell<LoadingStatus> by ImmutableCell(status) {
|
||||||
|
|
||||||
|
override suspend fun awaitLoad() {
|
||||||
|
// Nothing to await.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class LoadingStatusCellImpl(
|
||||||
|
private val cellDelegate: SimpleCell<LoadingStatus>,
|
||||||
|
private val dataName: String,
|
||||||
|
) : LoadingStatusCell, Cell<LoadingStatus> by cellDelegate {
|
||||||
|
|
||||||
|
constructor(dataName: String) : this(SimpleCell(LoadingStatus.Uninitialized), dataName)
|
||||||
|
|
||||||
|
private var job: Job? = null
|
||||||
|
|
||||||
|
fun load(scope: CoroutineScope, loadData: suspend () -> Unit) {
|
||||||
|
logger.trace { "Loading $dataName." }
|
||||||
|
|
||||||
|
cellDelegate.value =
|
||||||
|
if (value == LoadingStatus.Uninitialized) LoadingStatus.InitialLoad
|
||||||
|
else LoadingStatus.Loading
|
||||||
|
|
||||||
|
job = scope.launch {
|
||||||
|
var success = false
|
||||||
|
|
||||||
|
try {
|
||||||
|
val duration = measureTime {
|
||||||
|
withContext(Dispatchers.Default) {
|
||||||
|
loadData()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.trace { "Loaded $dataName in ${duration.inWholeMilliseconds}ms." }
|
||||||
|
|
||||||
|
success = true
|
||||||
|
} catch (e: Exception) {
|
||||||
|
logger.error(e) { "Error while loading $dataName." }
|
||||||
|
} finally {
|
||||||
|
job = null
|
||||||
|
|
||||||
|
cellDelegate.value = if (success) LoadingStatus.Ok else LoadingStatus.Error
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
override suspend fun awaitLoad() {
|
||||||
|
job?.join()
|
||||||
|
}
|
||||||
|
}
|
@ -3,6 +3,9 @@ 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
|
||||||
|
|
||||||
class Column<T>(
|
class Column<T>(
|
||||||
val key: String,
|
val key: String,
|
||||||
@ -34,13 +37,17 @@ interface SortColumn<T> {
|
|||||||
abstract class TableController<T> : Controller() {
|
abstract class TableController<T> : Controller() {
|
||||||
private val sortColumns: MutableList<SortColumnImpl> = mutableListOf()
|
private val sortColumns: MutableList<SortColumnImpl> = mutableListOf()
|
||||||
|
|
||||||
/**
|
/** How many columns stay in place on the left side while scrolling. */
|
||||||
* How many columns stay in place on the left side while scrolling.
|
|
||||||
*/
|
|
||||||
open val fixedColumns: Int = 0
|
open val fixedColumns: Int = 0
|
||||||
open val hasFooter: Boolean = false
|
open val hasFooter: Boolean = false
|
||||||
|
|
||||||
|
/** Each value is represented by a row in the table. */
|
||||||
abstract val values: ListCell<T>
|
abstract val values: ListCell<T>
|
||||||
|
|
||||||
|
open val valuesStatus: LoadingStatusCell =
|
||||||
|
// Assume values are already loaded by default.
|
||||||
|
ImmutableLoadingStatusCell(LoadingStatus.Ok)
|
||||||
|
|
||||||
abstract val columns: ListCell<Column<T>>
|
abstract val columns: ListCell<Column<T>>
|
||||||
|
|
||||||
open fun sort(sortColumns: List<SortColumn<T>>) {
|
open fun sort(sortColumns: List<SortColumn<T>>) {
|
||||||
|
@ -5,6 +5,7 @@ import world.phantasmal.core.disposable.Disposable
|
|||||||
import world.phantasmal.core.disposable.Disposer
|
import world.phantasmal.core.disposable.Disposer
|
||||||
import world.phantasmal.observable.cell.Cell
|
import world.phantasmal.observable.cell.Cell
|
||||||
import world.phantasmal.observable.cell.trueCell
|
import world.phantasmal.observable.cell.trueCell
|
||||||
|
import world.phantasmal.webui.LoadingStatus
|
||||||
import world.phantasmal.webui.controllers.Column
|
import world.phantasmal.webui.controllers.Column
|
||||||
import world.phantasmal.webui.controllers.TableController
|
import world.phantasmal.webui.controllers.TableController
|
||||||
import world.phantasmal.webui.dom.*
|
import world.phantasmal.webui.dom.*
|
||||||
@ -25,6 +26,25 @@ class Table<T>(
|
|||||||
|
|
||||||
this@Table.className?.let { classList.add(it) }
|
this@Table.className?.let { classList.add(it) }
|
||||||
|
|
||||||
|
div {
|
||||||
|
className = "pw-table-notification"
|
||||||
|
|
||||||
|
observe(ctrl.valuesStatus) {
|
||||||
|
when (it) {
|
||||||
|
LoadingStatus.InitialLoad -> {
|
||||||
|
hidden = false
|
||||||
|
innerText = "Loading..."
|
||||||
|
}
|
||||||
|
LoadingStatus.Error -> {
|
||||||
|
hidden = false
|
||||||
|
innerText = "An error occurred while loading this table."
|
||||||
|
}
|
||||||
|
else -> {
|
||||||
|
hidden = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
thead {
|
thead {
|
||||||
tr {
|
tr {
|
||||||
className = "pw-table-row pw-table-header-row"
|
className = "pw-table-row pw-table-header-row"
|
||||||
@ -184,6 +204,20 @@ class Table<T>(
|
|||||||
background-color: var(--pw-bg-color);
|
background-color: var(--pw-bg-color);
|
||||||
border-collapse: collapse;
|
border-collapse: collapse;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.pw-table-notification {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
display: grid;
|
||||||
|
grid-template: 100% / 100%;
|
||||||
|
place-items: center;
|
||||||
|
text-align: center;
|
||||||
|
color: var(--pw-text-color-disabled);
|
||||||
|
font-size: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
.pw-table > thead {
|
.pw-table > thead {
|
||||||
position: sticky;
|
position: sticky;
|
||||||
|
Loading…
Reference in New Issue
Block a user