mirror of
https://github.com/DaanVandenBosch/phantasmal-world.git
synced 2025-04-05 07:18:29 +08:00
Fixed an issue with the way CSS was added to the DOM. Moved @AfterTest checks to utility function so test failures aren't hidden by @AfterTest failures.
This commit is contained in:
parent
fdb3d5bbb6
commit
e6d6f292f4
@ -3,7 +3,6 @@ package world.phantasmal.lib.compression.prs
|
|||||||
import world.phantasmal.lib.buffer.Buffer
|
import world.phantasmal.lib.buffer.Buffer
|
||||||
import world.phantasmal.lib.cursor.cursor
|
import world.phantasmal.lib.cursor.cursor
|
||||||
import kotlin.random.Random
|
import kotlin.random.Random
|
||||||
import kotlin.random.nextUInt
|
|
||||||
import kotlin.test.Test
|
import kotlin.test.Test
|
||||||
import kotlin.test.assertEquals
|
import kotlin.test.assertEquals
|
||||||
|
|
||||||
|
@ -16,7 +16,7 @@ class FlatTransformedVal<T>(
|
|||||||
return if (hasNoObservers()) {
|
return if (hasNoObservers()) {
|
||||||
super.value
|
super.value
|
||||||
} else {
|
} else {
|
||||||
computedVal.unsafeToNonNull<Val<T>>().value
|
computedVal.unsafeToNonNull().value
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -14,7 +14,7 @@ abstract class ObservableTests : TestSuite() {
|
|||||||
protected abstract fun create(): ObservableAndEmit
|
protected abstract fun create(): ObservableAndEmit
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun observable_calls_observers_when_events_are_emitted() {
|
fun observable_calls_observers_when_events_are_emitted() = test {
|
||||||
val (observable, emit) = create()
|
val (observable, emit) = create()
|
||||||
var changes = 0
|
var changes = 0
|
||||||
|
|
||||||
@ -36,7 +36,7 @@ abstract class ObservableTests : TestSuite() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun observable_does_not_call_observers_after_they_are_disposed() {
|
fun observable_does_not_call_observers_after_they_are_disposed() = test {
|
||||||
val (observable, emit) = create()
|
val (observable, emit) = create()
|
||||||
var changes = 0
|
var changes = 0
|
||||||
|
|
||||||
|
@ -13,7 +13,7 @@ class FlatTransformedValDependentValEmitsTests : RegularValTests() {
|
|||||||
* same.
|
* same.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
fun emits_a_change_when_its_direct_val_dependency_changes() {
|
fun emits_a_change_when_its_direct_val_dependency_changes() = test {
|
||||||
val v = SimpleVal(SimpleVal(7))
|
val v = SimpleVal(SimpleVal(7))
|
||||||
val fv = FlatTransformedVal(listOf(v)) { v.value }
|
val fv = FlatTransformedVal(listOf(v)) { v.value }
|
||||||
var observedValue: Int? = null
|
var observedValue: Int? = null
|
||||||
|
@ -13,7 +13,7 @@ abstract class RegularValTests : ValTests() {
|
|||||||
protected abstract fun createBoolean(bool: Boolean): ValAndEmit<Boolean>
|
protected abstract fun createBoolean(bool: Boolean): ValAndEmit<Boolean>
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun val_boolean_extensions() {
|
fun val_boolean_extensions() = test {
|
||||||
listOf(true, false).forEach { bool ->
|
listOf(true, false).forEach { bool ->
|
||||||
val (value) = createBoolean(bool)
|
val (value) = createBoolean(bool)
|
||||||
|
|
||||||
|
@ -5,7 +5,7 @@ import kotlin.test.Test
|
|||||||
|
|
||||||
class StaticValTests : TestSuite() {
|
class StaticValTests : TestSuite() {
|
||||||
@Test
|
@Test
|
||||||
fun observing_StaticVal_should_never_create_leaks() {
|
fun observing_StaticVal_should_never_create_leaks() = test {
|
||||||
val static = StaticVal("test value")
|
val static = StaticVal("test value")
|
||||||
|
|
||||||
static.observe {}
|
static.observe {}
|
||||||
|
@ -5,27 +5,27 @@ import kotlin.test.*
|
|||||||
|
|
||||||
class ValCreationTests : TestSuite() {
|
class ValCreationTests : TestSuite() {
|
||||||
@Test
|
@Test
|
||||||
fun test_value() {
|
fun test_value() = test {
|
||||||
assertEquals(7, value(7).value)
|
assertEquals(7, value(7).value)
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun test_trueVal() {
|
fun test_trueVal() = test {
|
||||||
assertTrue(trueVal().value)
|
assertTrue(trueVal().value)
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun test_falseVal() {
|
fun test_falseVal() = test {
|
||||||
assertFalse(falseVal().value)
|
assertFalse(falseVal().value)
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun test_nullVal() {
|
fun test_nullVal() = test {
|
||||||
assertNull(nullVal().value)
|
assertNull(nullVal().value)
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun test_mutableVal_with_initial_value() {
|
fun test_mutableVal_with_initial_value() = test {
|
||||||
val v = mutableVal(17)
|
val v = mutableVal(17)
|
||||||
|
|
||||||
assertEquals(17, v.value)
|
assertEquals(17, v.value)
|
||||||
@ -36,7 +36,7 @@ class ValCreationTests : TestSuite() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun test_mutableVal_with_getter_and_setter() {
|
fun test_mutableVal_with_getter_and_setter() = test {
|
||||||
var x = 17
|
var x = 17
|
||||||
val v = mutableVal({ x }, { x = it })
|
val v = mutableVal({ x }, { x = it })
|
||||||
|
|
||||||
|
@ -19,7 +19,7 @@ abstract class ValTests : ObservableTests() {
|
|||||||
* Otherwise it should only call the observer when it changes.
|
* Otherwise it should only call the observer when it changes.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
fun val_respects_call_now_argument() {
|
fun val_respects_call_now_argument() = test {
|
||||||
val (value, emit) = create()
|
val (value, emit) = create()
|
||||||
var changes = 0
|
var changes = 0
|
||||||
|
|
||||||
|
@ -14,7 +14,7 @@ abstract class ListValTests : ValTests() {
|
|||||||
abstract override fun create(): ListValAndAdd
|
abstract override fun create(): ListValAndAdd
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun listVal_updates_sizeVal_correctly() {
|
fun listVal_updates_sizeVal_correctly() = test {
|
||||||
val (list: List<*>, add) = create()
|
val (list: List<*>, add) = create()
|
||||||
|
|
||||||
assertEquals(0, list.sizeVal.value)
|
assertEquals(0, list.sizeVal.value)
|
||||||
|
@ -2,6 +2,8 @@ plugins {
|
|||||||
kotlin("multiplatform")
|
kotlin("multiplatform")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
val coroutinesVersion: String by project.ext
|
||||||
|
|
||||||
kotlin {
|
kotlin {
|
||||||
js {
|
js {
|
||||||
browser {}
|
browser {}
|
||||||
@ -13,10 +15,11 @@ kotlin {
|
|||||||
api(project(":core"))
|
api(project(":core"))
|
||||||
api(kotlin("test-common"))
|
api(kotlin("test-common"))
|
||||||
api(kotlin("test-annotations-common"))
|
api(kotlin("test-annotations-common"))
|
||||||
|
api("org.jetbrains.kotlinx:kotlinx-coroutines-core:$coroutinesVersion")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
val jsMain by getting {
|
named("jsMain") {
|
||||||
dependencies {
|
dependencies {
|
||||||
api(kotlin("test-js"))
|
api(kotlin("test-js"))
|
||||||
}
|
}
|
||||||
|
@ -4,31 +4,23 @@ import kotlinx.coroutines.CoroutineScope
|
|||||||
import kotlinx.coroutines.Job
|
import kotlinx.coroutines.Job
|
||||||
import world.phantasmal.core.disposable.Disposer
|
import world.phantasmal.core.disposable.Disposer
|
||||||
import world.phantasmal.core.disposable.TrackedDisposable
|
import world.phantasmal.core.disposable.TrackedDisposable
|
||||||
import kotlin.test.AfterTest
|
|
||||||
import kotlin.test.BeforeTest
|
|
||||||
import kotlin.test.assertEquals
|
import kotlin.test.assertEquals
|
||||||
|
|
||||||
abstract class TestSuite {
|
abstract class TestSuite {
|
||||||
private var initialDisposableCount: Int = 0
|
fun test(block: TestContext.() -> Unit) {
|
||||||
private var _disposer: Disposer? = null
|
val initialDisposableCount = TrackedDisposable.disposableCount
|
||||||
|
val disposer = Disposer()
|
||||||
|
|
||||||
protected val disposer: Disposer get() = _disposer!!
|
block(TestContext(disposer))
|
||||||
|
|
||||||
protected val scope: CoroutineScope = object : CoroutineScope {
|
|
||||||
override val coroutineContext = Job()
|
|
||||||
}
|
|
||||||
|
|
||||||
@BeforeTest
|
|
||||||
fun before() {
|
|
||||||
initialDisposableCount = TrackedDisposable.disposableCount
|
|
||||||
_disposer = Disposer()
|
|
||||||
}
|
|
||||||
|
|
||||||
@AfterTest
|
|
||||||
fun after() {
|
|
||||||
_disposer!!.dispose()
|
|
||||||
|
|
||||||
|
disposer.dispose()
|
||||||
val leakCount = TrackedDisposable.disposableCount - initialDisposableCount
|
val leakCount = TrackedDisposable.disposableCount - initialDisposableCount
|
||||||
assertEquals(0, leakCount, "TrackedDisposables were leaked")
|
assertEquals(0, leakCount, "TrackedDisposables were leaked")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class TestContext(val disposer: Disposer) {
|
||||||
|
val scope: CoroutineScope = object : CoroutineScope {
|
||||||
|
override val coroutineContext = Job()
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -9,18 +9,21 @@ class ApplicationWidget(
|
|||||||
scope: CoroutineScope,
|
scope: CoroutineScope,
|
||||||
private val navigationWidget: NavigationWidget,
|
private val navigationWidget: NavigationWidget,
|
||||||
private val mainContentWidget: MainContentWidget,
|
private val mainContentWidget: MainContentWidget,
|
||||||
) : Widget(scope, listOf(::style)) {
|
) : Widget(scope) {
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
div(className = "pw-application-application") {
|
div {
|
||||||
|
className = "pw-application-application"
|
||||||
|
|
||||||
addChild(navigationWidget)
|
addChild(navigationWidget)
|
||||||
addChild(mainContentWidget)
|
addChild(mainContentWidget)
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnusedSymbol")
|
companion object {
|
||||||
// language=css
|
init {
|
||||||
private fun style() = """
|
@Suppress("CssUnusedSymbol")
|
||||||
.pw-application-application {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-application-application {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
top: 0;
|
top: 0;
|
||||||
bottom: 0;
|
bottom: 0;
|
||||||
@ -28,9 +31,12 @@ private fun style() = """
|
|||||||
right: 0;
|
right: 0;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
}
|
}
|
||||||
.pw-application-application .pw-application-main-content {
|
.pw-application-application .pw-application-main-content {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -13,27 +13,33 @@ class MainContentWidget(
|
|||||||
scope: CoroutineScope,
|
scope: CoroutineScope,
|
||||||
private val ctrl: MainContentController,
|
private val ctrl: MainContentController,
|
||||||
private val toolViews: Map<PwTool, (CoroutineScope) -> Widget>,
|
private val toolViews: Map<PwTool, (CoroutineScope) -> Widget>,
|
||||||
) : Widget(scope, listOf(::style)) {
|
) : Widget(scope) {
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
div(className = "pw-application-main-content") {
|
div {
|
||||||
|
className = "pw-application-main-content"
|
||||||
|
|
||||||
ctrl.tools.forEach { (tool, active) ->
|
ctrl.tools.forEach { (tool, active) ->
|
||||||
toolViews[tool]?.let { createWidget ->
|
toolViews[tool]?.let { createWidget ->
|
||||||
addChild(LazyLoader(scope, hidden = !active, createWidget = createWidget))
|
addChild(LazyLoader(scope, hidden = !active, createWidget = createWidget))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnusedSymbol")
|
companion object {
|
||||||
// language=css
|
init {
|
||||||
private fun style() = """
|
@Suppress("CssUnusedSymbol")
|
||||||
.pw-application-main-content {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-application-main-content {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-application-main-content > * {
|
.pw-application-main-content > * {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -7,46 +7,49 @@ import world.phantasmal.webui.dom.div
|
|||||||
import world.phantasmal.webui.widgets.Widget
|
import world.phantasmal.webui.widgets.Widget
|
||||||
|
|
||||||
class NavigationWidget(scope: CoroutineScope, private val ctrl: NavigationController) :
|
class NavigationWidget(scope: CoroutineScope, private val ctrl: NavigationController) :
|
||||||
Widget(scope, listOf(::style)) {
|
Widget(scope) {
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
div(className = "pw-application-navigation") {
|
div {
|
||||||
|
className = "pw-application-navigation"
|
||||||
|
|
||||||
ctrl.tools.forEach { (tool, active) ->
|
ctrl.tools.forEach { (tool, active) ->
|
||||||
addChild(PwToolButton(scope, tool, active) { ctrl.setCurrentTool(tool) })
|
addChild(PwToolButton(scope, tool, active) { ctrl.setCurrentTool(tool) })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnusedSymbol", "CssUnresolvedCustomProperty")
|
companion object {
|
||||||
// language=css
|
init {
|
||||||
private fun style() = """
|
@Suppress("CssUnusedSymbol", "CssUnresolvedCustomProperty")
|
||||||
.pw-application-navigation {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-application-navigation {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
align-items: stretch;
|
align-items: stretch;
|
||||||
background-color: hsl(0, 0%, 10%);
|
background-color: hsl(0, 0%, 10%);
|
||||||
border-bottom: solid 2px var(--pw-bg-color);
|
border-bottom: solid 2px var(--pw-bg-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-application-navigation-spacer {
|
.pw-application-navigation-spacer {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-application-navigation-server {
|
.pw-application-navigation-server {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-application-navigation-server > * {
|
.pw-application-navigation-server > * {
|
||||||
margin: 0 2px;
|
margin: 0 2px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-application-navigation-time {
|
.pw-application-navigation-time {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-application-navigation-github {
|
.pw-application-navigation-github {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
@ -54,9 +57,12 @@ private fun style() = """
|
|||||||
width: 30px;
|
width: 30px;
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
color: var(--pw-control-text-color);
|
color: var(--pw-control-text-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-application-navigation-github:hover {
|
.pw-application-navigation-github:hover {
|
||||||
color: var(--pw-control-text-color-hover);
|
color: var(--pw-control-text-color-hover);
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -14,30 +14,36 @@ class PwToolButton(
|
|||||||
private val tool: PwTool,
|
private val tool: PwTool,
|
||||||
private val toggled: Observable<Boolean>,
|
private val toggled: Observable<Boolean>,
|
||||||
private val mouseDown: () -> Unit,
|
private val mouseDown: () -> Unit,
|
||||||
) : Control(scope, listOf(::style)) {
|
) : Control(scope) {
|
||||||
private val inputId = "pw-application-pw-tool-button-${tool.name.toLowerCase()}"
|
private val inputId = "pw-application-pw-tool-button-${tool.name.toLowerCase()}"
|
||||||
|
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
span(className = "pw-application-pw-tool-button") {
|
span {
|
||||||
input(type = "radio", id = inputId) {
|
className = "pw-application-pw-tool-button"
|
||||||
|
|
||||||
|
input {
|
||||||
|
type = "radio"
|
||||||
|
id = inputId
|
||||||
name = "pw-application-pw-tool-button"
|
name = "pw-application-pw-tool-button"
|
||||||
observe(toggled) { checked = it }
|
observe(toggled) { checked = it }
|
||||||
}
|
}
|
||||||
label(htmlFor = inputId) {
|
label {
|
||||||
|
htmlFor = inputId
|
||||||
textContent = tool.uiName
|
textContent = tool.uiName
|
||||||
onmousedown = { mouseDown() }
|
onmousedown = { mouseDown() }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnresolvedCustomProperty")
|
companion object {
|
||||||
// language=css
|
init {
|
||||||
private fun style() = """
|
@Suppress("CssUnresolvedCustomProperty")
|
||||||
.pw-application-pw-tool-button input {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-application-pw-tool-button input {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-application-pw-tool-button label {
|
.pw-application-pw-tool-button label {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
@ -46,15 +52,18 @@ private fun style() = """
|
|||||||
height: 100%;
|
height: 100%;
|
||||||
padding: 0 20px;
|
padding: 0 20px;
|
||||||
color: hsl(0, 0%, 65%);
|
color: hsl(0, 0%, 65%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-application-pw-tool-button label:hover {
|
.pw-application-pw-tool-button label:hover {
|
||||||
color: hsl(0, 0%, 85%);
|
color: hsl(0, 0%, 85%);
|
||||||
background-color: hsl(0, 0%, 12%);
|
background-color: hsl(0, 0%, 12%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-application-pw-tool-button input:checked + label {
|
.pw-application-pw-tool-button input:checked + label {
|
||||||
color: hsl(0, 0%, 85%);
|
color: hsl(0, 0%, 85%);
|
||||||
background-color: var(--pw-bg-color);
|
background-color: var(--pw-bg-color);
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -46,7 +46,7 @@ class DockWidget(
|
|||||||
scope: CoroutineScope,
|
scope: CoroutineScope,
|
||||||
hidden: Val<Boolean> = falseVal(),
|
hidden: Val<Boolean> = falseVal(),
|
||||||
private val item: DockedItem,
|
private val item: DockedItem,
|
||||||
) : Widget(scope, listOf(::style), hidden) {
|
) : Widget(scope, hidden) {
|
||||||
private lateinit var goldenLayout: GoldenLayout
|
private lateinit var goldenLayout: GoldenLayout
|
||||||
|
|
||||||
init {
|
init {
|
||||||
@ -56,7 +56,9 @@ class DockWidget(
|
|||||||
}
|
}
|
||||||
|
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
div(className = "pw-core-dock") {
|
div {
|
||||||
|
className = "pw-core-dock"
|
||||||
|
|
||||||
val idToCreate = mutableMapOf<String, (CoroutineScope) -> Widget>()
|
val idToCreate = mutableMapOf<String, (CoroutineScope) -> Widget>()
|
||||||
|
|
||||||
val config = newJsObject<GoldenLayout.Config> {
|
val config = newJsObject<GoldenLayout.Config> {
|
||||||
@ -141,29 +143,30 @@ class DockWidget(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
// Use #pw-root for higher specificity than the default GoldenLayout CSS.
|
companion object {
|
||||||
@Suppress("CssUnusedSymbol", "CssUnresolvedCustomProperty")
|
init {
|
||||||
// language=css
|
// Use #pw-root for higher specificity than the default GoldenLayout CSS.
|
||||||
private fun style() = """
|
@Suppress("CssUnusedSymbol", "CssUnresolvedCustomProperty")
|
||||||
.pw-core-dock {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-core-dock {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
#pw-root .lm_header {
|
#pw-root .lm_header {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
height: ${HEADER_HEIGHT + 4}px;
|
height: ${HEADER_HEIGHT + 4}px;
|
||||||
padding: 3px 0 0 0;
|
padding: 3px 0 0 0;
|
||||||
border-bottom: var(--pw-border);
|
border-bottom: var(--pw-border);
|
||||||
}
|
}
|
||||||
|
|
||||||
#pw-root .lm_header .lm_tabs {
|
#pw-root .lm_header .lm_tabs {
|
||||||
padding: 0 3px;
|
padding: 0 3px;
|
||||||
}
|
}
|
||||||
|
|
||||||
#pw-root .lm_header .lm_tabs .lm_tab {
|
#pw-root .lm_header .lm_tabs .lm_tab {
|
||||||
cursor: default;
|
cursor: default;
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
@ -174,24 +177,24 @@ private fun style() = """
|
|||||||
background-color: hsl(0, 0%, 12%);
|
background-color: hsl(0, 0%, 12%);
|
||||||
color: hsl(0, 0%, 75%);
|
color: hsl(0, 0%, 75%);
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
}
|
}
|
||||||
|
|
||||||
#pw-root .lm_header .lm_tabs .lm_tab:hover {
|
#pw-root .lm_header .lm_tabs .lm_tab:hover {
|
||||||
background-color: hsl(0, 0%, 18%);
|
background-color: hsl(0, 0%, 18%);
|
||||||
color: hsl(0, 0%, 85%);
|
color: hsl(0, 0%, 85%);
|
||||||
}
|
}
|
||||||
|
|
||||||
#pw-root .lm_header .lm_tabs .lm_tab.lm_active {
|
#pw-root .lm_header .lm_tabs .lm_tab.lm_active {
|
||||||
background-color: var(--pw-bg-color);
|
background-color: var(--pw-bg-color);
|
||||||
color: hsl(0, 0%, 90%);
|
color: hsl(0, 0%, 90%);
|
||||||
border-bottom-color: var(--pw-bg-color);
|
border-bottom-color: var(--pw-bg-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
#pw-root .lm_header .lm_controls > li {
|
#pw-root .lm_header .lm_controls > li {
|
||||||
cursor: default;
|
cursor: default;
|
||||||
}
|
}
|
||||||
|
|
||||||
#pw-root .lm_header .lm_controls .lm_close {
|
#pw-root .lm_header .lm_controls .lm_close {
|
||||||
/* a white 9x9 X shape */
|
/* a white 9x9 X shape */
|
||||||
background-image: url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAkAAAAJCAYAAADgkQYQAAAAQUlEQVR4nHXOQQ4AMAgCQeT/f6aXpsGK3jSTuCVJAAr7iBdoAwCKd0nwfaAdHbYERw5b44+E8JoBjEYGMBq5gAYP3usUDu2IvoUAAAAASUVORK5CYII=);
|
background-image: url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAkAAAAJCAYAAADgkQYQAAAAQUlEQVR4nHXOQQ4AMAgCQeT/f6aXpsGK3jSTuCVJAAr7iBdoAwCKd0nwfaAdHbYERw5b44+E8JoBjEYGMBq5gAYP3usUDu2IvoUAAAAASUVORK5CYII=);
|
||||||
background-position: center center;
|
background-position: center center;
|
||||||
@ -199,43 +202,46 @@ private fun style() = """
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
opacity: 0.4;
|
opacity: 0.4;
|
||||||
transition: opacity 300ms ease;
|
transition: opacity 300ms ease;
|
||||||
}
|
}
|
||||||
|
|
||||||
#pw-root .lm_header .lm_controls .lm_close:hover {
|
#pw-root .lm_header .lm_controls .lm_close:hover {
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
#pw-root .lm_content > * {
|
#pw-root .lm_content > * {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
/* Subtract HEADER_HEIGHT_DIFF px as workaround for bug related to headerHeight. */
|
/* Subtract HEADER_HEIGHT_DIFF px as workaround for bug related to headerHeight. */
|
||||||
height: calc(100% - ${HEADER_HEIGHT_DIFF}px);
|
height: calc(100% - ${HEADER_HEIGHT_DIFF}px);
|
||||||
}
|
}
|
||||||
|
|
||||||
#pw-root .lm_splitter {
|
#pw-root .lm_splitter {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
background-color: hsl(0, 0%, 20%);
|
background-color: hsl(0, 0%, 20%);
|
||||||
}
|
}
|
||||||
|
|
||||||
#pw-root .lm_splitter.lm_vertical {
|
#pw-root .lm_splitter.lm_vertical {
|
||||||
border-top: var(--pw-border);
|
border-top: var(--pw-border);
|
||||||
border-bottom: var(--pw-border);
|
border-bottom: var(--pw-border);
|
||||||
}
|
}
|
||||||
|
|
||||||
#pw-root .lm_splitter.lm_horizontal {
|
#pw-root .lm_splitter.lm_horizontal {
|
||||||
border-left: var(--pw-border);
|
border-left: var(--pw-border);
|
||||||
border-right: var(--pw-border);
|
border-right: var(--pw-border);
|
||||||
}
|
}
|
||||||
|
|
||||||
#pw-root .lm_dragProxy > .lm_content {
|
#pw-root .lm_dragProxy > .lm_content {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
background-color: var(--pw-bg-color);
|
background-color: var(--pw-bg-color);
|
||||||
border-left: var(--pw-border);
|
border-left: var(--pw-border);
|
||||||
border-right: var(--pw-border);
|
border-right: var(--pw-border);
|
||||||
border-bottom: var(--pw-border);
|
border-bottom: var(--pw-border);
|
||||||
}
|
}
|
||||||
|
|
||||||
#pw-root .lm_dropTargetIndicator {
|
#pw-root .lm_dropTargetIndicator {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
background-color: hsla(0, 0%, 50%, 0.2);
|
background-color: hsla(0, 0%, 50%, 0.2);
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -12,9 +12,11 @@ import kotlin.math.floor
|
|||||||
class RendererWidget(
|
class RendererWidget(
|
||||||
scope: CoroutineScope,
|
scope: CoroutineScope,
|
||||||
private val createEngine: (HTMLCanvasElement) -> Engine,
|
private val createEngine: (HTMLCanvasElement) -> Engine,
|
||||||
) : Widget(scope, listOf(::style)) {
|
) : Widget(scope) {
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
canvas(className = "pw-core-renderer") {
|
canvas {
|
||||||
|
className = "pw-core-renderer"
|
||||||
|
|
||||||
observeResize()
|
observeResize()
|
||||||
addDisposable(QuestRenderer(this, createEngine))
|
addDisposable(QuestRenderer(this, createEngine))
|
||||||
}
|
}
|
||||||
@ -24,13 +26,17 @@ class RendererWidget(
|
|||||||
canvas.width = floor(width).toInt()
|
canvas.width = floor(width).toInt()
|
||||||
canvas.height = floor(height).toInt()
|
canvas.height = floor(height).toInt()
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnusedSymbol")
|
companion object {
|
||||||
// language=css
|
init {
|
||||||
private fun style() = """
|
@Suppress("CssUnusedSymbol")
|
||||||
.pw-core-renderer {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-core-renderer {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -0,0 +1,39 @@
|
|||||||
|
package world.phantasmal.web.core.widgets
|
||||||
|
|
||||||
|
import kotlinx.coroutines.CoroutineScope
|
||||||
|
import org.w3c.dom.Node
|
||||||
|
import world.phantasmal.observable.value.Val
|
||||||
|
import world.phantasmal.observable.value.trueVal
|
||||||
|
import world.phantasmal.webui.dom.div
|
||||||
|
import world.phantasmal.webui.widgets.Label
|
||||||
|
import world.phantasmal.webui.widgets.Widget
|
||||||
|
|
||||||
|
class UnavailableWidget(
|
||||||
|
scope: CoroutineScope,
|
||||||
|
hidden: Val<Boolean>,
|
||||||
|
private val message: String,
|
||||||
|
) : Widget(scope, hidden) {
|
||||||
|
override fun Node.createElement() =
|
||||||
|
div {
|
||||||
|
className = "pw-core-unavailable"
|
||||||
|
|
||||||
|
addWidget(Label(scope, disabled = trueVal(), text = message))
|
||||||
|
}
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
init {
|
||||||
|
@Suppress("CssUnusedSymbol")
|
||||||
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-core-unavailable {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -6,9 +6,11 @@ import world.phantasmal.webui.dom.div
|
|||||||
import world.phantasmal.webui.dom.p
|
import world.phantasmal.webui.dom.p
|
||||||
import world.phantasmal.webui.widgets.Widget
|
import world.phantasmal.webui.widgets.Widget
|
||||||
|
|
||||||
class HelpWidget(scope: CoroutineScope) : Widget(scope, listOf(::style)) {
|
class HelpWidget(scope: CoroutineScope) : Widget(scope) {
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
div(className = "pw-hunt-optimizer-help") {
|
div {
|
||||||
|
className = "pw-hunt-optimizer-help"
|
||||||
|
|
||||||
p {
|
p {
|
||||||
textContent =
|
textContent =
|
||||||
"Add some items with the combo box on the left to see the optimal combination of hunt methods on the right."
|
"Add some items with the combo box on the left to see the optimal combination of hunt methods on the right."
|
||||||
@ -25,18 +27,22 @@ class HelpWidget(scope: CoroutineScope) : Widget(scope, listOf(::style)) {
|
|||||||
"The optimal result is calculated using linear optimization. The optimizer takes into account rare enemies and the fact that pan arms can be split in two."
|
"The optimal result is calculated using linear optimization. The optimizer takes into account rare enemies and the fact that pan arms can be split in two."
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnusedSymbol")
|
companion object {
|
||||||
// language=css
|
init {
|
||||||
private fun style() = """
|
@Suppress("CssUnusedSymbol")
|
||||||
.pw-hunt-optimizer-help {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-hunt-optimizer-help {
|
||||||
cursor: initial;
|
cursor: initial;
|
||||||
user-select: text;
|
user-select: text;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-hunt-optimizer-help p {
|
.pw-hunt-optimizer-help p {
|
||||||
margin: 1em;
|
margin: 1em;
|
||||||
max-width: 600px;
|
max-width: 600px;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -12,8 +12,11 @@ class HuntOptimizerWidget(
|
|||||||
scope: CoroutineScope,
|
scope: CoroutineScope,
|
||||||
private val ctrl: HuntOptimizerController,
|
private val ctrl: HuntOptimizerController,
|
||||||
private val createMethodsWidget: (CoroutineScope) -> MethodsWidget,
|
private val createMethodsWidget: (CoroutineScope) -> MethodsWidget,
|
||||||
) : Widget(scope, listOf(::style)) {
|
) : Widget(scope) {
|
||||||
override fun Node.createElement() = div(className = "pw-hunt-optimizer-hunt-optimizer") {
|
override fun Node.createElement() =
|
||||||
|
div {
|
||||||
|
className = "pw-hunt-optimizer-hunt-optimizer"
|
||||||
|
|
||||||
addChild(TabContainer(
|
addChild(TabContainer(
|
||||||
scope,
|
scope,
|
||||||
ctrl = ctrl,
|
ctrl = ctrl,
|
||||||
@ -31,18 +34,22 @@ class HuntOptimizerWidget(
|
|||||||
}
|
}
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnusedSymbol")
|
companion object {
|
||||||
// language=css
|
init {
|
||||||
private fun style() = """
|
@Suppress("CssUnusedSymbol")
|
||||||
.pw-hunt-optimizer-hunt-optimizer {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-hunt-optimizer-hunt-optimizer {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-hunt-optimizer-hunt-optimizer > * {
|
.pw-hunt-optimizer-hunt-optimizer > * {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -11,19 +11,25 @@ class MethodsForEpisodeWidget(
|
|||||||
scope: CoroutineScope,
|
scope: CoroutineScope,
|
||||||
private val ctrl: MethodsController,
|
private val ctrl: MethodsController,
|
||||||
private val episode: Episode,
|
private val episode: Episode,
|
||||||
) : Widget(scope, listOf(::style)) {
|
) : Widget(scope) {
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
div(className = "pw-hunt-optimizer-methods-for-episode") {
|
div {
|
||||||
|
className = "pw-hunt-optimizer-methods-for-episode"
|
||||||
|
|
||||||
bindChildrenTo(ctrl.episodeToMethods.getValue(episode)) { method, _ ->
|
bindChildrenTo(ctrl.episodeToMethods.getValue(episode)) { method, _ ->
|
||||||
div { textContent = method.name }
|
div { textContent = method.name }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnusedSymbol")
|
companion object {
|
||||||
// language=css
|
init {
|
||||||
private fun style() = """
|
@Suppress("CssUnusedSymbol")
|
||||||
.pw-hunt-optimizer-methods-for-episode {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-hunt-optimizer-methods-for-episode {
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -10,25 +10,31 @@ import world.phantasmal.webui.widgets.Widget
|
|||||||
class MethodsWidget(
|
class MethodsWidget(
|
||||||
scope: CoroutineScope,
|
scope: CoroutineScope,
|
||||||
private val ctrl: MethodsController,
|
private val ctrl: MethodsController,
|
||||||
) : Widget(scope, listOf(::style)) {
|
) : Widget(scope) {
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
div(className = "pw-hunt-optimizer-methods") {
|
div {
|
||||||
|
className = "pw-hunt-optimizer-methods"
|
||||||
|
|
||||||
addChild(TabContainer(scope, ctrl = ctrl, createWidget = { scope, tab ->
|
addChild(TabContainer(scope, ctrl = ctrl, createWidget = { scope, tab ->
|
||||||
MethodsForEpisodeWidget(scope, ctrl, tab.episode)
|
MethodsForEpisodeWidget(scope, ctrl, tab.episode)
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnusedSymbol")
|
companion object {
|
||||||
// language=css
|
init {
|
||||||
private fun style() = """
|
@Suppress("CssUnusedSymbol")
|
||||||
.pw-hunt-optimizer-methods {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-hunt-optimizer-methods {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-hunt-optimizer-methods > * {
|
.pw-hunt-optimizer-methods > * {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -7,5 +7,8 @@ import world.phantasmal.web.questEditor.stores.QuestEditorStore
|
|||||||
import world.phantasmal.webui.controllers.Controller
|
import world.phantasmal.webui.controllers.Controller
|
||||||
|
|
||||||
class QuestInfoController(scope: CoroutineScope, store: QuestEditorStore) : Controller(scope) {
|
class QuestInfoController(scope: CoroutineScope, store: QuestEditorStore) : Controller(scope) {
|
||||||
|
val unavailable = store.currentQuest.transform { it == null }
|
||||||
|
val episode: Val<String> = store.currentQuest.transform { it?.episode?.name ?: "" }
|
||||||
val id: Val<Int> = store.currentQuest.flatTransform { it?.id ?: value(0) }
|
val id: Val<Int> = store.currentQuest.flatTransform { it?.id ?: value(0) }
|
||||||
|
val name: Val<String> = store.currentQuest.flatTransform { it?.name ?: value("") }
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
package world.phantasmal.web.questEditor.models
|
package world.phantasmal.web.questEditor.models
|
||||||
|
|
||||||
|
import world.phantasmal.lib.fileFormats.quest.Episode
|
||||||
import world.phantasmal.observable.value.Val
|
import world.phantasmal.observable.value.Val
|
||||||
import world.phantasmal.observable.value.mutableVal
|
import world.phantasmal.observable.value.mutableVal
|
||||||
|
|
||||||
@ -9,6 +10,7 @@ class QuestModel(
|
|||||||
name: String,
|
name: String,
|
||||||
shortDescription: String,
|
shortDescription: String,
|
||||||
longDescription: String,
|
longDescription: String,
|
||||||
|
val episode: Episode,
|
||||||
) {
|
) {
|
||||||
private val _id = mutableVal(0)
|
private val _id = mutableVal(0)
|
||||||
private val _language = mutableVal(0)
|
private val _language = mutableVal(0)
|
||||||
|
@ -9,6 +9,7 @@ fun convertQuestToModel(quest: Quest): QuestModel {
|
|||||||
quest.language,
|
quest.language,
|
||||||
quest.name,
|
quest.name,
|
||||||
quest.shortDescription,
|
quest.shortDescription,
|
||||||
quest.longDescription
|
quest.longDescription,
|
||||||
|
quest.episode,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -12,7 +12,10 @@ class QuestEditorToolbar(
|
|||||||
scope: CoroutineScope,
|
scope: CoroutineScope,
|
||||||
private val ctrl: QuestEditorToolbarController,
|
private val ctrl: QuestEditorToolbarController,
|
||||||
) : Widget(scope) {
|
) : Widget(scope) {
|
||||||
override fun Node.createElement() = div(className = "pw-quest-editor-toolbar") {
|
override fun Node.createElement() =
|
||||||
|
div {
|
||||||
|
className = "pw-quest-editor-toolbar"
|
||||||
|
|
||||||
addChild(Toolbar(
|
addChild(Toolbar(
|
||||||
scope,
|
scope,
|
||||||
children = listOf(
|
children = listOf(
|
||||||
|
@ -22,9 +22,11 @@ open class QuestEditorWidget(
|
|||||||
private val toolbar: Widget,
|
private val toolbar: Widget,
|
||||||
private val createQuestInfoWidget: (CoroutineScope) -> Widget,
|
private val createQuestInfoWidget: (CoroutineScope) -> Widget,
|
||||||
private val createQuestRendererWidget: (CoroutineScope) -> Widget,
|
private val createQuestRendererWidget: (CoroutineScope) -> Widget,
|
||||||
) : Widget(scope, listOf(::style)) {
|
) : Widget(scope) {
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
div(className = "pw-quest-editor-quest-editor") {
|
div {
|
||||||
|
className = "pw-quest-editor-quest-editor"
|
||||||
|
|
||||||
addChild(toolbar)
|
addChild(toolbar)
|
||||||
addChild(DockWidget(
|
addChild(DockWidget(
|
||||||
scope,
|
scope,
|
||||||
@ -93,17 +95,21 @@ open class QuestEditorWidget(
|
|||||||
)
|
)
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnusedSymbol")
|
companion object {
|
||||||
// language=css
|
init {
|
||||||
private fun style() = """
|
@Suppress("CssUnusedSymbol")
|
||||||
.pw-quest-editor-quest-editor {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-quest-editor-quest-editor {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
.pw-quest-editor-quest-editor > * {
|
.pw-quest-editor-quest-editor > * {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -2,21 +2,29 @@ package world.phantasmal.web.questEditor.widgets
|
|||||||
|
|
||||||
import kotlinx.coroutines.CoroutineScope
|
import kotlinx.coroutines.CoroutineScope
|
||||||
import org.w3c.dom.Node
|
import org.w3c.dom.Node
|
||||||
|
import world.phantasmal.observable.value.not
|
||||||
|
import world.phantasmal.web.core.widgets.UnavailableWidget
|
||||||
import world.phantasmal.web.questEditor.controllers.QuestInfoController
|
import world.phantasmal.web.questEditor.controllers.QuestInfoController
|
||||||
import world.phantasmal.webui.dom.*
|
import world.phantasmal.webui.dom.*
|
||||||
import world.phantasmal.webui.widgets.IntInput
|
import world.phantasmal.webui.widgets.IntInput
|
||||||
|
import world.phantasmal.webui.widgets.TextInput
|
||||||
import world.phantasmal.webui.widgets.Widget
|
import world.phantasmal.webui.widgets.Widget
|
||||||
|
|
||||||
class QuestInfoWidget(
|
class QuestInfoWidget(
|
||||||
scope: CoroutineScope,
|
scope: CoroutineScope,
|
||||||
private val ctrl: QuestInfoController,
|
private val ctrl: QuestInfoController,
|
||||||
) : Widget(scope, listOf(::style)) {
|
) : Widget(scope) {
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
div(className = "pw-quest-editor-quest-info", tabIndex = -1) {
|
div {
|
||||||
|
className = "pw-quest-editor-quest-info"
|
||||||
|
tabIndex = -1
|
||||||
|
|
||||||
table {
|
table {
|
||||||
|
hidden(ctrl.unavailable)
|
||||||
|
|
||||||
tr {
|
tr {
|
||||||
th { textContent = "Episode:" }
|
th { textContent = "Episode:" }
|
||||||
td()
|
td { text(ctrl.episode) }
|
||||||
}
|
}
|
||||||
tr {
|
tr {
|
||||||
th { textContent = "ID:" }
|
th { textContent = "ID:" }
|
||||||
@ -25,41 +33,60 @@ class QuestInfoWidget(
|
|||||||
this@QuestInfoWidget.scope,
|
this@QuestInfoWidget.scope,
|
||||||
valueVal = ctrl.id,
|
valueVal = ctrl.id,
|
||||||
min = 0,
|
min = 0,
|
||||||
step = 0
|
step = 1
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tr {
|
||||||
|
th { textContent = "Name:" }
|
||||||
|
td {
|
||||||
|
addChild(TextInput(
|
||||||
|
this@QuestInfoWidget.scope,
|
||||||
|
valueVal = ctrl.name,
|
||||||
|
maxLength = 32
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
addChild(UnavailableWidget(
|
||||||
|
scope,
|
||||||
|
hidden = !ctrl.unavailable,
|
||||||
|
message = "No quest loaded."
|
||||||
|
))
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnusedSymbol")
|
companion object {
|
||||||
// language=css
|
init {
|
||||||
private fun style() = """
|
@Suppress("CssUnusedSymbol")
|
||||||
.pw-quest-editor-quest-info {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-quest-editor-quest-info {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
padding: 3px;
|
padding: 3px;
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
outline: none;
|
outline: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-quest-editor-quest-info table {
|
.pw-quest-editor-quest-info table {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-quest-editor-quest-info th {
|
.pw-quest-editor-quest-info th {
|
||||||
text-align: left;
|
text-align: left;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-quest-editor-quest-info .pw-text-input {
|
.pw-quest-editor-quest-info .pw-text-input {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-quest-editor-quest-info .pw-text-area {
|
.pw-quest-editor-quest-info .pw-text-area {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-quest-editor-quest-info textarea {
|
.pw-quest-editor-quest-info textarea {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -11,20 +11,27 @@ import world.phantasmal.webui.widgets.Widget
|
|||||||
abstract class QuestRendererWidget(
|
abstract class QuestRendererWidget(
|
||||||
scope: CoroutineScope,
|
scope: CoroutineScope,
|
||||||
private val createEngine: (HTMLCanvasElement) -> Engine,
|
private val createEngine: (HTMLCanvasElement) -> Engine,
|
||||||
) : Widget(scope, listOf(::style)) {
|
) : Widget(scope) {
|
||||||
override fun Node.createElement() = div(className = "pw-quest-editor-quest-renderer") {
|
override fun Node.createElement() =
|
||||||
|
div {
|
||||||
|
className = "pw-quest-editor-quest-renderer"
|
||||||
|
|
||||||
addChild(RendererWidget(scope, createEngine))
|
addChild(RendererWidget(scope, createEngine))
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnusedSymbol")
|
companion object {
|
||||||
// language=css
|
init {
|
||||||
private fun style() = """
|
@Suppress("CssUnusedSymbol")
|
||||||
.pw-quest-editor-quest-renderer {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-quest-editor-quest-renderer {
|
||||||
display: flex;
|
display: flex;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
.pw-quest-editor-quest-renderer > * {
|
.pw-quest-editor-quest-renderer > * {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -17,7 +17,7 @@ import kotlin.test.Test
|
|||||||
|
|
||||||
class ApplicationTests : TestSuite() {
|
class ApplicationTests : TestSuite() {
|
||||||
@Test
|
@Test
|
||||||
fun initialization_and_shutdown_should_succeed_without_throwing() {
|
fun initialization_and_shutdown_should_succeed_without_throwing() = test {
|
||||||
(listOf(null) + PwTool.values().toList()).forEach { tool ->
|
(listOf(null) + PwTool.values().toList()).forEach { tool ->
|
||||||
Disposer().use { disposer ->
|
Disposer().use { disposer ->
|
||||||
val httpClient = HttpClient {
|
val httpClient = HttpClient {
|
||||||
|
@ -10,7 +10,7 @@ import kotlin.test.assertFalse
|
|||||||
|
|
||||||
class PathAwareTabControllerTests : TestSuite() {
|
class PathAwareTabControllerTests : TestSuite() {
|
||||||
@Test
|
@Test
|
||||||
fun activeTab_is_initialized_correctly() {
|
fun activeTab_is_initialized_correctly() = test {
|
||||||
setup { ctrl, appUrl ->
|
setup { ctrl, appUrl ->
|
||||||
assertEquals("/b", ctrl.activeTab.value?.path)
|
assertEquals("/b", ctrl.activeTab.value?.path)
|
||||||
assertFalse(appUrl.canGoBack)
|
assertFalse(appUrl.canGoBack)
|
||||||
@ -19,7 +19,7 @@ class PathAwareTabControllerTests : TestSuite() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun applicationUrl_changes_when_activeTab_changes() {
|
fun applicationUrl_changes_when_activeTab_changes() = test {
|
||||||
setup { ctrl, appUrl ->
|
setup { ctrl, appUrl ->
|
||||||
ctrl.setActiveTab(ctrl.tabs[2])
|
ctrl.setActiveTab(ctrl.tabs[2])
|
||||||
|
|
||||||
@ -30,7 +30,7 @@ class PathAwareTabControllerTests : TestSuite() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun activeTab_changes_when_applicationUrl_changes() {
|
fun activeTab_changes_when_applicationUrl_changes() = test {
|
||||||
setup { ctrl, applicationUrl ->
|
setup { ctrl, applicationUrl ->
|
||||||
applicationUrl.pushUrl("/${PwTool.HuntOptimizer.slug}/c")
|
applicationUrl.pushUrl("/${PwTool.HuntOptimizer.slug}/c")
|
||||||
|
|
||||||
@ -39,7 +39,7 @@ class PathAwareTabControllerTests : TestSuite() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun applicationUrl_changes_when_switch_to_tool_with_tabs() {
|
fun applicationUrl_changes_when_switch_to_tool_with_tabs() = test {
|
||||||
val appUrl = TestApplicationUrl("/")
|
val appUrl = TestApplicationUrl("/")
|
||||||
val uiStore = disposer.add(UiStore(scope, appUrl))
|
val uiStore = disposer.add(UiStore(scope, appUrl))
|
||||||
|
|
||||||
@ -66,7 +66,7 @@ class PathAwareTabControllerTests : TestSuite() {
|
|||||||
assertEquals("/${uiStore.defaultTool.slug}", appUrl.url.value)
|
assertEquals("/${uiStore.defaultTool.slug}", appUrl.url.value)
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun setup(
|
private fun TestContext.setup(
|
||||||
block: (PathAwareTabController<PathAwareTab>, applicationUrl: TestApplicationUrl) -> Unit,
|
block: (PathAwareTabController<PathAwareTab>, applicationUrl: TestApplicationUrl) -> Unit,
|
||||||
) {
|
) {
|
||||||
val applicationUrl = TestApplicationUrl("/${PwTool.HuntOptimizer.slug}/b")
|
val applicationUrl = TestApplicationUrl("/${PwTool.HuntOptimizer.slug}/b")
|
||||||
|
@ -9,7 +9,7 @@ import kotlin.test.assertEquals
|
|||||||
|
|
||||||
class UiStoreTests : TestSuite() {
|
class UiStoreTests : TestSuite() {
|
||||||
@Test
|
@Test
|
||||||
fun applicationUrl_is_initialized_correctly() {
|
fun applicationUrl_is_initialized_correctly() = test {
|
||||||
val applicationUrl = TestApplicationUrl("/")
|
val applicationUrl = TestApplicationUrl("/")
|
||||||
val uiStore = disposer.add(UiStore(scope, applicationUrl))
|
val uiStore = disposer.add(UiStore(scope, applicationUrl))
|
||||||
|
|
||||||
@ -18,7 +18,7 @@ class UiStoreTests : TestSuite() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun applicationUrl_changes_when_tool_changes() {
|
fun applicationUrl_changes_when_tool_changes() = test {
|
||||||
val applicationUrl = TestApplicationUrl("/")
|
val applicationUrl = TestApplicationUrl("/")
|
||||||
val uiStore = disposer.add(UiStore(scope, applicationUrl))
|
val uiStore = disposer.add(UiStore(scope, applicationUrl))
|
||||||
|
|
||||||
@ -31,7 +31,7 @@ class UiStoreTests : TestSuite() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun applicationUrl_changes_when_path_changes() {
|
fun applicationUrl_changes_when_path_changes() = test {
|
||||||
val applicationUrl = TestApplicationUrl("/")
|
val applicationUrl = TestApplicationUrl("/")
|
||||||
val uiStore = disposer.add(UiStore(scope, applicationUrl))
|
val uiStore = disposer.add(UiStore(scope, applicationUrl))
|
||||||
|
|
||||||
@ -46,7 +46,7 @@ class UiStoreTests : TestSuite() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun currentTool_and_path_change_when_applicationUrl_changes() {
|
fun currentTool_and_path_change_when_applicationUrl_changes() = test {
|
||||||
val applicationUrl = TestApplicationUrl("/")
|
val applicationUrl = TestApplicationUrl("/")
|
||||||
val uiStore = disposer.add(UiStore(scope, applicationUrl))
|
val uiStore = disposer.add(UiStore(scope, applicationUrl))
|
||||||
|
|
||||||
@ -61,7 +61,7 @@ class UiStoreTests : TestSuite() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun browser_navigation_stack_is_manipulated_correctly() {
|
fun browser_navigation_stack_is_manipulated_correctly() = test {
|
||||||
val appUrl = TestApplicationUrl("/")
|
val appUrl = TestApplicationUrl("/")
|
||||||
val uiStore = disposer.add(UiStore(scope, appUrl))
|
val uiStore = disposer.add(UiStore(scope, appUrl))
|
||||||
|
|
||||||
|
@ -14,7 +14,7 @@ import kotlin.test.Test
|
|||||||
|
|
||||||
class HuntOptimizerTests : TestSuite() {
|
class HuntOptimizerTests : TestSuite() {
|
||||||
@Test
|
@Test
|
||||||
fun initialization_and_shutdown_should_succeed_without_throwing() {
|
fun initialization_and_shutdown_should_succeed_without_throwing() = test {
|
||||||
val httpClient = HttpClient {
|
val httpClient = HttpClient {
|
||||||
install(JsonFeature) {
|
install(JsonFeature) {
|
||||||
serializer = KotlinxSerializer(kotlinx.serialization.json.Json {
|
serializer = KotlinxSerializer(kotlinx.serialization.json.Json {
|
||||||
|
@ -14,7 +14,7 @@ import kotlin.test.Test
|
|||||||
|
|
||||||
class QuestEditorTests : TestSuite() {
|
class QuestEditorTests : TestSuite() {
|
||||||
@Test
|
@Test
|
||||||
fun initialization_and_shutdown_should_succeed_without_throwing() {
|
fun initialization_and_shutdown_should_succeed_without_throwing() = test {
|
||||||
val httpClient = HttpClient {
|
val httpClient = HttpClient {
|
||||||
install(JsonFeature) {
|
install(JsonFeature) {
|
||||||
serializer = KotlinxSerializer(kotlinx.serialization.json.Json {
|
serializer = KotlinxSerializer(kotlinx.serialization.json.Json {
|
||||||
|
@ -2,14 +2,13 @@ package world.phantasmal.webui.dom
|
|||||||
|
|
||||||
import kotlinx.browser.document
|
import kotlinx.browser.document
|
||||||
import kotlinx.dom.appendText
|
import kotlinx.dom.appendText
|
||||||
import kotlinx.dom.clear
|
import org.w3c.dom.AddEventListenerOptions
|
||||||
import org.w3c.dom.*
|
import org.w3c.dom.HTMLElement
|
||||||
|
import org.w3c.dom.HTMLStyleElement
|
||||||
import org.w3c.dom.events.Event
|
import org.w3c.dom.events.Event
|
||||||
import org.w3c.dom.events.EventTarget
|
import org.w3c.dom.events.EventTarget
|
||||||
import world.phantasmal.core.disposable.Disposable
|
import world.phantasmal.core.disposable.Disposable
|
||||||
import world.phantasmal.core.disposable.disposable
|
import world.phantasmal.core.disposable.disposable
|
||||||
import world.phantasmal.observable.value.list.ListVal
|
|
||||||
import world.phantasmal.observable.value.list.ListValChangeEvent
|
|
||||||
|
|
||||||
fun <E : Event> disposableListener(
|
fun <E : Event> disposableListener(
|
||||||
target: EventTarget,
|
target: EventTarget,
|
||||||
|
@ -3,227 +3,68 @@ package world.phantasmal.webui.dom
|
|||||||
import kotlinx.browser.document
|
import kotlinx.browser.document
|
||||||
import org.w3c.dom.*
|
import org.w3c.dom.*
|
||||||
|
|
||||||
fun Node.a(
|
fun Node.a(block: HTMLAnchorElement.() -> Unit = {}): HTMLAnchorElement =
|
||||||
href: String? = null,
|
appendHtmlEl("A", block)
|
||||||
id: String? = null,
|
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLAnchorElement.() -> Unit = {},
|
|
||||||
): HTMLAnchorElement =
|
|
||||||
appendHtmlEl("A", id, className, title, tabIndex) {
|
|
||||||
if (href != null) this.href = href
|
|
||||||
block()
|
|
||||||
}
|
|
||||||
|
|
||||||
fun Node.button(
|
fun Node.button(block: HTMLButtonElement.() -> Unit = {}): HTMLButtonElement =
|
||||||
type: String? = null,
|
appendHtmlEl("BUTTON", block)
|
||||||
id: String? = null,
|
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLButtonElement.() -> Unit = {},
|
|
||||||
): HTMLButtonElement =
|
|
||||||
appendHtmlEl("BUTTON", id, className, title, tabIndex) {
|
|
||||||
if (type != null) this.type = type
|
|
||||||
block()
|
|
||||||
}
|
|
||||||
|
|
||||||
fun Node.canvas(
|
fun Node.canvas(block: HTMLCanvasElement.() -> Unit = {}): HTMLCanvasElement =
|
||||||
id: String? = null,
|
appendHtmlEl("CANVAS", block)
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLCanvasElement.() -> Unit = {},
|
|
||||||
): HTMLCanvasElement =
|
|
||||||
appendHtmlEl("CANVAS", id, className, title, tabIndex, block)
|
|
||||||
|
|
||||||
fun Node.div(
|
fun Node.div(block: HTMLDivElement .() -> Unit = {}): HTMLDivElement =
|
||||||
id: String? = null,
|
appendHtmlEl("DIV", block)
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLDivElement .() -> Unit = {},
|
|
||||||
): HTMLDivElement =
|
|
||||||
appendHtmlEl("DIV", id, className, title, tabIndex, block)
|
|
||||||
|
|
||||||
fun Node.form(
|
fun Node.form(block: HTMLFormElement.() -> Unit = {}): HTMLFormElement =
|
||||||
id: String? = null,
|
appendHtmlEl("FORM", block)
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLFormElement.() -> Unit = {},
|
|
||||||
): HTMLFormElement =
|
|
||||||
appendHtmlEl("FORM", id, className, title, tabIndex, block)
|
|
||||||
|
|
||||||
fun Node.h1(
|
fun Node.h1(block: HTMLHeadingElement.() -> Unit = {}): HTMLHeadingElement =
|
||||||
id: String? = null,
|
appendHtmlEl("H1", block)
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLHeadingElement.() -> Unit = {},
|
|
||||||
): HTMLHeadingElement =
|
|
||||||
appendHtmlEl("H1", id, className, title, tabIndex, block)
|
|
||||||
|
|
||||||
fun Node.h2(
|
fun Node.h2(block: HTMLHeadingElement.() -> Unit = {}): HTMLHeadingElement =
|
||||||
id: String? = null,
|
appendHtmlEl("H2", block)
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLHeadingElement.() -> Unit = {},
|
|
||||||
): HTMLHeadingElement =
|
|
||||||
appendHtmlEl("H2", id, className, title, tabIndex, block)
|
|
||||||
|
|
||||||
fun Node.header(
|
fun Node.header(block: HTMLElement.() -> Unit = {}): HTMLElement =
|
||||||
id: String? = null,
|
appendHtmlEl("HEADER", block)
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLElement.() -> Unit = {},
|
|
||||||
): HTMLElement =
|
|
||||||
appendHtmlEl("HEADER", id, className, title, tabIndex, block)
|
|
||||||
|
|
||||||
fun Node.img(
|
fun Node.img(block: HTMLImageElement.() -> Unit = {}): HTMLImageElement =
|
||||||
src: String? = null,
|
appendHtmlEl("IMG", block)
|
||||||
width: Int? = null,
|
|
||||||
height: Int? = null,
|
|
||||||
alt: String? = null,
|
|
||||||
id: String? = null,
|
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLImageElement.() -> Unit = {},
|
|
||||||
): HTMLImageElement =
|
|
||||||
appendHtmlEl("IMG", id, className, title, tabIndex) {
|
|
||||||
if (src != null) this.src = src
|
|
||||||
if (width != null) this.width = width
|
|
||||||
if (height != null) this.height = height
|
|
||||||
if (alt != null) this.alt = alt
|
|
||||||
block()
|
|
||||||
}
|
|
||||||
|
|
||||||
fun Node.input(
|
fun Node.input(block: HTMLInputElement.() -> Unit = {}): HTMLInputElement =
|
||||||
type: String? = null,
|
appendHtmlEl("INPUT", block)
|
||||||
id: String? = null,
|
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLInputElement.() -> Unit = {},
|
|
||||||
): HTMLInputElement =
|
|
||||||
appendHtmlEl("INPUT", id, className, title, tabIndex) {
|
|
||||||
if (type != null) this.type = type
|
|
||||||
block()
|
|
||||||
}
|
|
||||||
|
|
||||||
fun Node.label(
|
fun Node.label(block: HTMLLabelElement.() -> Unit = {}): HTMLLabelElement =
|
||||||
htmlFor: String? = null,
|
appendHtmlEl("LABEL", block)
|
||||||
id: String? = null,
|
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLLabelElement.() -> Unit = {},
|
|
||||||
): HTMLLabelElement =
|
|
||||||
appendHtmlEl("LABEL", id, className, title, tabIndex) {
|
|
||||||
if (htmlFor != null) this.htmlFor = htmlFor
|
|
||||||
block()
|
|
||||||
}
|
|
||||||
|
|
||||||
fun Node.main(
|
fun Node.main(block: HTMLElement.() -> Unit = {}): HTMLElement =
|
||||||
id: String? = null,
|
appendHtmlEl("MAIN", block)
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLElement.() -> Unit = {},
|
|
||||||
): HTMLElement =
|
|
||||||
appendHtmlEl("MAIN", id, className, title, tabIndex, block)
|
|
||||||
|
|
||||||
fun Node.p(
|
fun Node.p(block: HTMLParagraphElement.() -> Unit = {}): HTMLParagraphElement =
|
||||||
id: String? = null,
|
appendHtmlEl("P", block)
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLParagraphElement.() -> Unit = {},
|
|
||||||
): HTMLParagraphElement =
|
|
||||||
appendHtmlEl("P", id, className, title, tabIndex, block)
|
|
||||||
|
|
||||||
fun Node.span(
|
fun Node.span(block: HTMLSpanElement.() -> Unit = {}): HTMLSpanElement =
|
||||||
id: String? = null,
|
appendHtmlEl("SPAN", block)
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLSpanElement.() -> Unit = {},
|
|
||||||
): HTMLSpanElement =
|
|
||||||
appendHtmlEl("SPAN", id, className, title, tabIndex, block)
|
|
||||||
|
|
||||||
fun Node.table(
|
fun Node.table(block: HTMLTableElement.() -> Unit = {}): HTMLTableElement =
|
||||||
id: String? = null,
|
appendHtmlEl("TABLE", block)
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLTableElement.() -> Unit = {},
|
|
||||||
): HTMLTableElement =
|
|
||||||
appendHtmlEl("TABLE", id, className, title, tabIndex, block)
|
|
||||||
|
|
||||||
fun Node.td(
|
fun Node.td(block: HTMLTableCellElement.() -> Unit = {}): HTMLTableCellElement =
|
||||||
id: String? = null,
|
appendHtmlEl("TD", block)
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLTableCellElement.() -> Unit = {},
|
|
||||||
): HTMLTableCellElement =
|
|
||||||
appendHtmlEl("TD", id, className, title, tabIndex, block)
|
|
||||||
|
|
||||||
fun Node.th(
|
fun Node.th(block: HTMLTableCellElement.() -> Unit = {}): HTMLTableCellElement =
|
||||||
id: String? = null,
|
appendHtmlEl("TH", block)
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLTableCellElement.() -> Unit = {},
|
|
||||||
): HTMLTableCellElement =
|
|
||||||
appendHtmlEl("TH", id, className, title, tabIndex, block)
|
|
||||||
|
|
||||||
fun Node.tr(
|
fun Node.tr(block: HTMLTableRowElement.() -> Unit = {}): HTMLTableRowElement =
|
||||||
id: String? = null,
|
appendHtmlEl("TR", block)
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: HTMLTableRowElement.() -> Unit = {},
|
|
||||||
): HTMLTableRowElement =
|
|
||||||
appendHtmlEl("TR", id, className, title, tabIndex, block)
|
|
||||||
|
|
||||||
fun <T : HTMLElement> Node.appendHtmlEl(
|
fun <T : HTMLElement> Node.appendHtmlEl(tagName: String, block: T.() -> Unit): T =
|
||||||
tagName: String,
|
appendChild(newHtmlEl(tagName, block)).unsafeCast<T>()
|
||||||
id: String? = null,
|
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: T.() -> Unit,
|
|
||||||
): T =
|
|
||||||
appendChild(newHtmlEl(tagName, id, className, title, tabIndex, block)).unsafeCast<T>()
|
|
||||||
|
|
||||||
fun <T : HTMLElement> newHtmlEl(
|
fun <T : HTMLElement> newHtmlEl(tagName: String, block: T.() -> Unit): T =
|
||||||
tagName: String,
|
newEl(tagName, block)
|
||||||
id: String? = null,
|
|
||||||
className: String? = null,
|
|
||||||
title: String? = null,
|
|
||||||
tabIndex: Int? = null,
|
|
||||||
block: T.() -> Unit,
|
|
||||||
): T =
|
|
||||||
newEl(tagName, id, className) {
|
|
||||||
if (title != null) this.title = title
|
|
||||||
if (tabIndex != null) this.tabIndex = tabIndex
|
|
||||||
block()
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun <T : Element> newEl(
|
private fun <T : Element> newEl(tagName: String, block: T.() -> Unit): T {
|
||||||
tagName: String,
|
|
||||||
id: String? = null,
|
|
||||||
className: String?,
|
|
||||||
block: T.() -> Unit,
|
|
||||||
): T {
|
|
||||||
val el = document.createElement(tagName).unsafeCast<T>()
|
val el = document.createElement(tagName).unsafeCast<T>()
|
||||||
if (id != null) el.id = id
|
|
||||||
if (className != null) el.className = className
|
|
||||||
el.block()
|
el.block()
|
||||||
return el
|
return el
|
||||||
}
|
}
|
||||||
|
@ -15,13 +15,18 @@ open class Button(
|
|||||||
private val text: String? = null,
|
private val text: String? = null,
|
||||||
private val textVal: Val<String>? = null,
|
private val textVal: Val<String>? = null,
|
||||||
private val onclick: ((MouseEvent) -> Unit)? = null,
|
private val onclick: ((MouseEvent) -> Unit)? = null,
|
||||||
) : Control(scope, listOf(::style), hidden, disabled) {
|
) : Control(scope, hidden, disabled) {
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
button(className = "pw-button") {
|
button {
|
||||||
|
className = "pw-button"
|
||||||
onclick = this@Button.onclick
|
onclick = this@Button.onclick
|
||||||
|
|
||||||
span(className = "pw-button-inner") {
|
span {
|
||||||
span(className = "pw-button-center") {
|
className = "pw-button-inner"
|
||||||
|
|
||||||
|
span {
|
||||||
|
className = "pw-button-center"
|
||||||
|
|
||||||
if (textVal != null) {
|
if (textVal != null) {
|
||||||
observe(textVal) {
|
observe(textVal) {
|
||||||
textContent = it
|
textContent = it
|
||||||
@ -35,12 +40,13 @@ open class Button(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnusedSymbol", "CssUnresolvedCustomProperty")
|
companion object {
|
||||||
|
init {
|
||||||
|
@Suppress("CssUnusedSymbol", "CssUnresolvedCustomProperty")
|
||||||
// language=css
|
// language=css
|
||||||
private fun style() = """
|
style("""
|
||||||
.pw-button {
|
.pw-button {
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
align-items: stretch;
|
align-items: stretch;
|
||||||
@ -54,9 +60,9 @@ private fun style() = """
|
|||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
font-family: var(--pw-font-family), sans-serif;
|
font-family: var(--pw-font-family), sans-serif;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-button .pw-button-inner {
|
.pw-button .pw-button-inner {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
@ -67,47 +73,50 @@ private fun style() = """
|
|||||||
padding: 3px 5px;
|
padding: 3px 5px;
|
||||||
border: var(--pw-control-inner-border);
|
border: var(--pw-control-inner-border);
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-button:hover .pw-button-inner {
|
.pw-button:hover .pw-button-inner {
|
||||||
background-color: var(--pw-control-bg-color-hover);
|
background-color: var(--pw-control-bg-color-hover);
|
||||||
border-color: hsl(0, 0%, 40%);
|
border-color: hsl(0, 0%, 40%);
|
||||||
color: var(--pw-control-text-color-hover);
|
color: var(--pw-control-text-color-hover);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-button:active .pw-button-inner {
|
.pw-button:active .pw-button-inner {
|
||||||
background-color: hsl(0, 0%, 20%);
|
background-color: hsl(0, 0%, 20%);
|
||||||
border-color: hsl(0, 0%, 30%);
|
border-color: hsl(0, 0%, 30%);
|
||||||
color: hsl(0, 0%, 75%);
|
color: hsl(0, 0%, 75%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-button:focus-within .pw-button-inner {
|
.pw-button:focus-within .pw-button-inner {
|
||||||
border: var(--pw-control-inner-border-focus);
|
border: var(--pw-control-inner-border-focus);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-button:disabled .pw-button-inner {
|
.pw-button:disabled .pw-button-inner {
|
||||||
background-color: hsl(0, 0%, 15%);
|
background-color: hsl(0, 0%, 15%);
|
||||||
border-color: hsl(0, 0%, 25%);
|
border-color: hsl(0, 0%, 25%);
|
||||||
color: hsl(0, 0%, 55%);
|
color: hsl(0, 0%, 55%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-button-inner > * {
|
.pw-button-inner > * {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
margin: 0 3px;
|
margin: 0 3px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-button-center {
|
.pw-button-center {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
text-overflow: ellipsis;
|
text-overflow: ellipsis;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-button-left,
|
.pw-button-left,
|
||||||
.pw-button-right {
|
.pw-button-right {
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
align-content: center;
|
align-content: center;
|
||||||
font-size: 11px;
|
font-size: 11px;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -10,7 +10,6 @@ import world.phantasmal.observable.value.falseVal
|
|||||||
*/
|
*/
|
||||||
abstract class Control(
|
abstract class Control(
|
||||||
scope: CoroutineScope,
|
scope: CoroutineScope,
|
||||||
styles: List<() -> String>,
|
|
||||||
hidden: Val<Boolean> = falseVal(),
|
hidden: Val<Boolean> = falseVal(),
|
||||||
disabled: Val<Boolean> = falseVal(),
|
disabled: Val<Boolean> = falseVal(),
|
||||||
) : Widget(scope, styles, hidden, disabled)
|
) : Widget(scope, hidden, disabled)
|
||||||
|
@ -9,7 +9,6 @@ import world.phantasmal.webui.dom.span
|
|||||||
|
|
||||||
abstract class Input<T>(
|
abstract class Input<T>(
|
||||||
scope: CoroutineScope,
|
scope: CoroutineScope,
|
||||||
styles: List<() -> String>,
|
|
||||||
hidden: Val<Boolean>,
|
hidden: Val<Boolean>,
|
||||||
disabled: Val<Boolean>,
|
disabled: Val<Boolean>,
|
||||||
label: String?,
|
label: String?,
|
||||||
@ -21,12 +20,12 @@ abstract class Input<T>(
|
|||||||
private val value: T?,
|
private val value: T?,
|
||||||
private val valueVal: Val<T>?,
|
private val valueVal: Val<T>?,
|
||||||
private val setValue: ((T) -> Unit)?,
|
private val setValue: ((T) -> Unit)?,
|
||||||
|
private val maxLength: Int?,
|
||||||
private val min: Int?,
|
private val min: Int?,
|
||||||
private val max: Int?,
|
private val max: Int?,
|
||||||
private val step: Int?,
|
private val step: Int?,
|
||||||
) : LabelledControl(
|
) : LabelledControl(
|
||||||
scope,
|
scope,
|
||||||
styles + ::style,
|
|
||||||
hidden,
|
hidden,
|
||||||
disabled,
|
disabled,
|
||||||
label,
|
label,
|
||||||
@ -34,11 +33,12 @@ abstract class Input<T>(
|
|||||||
preferredLabelPosition,
|
preferredLabelPosition,
|
||||||
) {
|
) {
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
span(className = "pw-input") {
|
span {
|
||||||
classList.add(className)
|
classList.add("pw-input", this@Input.className)
|
||||||
|
|
||||||
input(className = "pw-input-inner", type = inputType) {
|
input {
|
||||||
classList.add(inputClassName)
|
classList.add("pw-input-inner", inputClassName)
|
||||||
|
type = inputType
|
||||||
|
|
||||||
observe(this@Input.disabled) { disabled = it }
|
observe(this@Input.disabled) { disabled = it }
|
||||||
|
|
||||||
@ -58,36 +58,30 @@ abstract class Input<T>(
|
|||||||
setInputValue(this, this@Input.value)
|
setInputValue(this, this@Input.value)
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this@Input.min != null) {
|
this@Input.maxLength?.let { maxLength = it }
|
||||||
min = this@Input.min.toString()
|
this@Input.min?.let { min = it.toString() }
|
||||||
}
|
this@Input.max?.let { max = it.toString() }
|
||||||
|
this@Input.step?.let { step = it.toString() }
|
||||||
if (this@Input.max != null) {
|
|
||||||
max = this@Input.max.toString()
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this@Input.step != null) {
|
|
||||||
step = this@Input.step.toString()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected abstract fun getInputValue(input: HTMLInputElement): T
|
protected abstract fun getInputValue(input: HTMLInputElement): T
|
||||||
|
|
||||||
protected abstract fun setInputValue(input: HTMLInputElement, value: T)
|
protected abstract fun setInputValue(input: HTMLInputElement, value: T)
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnusedSymbol", "CssUnresolvedCustomProperty")
|
companion object {
|
||||||
// language=css
|
init {
|
||||||
private fun style() = """
|
@Suppress("CssUnusedSymbol", "CssUnresolvedCustomProperty")
|
||||||
.pw-input {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-input {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
height: 24px;
|
height: 24px;
|
||||||
border: var(--pw-input-border);
|
border: var(--pw-input-border);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-input .pw-input-inner {
|
.pw-input .pw-input-inner {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
@ -97,22 +91,25 @@ private fun style() = """
|
|||||||
color: var(--pw-input-text-color);
|
color: var(--pw-input-text-color);
|
||||||
outline: none;
|
outline: none;
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-input:hover {
|
.pw-input:hover {
|
||||||
border: var(--pw-input-border-hover);
|
border: var(--pw-input-border-hover);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-input:focus-within {
|
.pw-input:focus-within {
|
||||||
border: var(--pw-input-border-focus);
|
border: var(--pw-input-border-focus);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-input.disabled {
|
.pw-input.disabled {
|
||||||
border: var(--pw-input-border-disabled);
|
border: var(--pw-input-border-disabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-input.disabled .pw-input-inner {
|
.pw-input.disabled .pw-input-inner {
|
||||||
color: var(--pw-input-text-color-disabled);
|
color: var(--pw-input-text-color-disabled);
|
||||||
background-color: var(--pw-input-bg-color-disabled);
|
background-color: var(--pw-input-bg-color-disabled);
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -12,22 +12,29 @@ class Label(
|
|||||||
disabled: Val<Boolean> = falseVal(),
|
disabled: Val<Boolean> = falseVal(),
|
||||||
private val text: String? = null,
|
private val text: String? = null,
|
||||||
private val textVal: Val<String>? = null,
|
private val textVal: Val<String>? = null,
|
||||||
private val htmlFor: String?,
|
private val htmlFor: String? = null,
|
||||||
) : Widget(scope, listOf(::style), hidden, disabled) {
|
) : Widget(scope, hidden, disabled) {
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
label(htmlFor) {
|
label {
|
||||||
|
className = "pw-label"
|
||||||
|
this@Label.htmlFor?.let { htmlFor = it }
|
||||||
|
|
||||||
if (textVal != null) {
|
if (textVal != null) {
|
||||||
observe(textVal) { textContent = it }
|
text(textVal)
|
||||||
} else if (text != null) {
|
} else if (text != null) {
|
||||||
textContent = text
|
textContent = text
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnusedSymbol", "CssUnresolvedCustomProperty")
|
companion object{
|
||||||
// language=css
|
init {
|
||||||
private fun style() = """
|
@Suppress("CssUnusedSymbol", "CssUnresolvedCustomProperty")
|
||||||
.pw-label.disabled {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-label.pw-disabled {
|
||||||
color: var(--pw-text-color-disabled);
|
color: var(--pw-text-color-disabled);
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -11,13 +11,12 @@ enum class LabelPosition {
|
|||||||
|
|
||||||
abstract class LabelledControl(
|
abstract class LabelledControl(
|
||||||
scope: CoroutineScope,
|
scope: CoroutineScope,
|
||||||
styles: List<() -> String>,
|
|
||||||
hidden: Val<Boolean> = falseVal(),
|
hidden: Val<Boolean> = falseVal(),
|
||||||
disabled: Val<Boolean> = falseVal(),
|
disabled: Val<Boolean> = falseVal(),
|
||||||
label: String? = null,
|
label: String? = null,
|
||||||
labelVal: Val<String>? = null,
|
labelVal: Val<String>? = null,
|
||||||
val preferredLabelPosition: LabelPosition,
|
val preferredLabelPosition: LabelPosition,
|
||||||
) : Control(scope, styles, hidden, disabled) {
|
) : Control(scope, hidden, disabled) {
|
||||||
val label: Label? by lazy {
|
val label: Label? by lazy {
|
||||||
if (label == null && labelVal == null) {
|
if (label == null && labelVal == null) {
|
||||||
null
|
null
|
||||||
|
@ -11,11 +11,13 @@ class LazyLoader(
|
|||||||
hidden: Val<Boolean> = falseVal(),
|
hidden: Val<Boolean> = falseVal(),
|
||||||
disabled: Val<Boolean> = falseVal(),
|
disabled: Val<Boolean> = falseVal(),
|
||||||
private val createWidget: (CoroutineScope) -> Widget,
|
private val createWidget: (CoroutineScope) -> Widget,
|
||||||
) : Widget(scope, listOf(::style), hidden, disabled) {
|
) : Widget(scope, hidden, disabled) {
|
||||||
private var initialized = false
|
private var initialized = false
|
||||||
|
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
div(className = "pw-lazy-loader") {
|
div {
|
||||||
|
className = "pw-lazy-loader"
|
||||||
|
|
||||||
observe(this@LazyLoader.hidden) { h ->
|
observe(this@LazyLoader.hidden) { h ->
|
||||||
if (!h && !initialized) {
|
if (!h && !initialized) {
|
||||||
initialized = true
|
initialized = true
|
||||||
@ -23,19 +25,23 @@ class LazyLoader(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnusedSymbol")
|
companion object {
|
||||||
// language=css
|
init {
|
||||||
private fun style() = """
|
@Suppress("CssUnusedSymbol")
|
||||||
.pw-lazy-loader {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-lazy-loader {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
align-items: stretch;
|
align-items: stretch;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-lazy-loader > * {
|
.pw-lazy-loader > * {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -18,7 +18,6 @@ abstract class NumberInput<T : Number>(
|
|||||||
step: Int?,
|
step: Int?,
|
||||||
) : Input<T>(
|
) : Input<T>(
|
||||||
scope,
|
scope,
|
||||||
listOf(::style),
|
|
||||||
hidden,
|
hidden,
|
||||||
disabled,
|
disabled,
|
||||||
label,
|
label,
|
||||||
@ -30,19 +29,24 @@ abstract class NumberInput<T : Number>(
|
|||||||
value,
|
value,
|
||||||
valueVal,
|
valueVal,
|
||||||
setValue,
|
setValue,
|
||||||
|
maxLength = null,
|
||||||
min,
|
min,
|
||||||
max,
|
max,
|
||||||
step,
|
step,
|
||||||
)
|
) {
|
||||||
|
companion object {
|
||||||
@Suppress("CssUnusedSymbol")
|
init {
|
||||||
// language=css
|
@Suppress("CssUnusedSymbol")
|
||||||
private fun style() = """
|
// language=css
|
||||||
.pw-number-input {
|
style("""
|
||||||
|
.pw-number-input {
|
||||||
width: 54px;
|
width: 54px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-number-input .pw-number-input-inner {
|
.pw-number-input .pw-number-input-inner {
|
||||||
padding-right: 1px;
|
padding-right: 1px;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -15,15 +15,18 @@ class TabContainer<T : Tab>(
|
|||||||
disabled: Val<Boolean> = falseVal(),
|
disabled: Val<Boolean> = falseVal(),
|
||||||
private val ctrl: TabController<T>,
|
private val ctrl: TabController<T>,
|
||||||
private val createWidget: (CoroutineScope, T) -> Widget,
|
private val createWidget: (CoroutineScope, T) -> Widget,
|
||||||
) : Widget(scope, listOf(::style), hidden, disabled) {
|
) : Widget(scope, hidden, disabled) {
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
div(className = "pw-tab-container") {
|
div {
|
||||||
div(className = "pw-tab-container-bar") {
|
className = "pw-tab-container"
|
||||||
|
|
||||||
|
div {
|
||||||
|
className = "pw-tab-container-bar"
|
||||||
|
|
||||||
for (tab in ctrl.tabs) {
|
for (tab in ctrl.tabs) {
|
||||||
span(
|
span {
|
||||||
className = "pw-tab-container-tab",
|
className = "pw-tab-container-tab"
|
||||||
title = tab.title,
|
title = tab.title
|
||||||
) {
|
|
||||||
textContent = tab.title
|
textContent = tab.title
|
||||||
|
|
||||||
observe(ctrl.activeTab) {
|
observe(ctrl.activeTab) {
|
||||||
@ -38,7 +41,9 @@ class TabContainer<T : Tab>(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
div(className = "pw-tab-container-panes") {
|
div {
|
||||||
|
className = "pw-tab-container-panes"
|
||||||
|
|
||||||
for (tab in ctrl.tabs) {
|
for (tab in ctrl.tabs) {
|
||||||
addChild(
|
addChild(
|
||||||
LazyLoader(
|
LazyLoader(
|
||||||
@ -57,26 +62,25 @@ class TabContainer<T : Tab>(
|
|||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
private const val ACTIVE_CLASS = "pw-active"
|
private const val ACTIVE_CLASS = "pw-active"
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnresolvedCustomProperty", "CssUnusedSymbol")
|
init {
|
||||||
// language=css
|
@Suppress("CssUnresolvedCustomProperty", "CssUnusedSymbol")
|
||||||
private fun style() = """
|
// language=css
|
||||||
.pw-tab-container {
|
style("""
|
||||||
|
.pw-tab-container {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-tab-container-bar {
|
.pw-tab-container-bar {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
height: 28px;
|
height: 28px;
|
||||||
min-height: 28px; /* To avoid bar from getting squished when pane content gets larger than pane in Firefox. */
|
min-height: 28px; /* To avoid bar from getting squished when pane content gets larger than pane in Firefox. */
|
||||||
padding: 3px 3px 0 3px;
|
padding: 3px 3px 0 3px;
|
||||||
border-bottom: var(--pw-border);
|
border-bottom: var(--pw-border);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-tab-container-tab {
|
.pw-tab-container-tab {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
@ -87,27 +91,30 @@ private fun style() = """
|
|||||||
background-color: var(--pw-tab-bg-color);
|
background-color: var(--pw-tab-bg-color);
|
||||||
color: var(--pw-tab-text-color);
|
color: var(--pw-tab-text-color);
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-tab-container-tab:hover {
|
.pw-tab-container-tab:hover {
|
||||||
background-color: var(--pw-tab-bg-color-hover);
|
background-color: var(--pw-tab-bg-color-hover);
|
||||||
color: var(--pw-tab-text-color-hover);
|
color: var(--pw-tab-text-color-hover);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-tab-container-tab.pw-active {
|
.pw-tab-container-tab.pw-active {
|
||||||
background-color: var(--pw-tab-bg-color-active);
|
background-color: var(--pw-tab-bg-color-active);
|
||||||
color: var(--pw-tab-text-color-active);
|
color: var(--pw-tab-text-color-active);
|
||||||
border-bottom-color: var(--pw-tab-bg-color-active);
|
border-bottom-color: var(--pw-tab-bg-color-active);
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-tab-container-panes {
|
.pw-tab-container-panes {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-tab-container-panes > * {
|
.pw-tab-container-panes > * {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -0,0 +1,42 @@
|
|||||||
|
package world.phantasmal.webui.widgets
|
||||||
|
|
||||||
|
import kotlinx.coroutines.CoroutineScope
|
||||||
|
import org.w3c.dom.HTMLInputElement
|
||||||
|
import world.phantasmal.observable.value.Val
|
||||||
|
import world.phantasmal.observable.value.falseVal
|
||||||
|
|
||||||
|
class TextInput(
|
||||||
|
scope: CoroutineScope,
|
||||||
|
hidden: Val<Boolean> = falseVal(),
|
||||||
|
disabled: Val<Boolean> = falseVal(),
|
||||||
|
label: String? = null,
|
||||||
|
labelVal: Val<String>? = null,
|
||||||
|
preferredLabelPosition: LabelPosition = LabelPosition.Before,
|
||||||
|
value: String? = null,
|
||||||
|
valueVal: Val<String>? = null,
|
||||||
|
setValue: ((String) -> Unit)? = null,
|
||||||
|
maxLength: Int? = null
|
||||||
|
) : Input<String>(
|
||||||
|
scope,
|
||||||
|
hidden,
|
||||||
|
disabled,
|
||||||
|
label,
|
||||||
|
labelVal,
|
||||||
|
preferredLabelPosition,
|
||||||
|
className = "pw-text-input",
|
||||||
|
inputClassName = "pw-number-text-inner",
|
||||||
|
inputType = "text",
|
||||||
|
value,
|
||||||
|
valueVal,
|
||||||
|
setValue,
|
||||||
|
maxLength,
|
||||||
|
min = null,
|
||||||
|
max = null,
|
||||||
|
step = null
|
||||||
|
) {
|
||||||
|
override fun getInputValue(input: HTMLInputElement): String = input.value
|
||||||
|
|
||||||
|
override fun setInputValue(input: HTMLInputElement, value: String) {
|
||||||
|
input.value = value
|
||||||
|
}
|
||||||
|
}
|
@ -11,15 +11,19 @@ class Toolbar(
|
|||||||
hidden: Val<Boolean> = falseVal(),
|
hidden: Val<Boolean> = falseVal(),
|
||||||
disabled: Val<Boolean> = falseVal(),
|
disabled: Val<Boolean> = falseVal(),
|
||||||
children: List<Widget>,
|
children: List<Widget>,
|
||||||
) : Widget(scope, listOf(::style), hidden, disabled) {
|
) : Widget(scope, hidden, disabled) {
|
||||||
private val childWidgets = children
|
private val childWidgets = children
|
||||||
|
|
||||||
override fun Node.createElement() =
|
override fun Node.createElement() =
|
||||||
div(className = "pw-toolbar") {
|
div {
|
||||||
|
className = "pw-toolbar"
|
||||||
|
|
||||||
childWidgets.forEach { child ->
|
childWidgets.forEach { child ->
|
||||||
// Group labelled controls and their labels together.
|
// Group labelled controls and their labels together.
|
||||||
if (child is LabelledControl && child.label != null) {
|
if (child is LabelledControl && child.label != null) {
|
||||||
div(className = "pw-toolbar-group") {
|
div {
|
||||||
|
className = "pw-toolbar-group"
|
||||||
|
|
||||||
when (child.preferredLabelPosition) {
|
when (child.preferredLabelPosition) {
|
||||||
LabelPosition.Before -> {
|
LabelPosition.Before -> {
|
||||||
addChild(child.label!!)
|
addChild(child.label!!)
|
||||||
@ -36,36 +40,40 @@ class Toolbar(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
@Suppress("CssUnusedSymbol", "CssUnresolvedCustomProperty")
|
companion object {
|
||||||
// language=css
|
init {
|
||||||
private fun style() = """
|
@Suppress("CssUnusedSymbol", "CssUnresolvedCustomProperty")
|
||||||
.pw-toolbar {
|
// language=css
|
||||||
|
style("""
|
||||||
|
.pw-toolbar {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
border-bottom: var(--pw-border);
|
border-bottom: var(--pw-border);
|
||||||
padding: 0 2px;
|
padding: 0 2px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-toolbar > * {
|
.pw-toolbar > * {
|
||||||
margin: 2px 1px;
|
margin: 2px 1px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-toolbar > .pw-toolbar-group {
|
.pw-toolbar > .pw-toolbar-group {
|
||||||
margin: 2px 3px;
|
margin: 2px 3px;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-toolbar > .pw-toolbar-group > * {
|
.pw-toolbar > .pw-toolbar-group > * {
|
||||||
margin: 0 2px;
|
margin: 0 2px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pw-toolbar .pw-input {
|
.pw-toolbar .pw-input {
|
||||||
height: 26px;
|
height: 26px;
|
||||||
|
}
|
||||||
|
""".trimIndent())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
"""
|
|
||||||
|
@ -2,10 +2,10 @@ package world.phantasmal.webui.widgets
|
|||||||
|
|
||||||
import kotlinx.browser.document
|
import kotlinx.browser.document
|
||||||
import kotlinx.coroutines.CoroutineScope
|
import kotlinx.coroutines.CoroutineScope
|
||||||
import kotlinx.dom.appendText
|
|
||||||
import kotlinx.dom.clear
|
import kotlinx.dom.clear
|
||||||
import org.w3c.dom.*
|
import org.w3c.dom.*
|
||||||
import world.phantasmal.core.disposable.disposable
|
import world.phantasmal.core.disposable.disposable
|
||||||
|
import world.phantasmal.observable.Observable
|
||||||
import world.phantasmal.observable.value.Val
|
import world.phantasmal.observable.value.Val
|
||||||
import world.phantasmal.observable.value.falseVal
|
import world.phantasmal.observable.value.falseVal
|
||||||
import world.phantasmal.observable.value.list.ListVal
|
import world.phantasmal.observable.value.list.ListVal
|
||||||
@ -16,7 +16,6 @@ import world.phantasmal.webui.DisposableContainer
|
|||||||
|
|
||||||
abstract class Widget(
|
abstract class Widget(
|
||||||
protected val scope: CoroutineScope,
|
protected val scope: CoroutineScope,
|
||||||
private val styles: List<() -> String> = emptyList(),
|
|
||||||
/**
|
/**
|
||||||
* By default determines the hidden attribute of its [element].
|
* By default determines the hidden attribute of its [element].
|
||||||
*/
|
*/
|
||||||
@ -33,13 +32,6 @@ abstract class Widget(
|
|||||||
private var resizeObserverInitialized = false
|
private var resizeObserverInitialized = false
|
||||||
|
|
||||||
private val elementDelegate = lazy {
|
private val elementDelegate = lazy {
|
||||||
// Add CSS declarations to stylesheet if this is the first time we're encountering them.
|
|
||||||
styles.forEach { style ->
|
|
||||||
if (STYLES_ADDED.add(style)) {
|
|
||||||
STYLE_EL.appendText(style())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
val el = document.createDocumentFragment().createElement()
|
val el = document.createDocumentFragment().createElement()
|
||||||
|
|
||||||
observe(hidden) { hidden ->
|
observe(hidden) { hidden ->
|
||||||
@ -101,6 +93,23 @@ abstract class Widget(
|
|||||||
super.internalDispose()
|
super.internalDispose()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected fun Node.text(observable: Observable<String>) {
|
||||||
|
observe(observable) { textContent = it }
|
||||||
|
}
|
||||||
|
|
||||||
|
protected fun HTMLElement.hidden(observable: Observable<Boolean>) {
|
||||||
|
observe(observable) { hidden = it }
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Appends a widget's element to the receiving node.
|
||||||
|
*/
|
||||||
|
protected fun <T : Widget> Node.addWidget(widget: T): T {
|
||||||
|
addDisposable(widget)
|
||||||
|
appendChild(widget.element)
|
||||||
|
return widget
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Adds a child widget to [children] and appends its element to the receiving node.
|
* Adds a child widget to [children] and appends its element to the receiving node.
|
||||||
*/
|
*/
|
||||||
@ -112,7 +121,7 @@ abstract class Widget(
|
|||||||
return child
|
return child
|
||||||
}
|
}
|
||||||
|
|
||||||
fun <T> Node.bindChildrenTo(
|
protected fun <T> Node.bindChildrenTo(
|
||||||
list: ListVal<T>,
|
list: ListVal<T>,
|
||||||
createChild: (T, Int) -> Node,
|
createChild: (T, Int) -> Node,
|
||||||
) {
|
) {
|
||||||
@ -198,7 +207,10 @@ abstract class Widget(
|
|||||||
document.head!!.append(el)
|
document.head!!.append(el)
|
||||||
el
|
el
|
||||||
}
|
}
|
||||||
private val STYLES_ADDED: MutableSet<() -> String> = mutableSetOf()
|
|
||||||
|
protected fun style(style: String) {
|
||||||
|
STYLE_EL.append(style)
|
||||||
|
}
|
||||||
|
|
||||||
protected fun setAncestorHidden(widget: Widget, hidden: Boolean) {
|
protected fun setAncestorHidden(widget: Widget, hidden: Boolean) {
|
||||||
widget._ancestorHidden.value = hidden
|
widget._ancestorHidden.value = hidden
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
package world.phantasmal.webui.widgets
|
package world.phantasmal.webui.widgets
|
||||||
|
|
||||||
|
import kotlinx.coroutines.CoroutineScope
|
||||||
import org.w3c.dom.Node
|
import org.w3c.dom.Node
|
||||||
import world.phantasmal.observable.value.Val
|
import world.phantasmal.observable.value.Val
|
||||||
import world.phantasmal.observable.value.falseVal
|
import world.phantasmal.observable.value.falseVal
|
||||||
@ -10,15 +11,16 @@ import world.phantasmal.webui.dom.div
|
|||||||
import kotlin.test.Test
|
import kotlin.test.Test
|
||||||
import kotlin.test.assertFalse
|
import kotlin.test.assertFalse
|
||||||
import kotlin.test.assertTrue
|
import kotlin.test.assertTrue
|
||||||
|
import kotlin.test.fail
|
||||||
|
|
||||||
class WidgetTests : TestSuite() {
|
class WidgetTests : TestSuite() {
|
||||||
@Test
|
@Test
|
||||||
fun ancestorHidden_and_selfOrAncestorHidden_should_update_when_hidden_changes() {
|
fun ancestorHidden_and_selfOrAncestorHidden_should_update_when_hidden_changes() = test {
|
||||||
val parentHidden = mutableVal(false)
|
val parentHidden = mutableVal(false)
|
||||||
val childHidden = mutableVal(false)
|
val childHidden = mutableVal(false)
|
||||||
val grandChild = DummyWidget()
|
val grandChild = DummyWidget(scope)
|
||||||
val child = DummyWidget(childHidden, grandChild)
|
val child = DummyWidget(scope, childHidden, grandChild)
|
||||||
val parent = disposer.add(DummyWidget(parentHidden, child))
|
val parent = disposer.add(DummyWidget(scope, parentHidden, child))
|
||||||
|
|
||||||
parent.element // Ensure widgets are fully initialized.
|
parent.element // Ensure widgets are fully initialized.
|
||||||
|
|
||||||
@ -50,9 +52,10 @@ class WidgetTests : TestSuite() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
fun added_child_widgets_should_have_ancestorHidden_and_selfOrAncestorHidden_set_correctly() {
|
fun added_child_widgets_should_have_ancestorHidden_and_selfOrAncestorHidden_set_correctly() =
|
||||||
val parent = disposer.add(DummyWidget(hidden = trueVal()))
|
test {
|
||||||
val child = parent.addChild(DummyWidget())
|
val parent = disposer.add(DummyWidget(scope, hidden = trueVal()))
|
||||||
|
val child = parent.addChild(DummyWidget(scope))
|
||||||
|
|
||||||
assertFalse(parent.ancestorHidden.value)
|
assertFalse(parent.ancestorHidden.value)
|
||||||
assertTrue(parent.selfOrAncestorHidden.value)
|
assertTrue(parent.selfOrAncestorHidden.value)
|
||||||
@ -61,6 +64,7 @@ class WidgetTests : TestSuite() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private inner class DummyWidget(
|
private inner class DummyWidget(
|
||||||
|
scope: CoroutineScope,
|
||||||
hidden: Val<Boolean> = falseVal(),
|
hidden: Val<Boolean> = falseVal(),
|
||||||
private val child: Widget? = null,
|
private val child: Widget? = null,
|
||||||
) : Widget(scope, hidden = hidden) {
|
) : Widget(scope, hidden = hidden) {
|
||||||
|
Loading…
Reference in New Issue
Block a user