mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
Table can now have a changing amount of columns.
This commit is contained in:
parent
aeded71dde
commit
e21bce7695
@ -5,7 +5,7 @@ import world.phantasmal.lib.cursor.Cursor
|
|||||||
class ItemPmt(
|
class ItemPmt(
|
||||||
val statBoosts: List<PmtStatBoost>,
|
val statBoosts: List<PmtStatBoost>,
|
||||||
val frames: List<PmtFrame>,
|
val frames: List<PmtFrame>,
|
||||||
val barriers: List<PmtFrame>,
|
val barriers: List<PmtBarrier>,
|
||||||
val units: List<PmtUnit>,
|
val units: List<PmtUnit>,
|
||||||
val tools: List<List<PmtTool>>,
|
val tools: List<List<PmtTool>>,
|
||||||
val weapons: List<List<PmtWeapon>>,
|
val weapons: List<List<PmtWeapon>>,
|
||||||
@ -42,6 +42,8 @@ class PmtFrame(
|
|||||||
val unknown1: Int,
|
val unknown1: Int,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
typealias PmtBarrier = PmtFrame
|
||||||
|
|
||||||
class PmtUnit(
|
class PmtUnit(
|
||||||
val id: Int,
|
val id: Int,
|
||||||
val type: Int,
|
val type: Int,
|
||||||
|
@ -12,5 +12,8 @@ private val EP_AND_NAME_TO_NPC_TYPE: Map<Pair<String, Episode>, NpcType> =
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Uniquely identifies an NPC. Tries to match on [NpcType.simpleName] and [NpcType.ultimateName].
|
||||||
|
*/
|
||||||
fun NpcType.Companion.fromNameAndEpisode(name: String, episode: Episode): NpcType? =
|
fun NpcType.Companion.fromNameAndEpisode(name: String, episode: Episode): NpcType? =
|
||||||
EP_AND_NAME_TO_NPC_TYPE[Pair(name, episode)]
|
EP_AND_NAME_TO_NPC_TYPE[Pair(name, episode)]
|
||||||
|
@ -136,7 +136,7 @@ private fun download(
|
|||||||
sectionId,
|
sectionId,
|
||||||
enemy = npcType,
|
enemy = npcType,
|
||||||
itemTypeId = itemType.id,
|
itemTypeId = itemType.id,
|
||||||
dropRate = dropRateNum.toDouble() / dropRateDenom.toDouble(),
|
anythingRate = dropRateNum.toDouble() / dropRateDenom.toDouble(),
|
||||||
rareRate = rareRateNum.toDouble() / rareRateDenom.toDouble(),
|
rareRate = rareRateNum.toDouble() / rareRateDenom.toDouble(),
|
||||||
))
|
))
|
||||||
} catch (e: Exception) {
|
} catch (e: Exception) {
|
||||||
|
@ -1,26 +0,0 @@
|
|||||||
package world.phantasmal.web.shared.dto
|
|
||||||
|
|
||||||
import kotlinx.serialization.Serializable
|
|
||||||
import world.phantasmal.lib.Episode
|
|
||||||
import world.phantasmal.lib.fileFormats.quest.NpcType
|
|
||||||
|
|
||||||
@Serializable
|
|
||||||
class EnemyDrop(
|
|
||||||
val difficulty: Difficulty,
|
|
||||||
val episode: Episode,
|
|
||||||
val sectionId: SectionId,
|
|
||||||
val enemy: NpcType,
|
|
||||||
val itemTypeId: Int,
|
|
||||||
val dropRate: Double,
|
|
||||||
val rareRate: Double,
|
|
||||||
)
|
|
||||||
|
|
||||||
@Serializable
|
|
||||||
class BoxDrop(
|
|
||||||
val difficulty: Difficulty,
|
|
||||||
val episode: Episode,
|
|
||||||
val sectionId: SectionId,
|
|
||||||
val areaId: Int,
|
|
||||||
val itemTypeId: Int,
|
|
||||||
val dropRate: Double,
|
|
||||||
)
|
|
@ -0,0 +1,43 @@
|
|||||||
|
package world.phantasmal.web.shared.dto
|
||||||
|
|
||||||
|
import kotlinx.serialization.Serializable
|
||||||
|
import world.phantasmal.lib.Episode
|
||||||
|
import world.phantasmal.lib.fileFormats.quest.NpcType
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
sealed class ItemDrop {
|
||||||
|
abstract val difficulty: Difficulty
|
||||||
|
abstract val episode: Episode
|
||||||
|
abstract val sectionId: SectionId
|
||||||
|
abstract val itemTypeId: Int
|
||||||
|
abstract val dropRate: Double
|
||||||
|
}
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
class EnemyDrop(
|
||||||
|
override val difficulty: Difficulty,
|
||||||
|
override val episode: Episode,
|
||||||
|
override val sectionId: SectionId,
|
||||||
|
val enemy: NpcType,
|
||||||
|
override val itemTypeId: Int,
|
||||||
|
/**
|
||||||
|
* Chance that this enemy drops anything at all.
|
||||||
|
*/
|
||||||
|
val anythingRate: Double,
|
||||||
|
/**
|
||||||
|
* Chance that an enemy drops this rare item, if it drops anything at all.
|
||||||
|
*/
|
||||||
|
val rareRate: Double,
|
||||||
|
) : ItemDrop() {
|
||||||
|
override val dropRate: Double = anythingRate * rareRate
|
||||||
|
}
|
||||||
|
|
||||||
|
@Serializable
|
||||||
|
class BoxDrop(
|
||||||
|
override val difficulty: Difficulty,
|
||||||
|
override val episode: Episode,
|
||||||
|
override val sectionId: SectionId,
|
||||||
|
val areaId: Int,
|
||||||
|
override val itemTypeId: Int,
|
||||||
|
override val dropRate: Double,
|
||||||
|
) : ItemDrop()
|
@ -37,8 +37,14 @@ class ItemDropStore(
|
|||||||
|
|
||||||
class EnemyDropTable(
|
class EnemyDropTable(
|
||||||
private val table: Map<Triple<Difficulty, SectionId, NpcType>, EnemyDrop>,
|
private val table: Map<Triple<Difficulty, SectionId, NpcType>, EnemyDrop>,
|
||||||
|
/**
|
||||||
|
* Mapping of [ItemType] ids to [EnemyDrop]s.
|
||||||
|
*/
|
||||||
private val itemTypeToDrops: Map<Int, List<EnemyDrop>>,
|
private val itemTypeToDrops: Map<Int, List<EnemyDrop>>,
|
||||||
) {
|
) {
|
||||||
|
fun getDrop(difficulty: Difficulty, sectionId: SectionId, npcType: NpcType): EnemyDrop? =
|
||||||
|
table[Triple(difficulty, sectionId, npcType)]
|
||||||
|
|
||||||
fun getDropsForItemType(itemType: ItemType): List<EnemyDrop> =
|
fun getDropsForItemType(itemType: ItemType): List<EnemyDrop> =
|
||||||
itemTypeToDrops[itemType.id] ?: emptyList()
|
itemTypeToDrops[itemType.id] ?: emptyList()
|
||||||
}
|
}
|
||||||
|
@ -3,6 +3,7 @@ package world.phantasmal.web.huntOptimizer.controllers
|
|||||||
import world.phantasmal.lib.Episode
|
import world.phantasmal.lib.Episode
|
||||||
import world.phantasmal.lib.fileFormats.quest.NpcType
|
import world.phantasmal.lib.fileFormats.quest.NpcType
|
||||||
import world.phantasmal.observable.value.list.ListVal
|
import world.phantasmal.observable.value.list.ListVal
|
||||||
|
import world.phantasmal.observable.value.list.listVal
|
||||||
import world.phantasmal.observable.value.list.mutableListVal
|
import world.phantasmal.observable.value.list.mutableListVal
|
||||||
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
|
||||||
@ -19,27 +20,27 @@ class MethodsForEpisodeController(
|
|||||||
private val methods = mutableListVal<HuntMethodModel>()
|
private val methods = mutableListVal<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 }
|
||||||
|
|
||||||
|
override val fixedColumns = 2
|
||||||
|
|
||||||
override val values: ListVal<HuntMethodModel> = methods
|
override val values: ListVal<HuntMethodModel> = methods
|
||||||
|
|
||||||
override val columns: List<Column<HuntMethodModel>> = listOf(
|
override val columns: ListVal<Column<HuntMethodModel>> = listVal(
|
||||||
Column(
|
Column(
|
||||||
key = METHOD_COL_KEY,
|
key = METHOD_COL_KEY,
|
||||||
title = "Method",
|
title = "Method",
|
||||||
fixed = true,
|
|
||||||
width = 250,
|
width = 250,
|
||||||
sortable = true,
|
sortable = true,
|
||||||
),
|
),
|
||||||
Column(
|
Column(
|
||||||
key = TIME_COL_KEY,
|
key = TIME_COL_KEY,
|
||||||
title = "Time",
|
title = "Time",
|
||||||
fixed = true,
|
|
||||||
width = 50,
|
width = 50,
|
||||||
input = true,
|
input = true,
|
||||||
sortable = true,
|
sortable = true,
|
||||||
),
|
),
|
||||||
*enemies.map { enemy ->
|
*enemies.map { enemy ->
|
||||||
// Word-wrap long names.
|
// Word-wrap long names.
|
||||||
val title = when(enemy) {
|
val title = when (enemy) {
|
||||||
NpcType.Gigobooma -> "Gigo-\nbooma"
|
NpcType.Gigobooma -> "Gigo-\nbooma"
|
||||||
NpcType.Shambertin -> "Shamber-\ntin"
|
NpcType.Shambertin -> "Shamber-\ntin"
|
||||||
else -> enemy.simpleName
|
else -> enemy.simpleName
|
||||||
@ -51,6 +52,7 @@ class MethodsForEpisodeController(
|
|||||||
headerClassName = "pw-hunt-optimizer-methods-for-episode-header-cell",
|
headerClassName = "pw-hunt-optimizer-methods-for-episode-header-cell",
|
||||||
className = "pw-hunt-optimizer-methods-for-episode-cell",
|
className = "pw-hunt-optimizer-methods-for-episode-cell",
|
||||||
sortable = true,
|
sortable = true,
|
||||||
|
textAlign = "right",
|
||||||
)
|
)
|
||||||
}.toTypedArray()
|
}.toTypedArray()
|
||||||
)
|
)
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -5,10 +5,6 @@ import world.phantasmal.observable.value.list.ListVal
|
|||||||
class Column<T>(
|
class Column<T>(
|
||||||
val key: String,
|
val key: String,
|
||||||
val title: String,
|
val title: String,
|
||||||
/**
|
|
||||||
* Whether the column stays in place while scrolling.
|
|
||||||
*/
|
|
||||||
val fixed: Boolean = false,
|
|
||||||
val width: Int,
|
val width: Int,
|
||||||
/**
|
/**
|
||||||
* Whether cells in this column contain an input widget.
|
* Whether cells in this column contain an input widget.
|
||||||
@ -18,6 +14,7 @@ class Column<T>(
|
|||||||
val sortable: Boolean = false,
|
val sortable: Boolean = false,
|
||||||
val headerClassName: String? = null,
|
val headerClassName: String? = null,
|
||||||
val className: String? = null,
|
val className: String? = null,
|
||||||
|
val textAlign: String? = null,
|
||||||
)
|
)
|
||||||
|
|
||||||
enum class SortDirection {
|
enum class SortDirection {
|
||||||
@ -33,8 +30,13 @@ 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.
|
||||||
|
*/
|
||||||
|
open val fixedColumns: Int = 0
|
||||||
|
|
||||||
abstract val values: ListVal<T>
|
abstract val values: ListVal<T>
|
||||||
abstract val columns: List<Column<T>>
|
abstract val columns: ListVal<Column<T>>
|
||||||
|
|
||||||
open fun sort(sortColumns: List<SortColumn<T>>) {
|
open fun sort(sortColumns: List<SortColumn<T>>) {
|
||||||
error("Not sortable.")
|
error("Not sortable.")
|
||||||
|
@ -176,7 +176,7 @@ fun <T> bindDisposableChildrenTo(
|
|||||||
},
|
},
|
||||||
childrenRemoved = {
|
childrenRemoved = {
|
||||||
disposer.disposeAll()
|
disposer.disposeAll()
|
||||||
}
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
disposable {
|
disposable {
|
||||||
@ -189,6 +189,7 @@ fun <T> bindChildrenTo(
|
|||||||
parent: Element,
|
parent: Element,
|
||||||
list: ListVal<T>,
|
list: ListVal<T>,
|
||||||
createChild: Node.(T, index: Int) -> Node,
|
createChild: Node.(T, index: Int) -> Node,
|
||||||
|
after: (ListValChangeEvent<T>) -> Unit = {},
|
||||||
): Disposable =
|
): Disposable =
|
||||||
bindChildrenTo(
|
bindChildrenTo(
|
||||||
parent,
|
parent,
|
||||||
@ -196,13 +197,15 @@ fun <T> bindChildrenTo(
|
|||||||
createChild,
|
createChild,
|
||||||
childrenRemoved = { _, _ ->
|
childrenRemoved = { _, _ ->
|
||||||
// Do nothing.
|
// Do nothing.
|
||||||
}
|
},
|
||||||
|
after,
|
||||||
)
|
)
|
||||||
|
|
||||||
fun <T> bindDisposableChildrenTo(
|
fun <T> bindDisposableChildrenTo(
|
||||||
parent: Element,
|
parent: Element,
|
||||||
list: ListVal<T>,
|
list: ListVal<T>,
|
||||||
createChild: Node.(T, index: Int) -> Pair<Node, Disposable>,
|
createChild: Node.(T, index: Int) -> Pair<Node, Disposable>,
|
||||||
|
after: (ListValChangeEvent<T>) -> Unit = {},
|
||||||
): Disposable {
|
): Disposable {
|
||||||
val disposer = Disposer()
|
val disposer = Disposer()
|
||||||
|
|
||||||
@ -216,7 +219,8 @@ fun <T> bindDisposableChildrenTo(
|
|||||||
},
|
},
|
||||||
childrenRemoved = { index, count ->
|
childrenRemoved = { index, count ->
|
||||||
disposer.removeAt(index, count)
|
disposer.removeAt(index, count)
|
||||||
}
|
},
|
||||||
|
after,
|
||||||
)
|
)
|
||||||
|
|
||||||
return disposable {
|
return disposable {
|
||||||
@ -249,6 +253,7 @@ private fun <T> bindChildrenTo(
|
|||||||
list: ListVal<T>,
|
list: ListVal<T>,
|
||||||
createChild: Node.(T, index: Int) -> Node,
|
createChild: Node.(T, index: Int) -> Node,
|
||||||
childrenRemoved: (index: Int, count: Int) -> Unit,
|
childrenRemoved: (index: Int, count: Int) -> Unit,
|
||||||
|
after: (ListValChangeEvent<T>) -> Unit,
|
||||||
): Disposable =
|
): Disposable =
|
||||||
list.observeList(callNow = true) { change: ListValChangeEvent<T> ->
|
list.observeList(callNow = true) { change: ListValChangeEvent<T> ->
|
||||||
if (change is ListValChangeEvent.Change) {
|
if (change is ListValChangeEvent.Change) {
|
||||||
@ -270,4 +275,6 @@ private fun <T> bindChildrenTo(
|
|||||||
parent.insertBefore(frag, parent.childNodes[change.index])
|
parent.insertBefore(frag, parent.childNodes[change.index])
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
after(change)
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
package world.phantasmal.webui.widgets
|
package world.phantasmal.webui.widgets
|
||||||
|
|
||||||
import org.w3c.dom.Node
|
import org.w3c.dom.*
|
||||||
import world.phantasmal.core.disposable.Disposer
|
import world.phantasmal.core.disposable.Disposer
|
||||||
import world.phantasmal.observable.value.Val
|
import world.phantasmal.observable.value.Val
|
||||||
import world.phantasmal.observable.value.trueVal
|
import world.phantasmal.observable.value.trueVal
|
||||||
@ -25,33 +25,16 @@ class Table<T>(
|
|||||||
tr {
|
tr {
|
||||||
className = "pw-table-row pw-table-header-row"
|
className = "pw-table-row pw-table-header-row"
|
||||||
|
|
||||||
var runningWidth = 0
|
addDisposable(bindChildrenTo(
|
||||||
|
this,
|
||||||
for (column in ctrl.columns) {
|
ctrl.columns,
|
||||||
th {
|
createChild = { column, _ ->
|
||||||
className = "pw-table-cell"
|
createHeaderRowCell(column)
|
||||||
|
},
|
||||||
column.headerClassName?.let { classList.add(it) }
|
after = {
|
||||||
|
positionFixedColumns(row = this, headerRow = true)
|
||||||
textContent = column.title
|
},
|
||||||
|
))
|
||||||
if (column.fixed) {
|
|
||||||
style.position = "sticky"
|
|
||||||
style.left = "${runningWidth}px"
|
|
||||||
runningWidth += column.width
|
|
||||||
}
|
|
||||||
|
|
||||||
style.width = "${column.width}px"
|
|
||||||
|
|
||||||
if (column.sortable) {
|
|
||||||
onmousedown = { e ->
|
|
||||||
if (e.buttons.toInt() == 1) {
|
|
||||||
ctrl.sortByColumn(column)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
tbody {
|
tbody {
|
||||||
@ -61,38 +44,16 @@ class Table<T>(
|
|||||||
val row = tr {
|
val row = tr {
|
||||||
className = "pw-table-row"
|
className = "pw-table-row"
|
||||||
|
|
||||||
var runningWidth = 0
|
addDisposable(bindChildrenTo(
|
||||||
|
this,
|
||||||
for (column in ctrl.columns) {
|
ctrl.columns,
|
||||||
(if (column.fixed) ::th else ::td) {
|
createChild = { column, _ ->
|
||||||
className = "pw-table-cell pw-table-body-cell"
|
createRowCell(column, value, rowDisposer)
|
||||||
|
},
|
||||||
column.className?.let { classList.add(it) }
|
after = {
|
||||||
|
positionFixedColumns(row = this, headerRow = false)
|
||||||
val child = renderCell(value, column)
|
},
|
||||||
|
))
|
||||||
if (child is Widget) {
|
|
||||||
rowDisposer.add(child)
|
|
||||||
append(child.element)
|
|
||||||
} else {
|
|
||||||
append(child)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (column.input) {
|
|
||||||
classList.add("pw-table-cell-input")
|
|
||||||
}
|
|
||||||
|
|
||||||
if (column.fixed) {
|
|
||||||
classList.add("pw-table-cell-fixed")
|
|
||||||
style.left = "${runningWidth}px"
|
|
||||||
runningWidth += column.width
|
|
||||||
}
|
|
||||||
|
|
||||||
style.width = "${column.width}px"
|
|
||||||
|
|
||||||
column.tooltip?.let { title = it(value) }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Pair(row, rowDisposer)
|
Pair(row, rowDisposer)
|
||||||
@ -100,6 +61,77 @@ class Table<T>(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun Node.createHeaderRowCell(column: Column<T>): HTMLTableCellElement =
|
||||||
|
th {
|
||||||
|
className = "pw-table-cell"
|
||||||
|
|
||||||
|
column.headerClassName?.let { classList.add(it) }
|
||||||
|
|
||||||
|
textContent = column.title
|
||||||
|
|
||||||
|
style.width = "${column.width}px"
|
||||||
|
|
||||||
|
if (column.sortable) {
|
||||||
|
onmousedown = { e ->
|
||||||
|
if (e.buttons.toInt() == 1) {
|
||||||
|
ctrl.sortByColumn(column)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun Node.createRowCell(
|
||||||
|
column: Column<T>,
|
||||||
|
value: T,
|
||||||
|
rowDisposer: Disposer,
|
||||||
|
): HTMLTableCellElement =
|
||||||
|
td {
|
||||||
|
className = "pw-table-cell pw-table-body-cell"
|
||||||
|
|
||||||
|
column.className?.let { classList.add(it) }
|
||||||
|
|
||||||
|
val child = renderCell(value, column)
|
||||||
|
|
||||||
|
if (child is Widget) {
|
||||||
|
rowDisposer.add(child)
|
||||||
|
append(child.element)
|
||||||
|
} else {
|
||||||
|
append(child)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (column.input) {
|
||||||
|
classList.add("pw-table-cell-input")
|
||||||
|
}
|
||||||
|
|
||||||
|
style.width = "${column.width}px"
|
||||||
|
|
||||||
|
column.tooltip?.let { title = it(value) }
|
||||||
|
|
||||||
|
column.textAlign?.let { style.textAlign = it }
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun positionFixedColumns(row: HTMLTableRowElement, headerRow: Boolean) {
|
||||||
|
val columns = ctrl.columns.value
|
||||||
|
var left = 0
|
||||||
|
|
||||||
|
for (index in columns.indices) {
|
||||||
|
val el = row.children[index].unsafeCast<HTMLElement>()
|
||||||
|
|
||||||
|
if (index < ctrl.fixedColumns) {
|
||||||
|
el.style.position = "sticky"
|
||||||
|
el.style.left = "${left}px"
|
||||||
|
|
||||||
|
if (!headerRow) {
|
||||||
|
el.classList.add("pw-table-cell-fixed")
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
el.style.position = ""
|
||||||
|
}
|
||||||
|
|
||||||
|
left += columns[index].width
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
init {
|
init {
|
||||||
@Suppress("CssUnresolvedCustomProperty", "CssUnusedSymbol")
|
@Suppress("CssUnresolvedCustomProperty", "CssUnusedSymbol")
|
||||||
@ -164,6 +196,7 @@ class Table<T>(
|
|||||||
}
|
}
|
||||||
|
|
||||||
.pw-table-cell-fixed {
|
.pw-table-cell-fixed {
|
||||||
|
font-weight: bold;
|
||||||
position: sticky;
|
position: sticky;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user