From 052acf1479737032b64bd0c20724377d4661951f Mon Sep 17 00:00:00 2001 From: AhmedShaer Date: Tue, 26 Sep 2023 14:00:12 +0300 Subject: [PATCH 1/3] Support RTL layout direction in redwood-layout-dom (#1509) --- .../dom/HTMLElementRedwoodLayoutWidgetFactory.kt | 4 ++-- .../kotlin/app/cash/redwood/layout/dom/utils.kt | 13 +++++++++++++ 2 files changed, 15 insertions(+), 2 deletions(-) diff --git a/redwood-layout-dom/src/commonMain/kotlin/app/cash/redwood/layout/dom/HTMLElementRedwoodLayoutWidgetFactory.kt b/redwood-layout-dom/src/commonMain/kotlin/app/cash/redwood/layout/dom/HTMLElementRedwoodLayoutWidgetFactory.kt index 8acf8e38e2..a284f93ca4 100644 --- a/redwood-layout-dom/src/commonMain/kotlin/app/cash/redwood/layout/dom/HTMLElementRedwoodLayoutWidgetFactory.kt +++ b/redwood-layout-dom/src/commonMain/kotlin/app/cash/redwood/layout/dom/HTMLElementRedwoodLayoutWidgetFactory.kt @@ -77,8 +77,8 @@ private class HTMLFlexContainer( override fun margin(margin: Margin) { value.style.apply { - marginLeft = margin.start.toPxString() - marginRight = margin.end.toPxString() + marginInlineStart = margin.start.toPxString() + marginInlineEnd = margin.end.toPxString() marginTop = margin.top.toPxString() marginBottom = margin.bottom.toPxString() } diff --git a/redwood-layout-dom/src/commonMain/kotlin/app/cash/redwood/layout/dom/utils.kt b/redwood-layout-dom/src/commonMain/kotlin/app/cash/redwood/layout/dom/utils.kt index 12e65b2171..ee037f4ab0 100644 --- a/redwood-layout-dom/src/commonMain/kotlin/app/cash/redwood/layout/dom/utils.kt +++ b/redwood-layout-dom/src/commonMain/kotlin/app/cash/redwood/layout/dom/utils.kt @@ -22,6 +22,7 @@ import app.cash.redwood.layout.api.Overflow import app.cash.redwood.ui.Density import app.cash.redwood.ui.Dp import kotlin.math.roundToInt +import org.w3c.dom.css.CSSStyleDeclaration internal fun Dp.toPxString(): String = with(Density(1.0)) { "${toPx().roundToInt()}px" @@ -56,3 +57,15 @@ internal fun Overflow.toCss() = when (this) { Overflow.Scroll -> "scroll" else -> throw AssertionError() } + +internal var CSSStyleDeclaration.marginInlineStart: String + get() = this.getPropertyValue("margin-inline-start") + set(value) { + this.setProperty("margin-inline-start", value) + } + +internal var CSSStyleDeclaration.marginInlineEnd: String + get() = this.getPropertyValue("margin-inline-end") + set(value) { + this.setProperty("margin-inline-end", value) + } From 52fcbda39ad05017cfd1573dfb032e585ca520c8 Mon Sep 17 00:00:00 2001 From: Jingwei Date: Tue, 26 Sep 2023 11:36:10 -0700 Subject: [PATCH 2/3] Add support to save non-mutable int in StateSnapshot (#1508) * Add support to save non-mutable int in StateSnapshot * clean up & add tests * comments --- .../redwood/treehouse/treehouseCompose.kt | 2 +- redwood-treehouse/build.gradle | 7 ++ .../cash/redwood/treehouse/StateSnapshot.kt | 32 ++++++--- .../redwood/treehouse/StateSnapshotTest.kt | 72 +++++++++++++++++++ 4 files changed, 102 insertions(+), 11 deletions(-) create mode 100644 redwood-treehouse/src/commonTest/kotlin/app/cash/redwood/treehouse/StateSnapshotTest.kt diff --git a/redwood-treehouse-guest/src/commonMain/kotlin/app/cash/redwood/treehouse/treehouseCompose.kt b/redwood-treehouse-guest/src/commonMain/kotlin/app/cash/redwood/treehouse/treehouseCompose.kt index 0e9ec91a45..292ca50101 100644 --- a/redwood-treehouse-guest/src/commonMain/kotlin/app/cash/redwood/treehouse/treehouseCompose.kt +++ b/redwood-treehouse-guest/src/commonMain/kotlin/app/cash/redwood/treehouse/treehouseCompose.kt @@ -109,7 +109,7 @@ private class RedwoodZiplineTreehouseUi( ) } - override fun snapshotState(): StateSnapshot? { + override fun snapshotState(): StateSnapshot { val savedState = saveableStateRegistry.performSave() return savedState.toStateSnapshot() } diff --git a/redwood-treehouse/build.gradle b/redwood-treehouse/build.gradle index b4b1f8bd43..5cb5e62f7a 100644 --- a/redwood-treehouse/build.gradle +++ b/redwood-treehouse/build.gradle @@ -21,6 +21,13 @@ kotlin { api libs.zipline } } + + commonTest { + dependencies { + implementation libs.kotlin.test + implementation libs.assertk + } + } } } diff --git a/redwood-treehouse/src/commonMain/kotlin/app/cash/redwood/treehouse/StateSnapshot.kt b/redwood-treehouse/src/commonMain/kotlin/app/cash/redwood/treehouse/StateSnapshot.kt index a276588b6f..fa0b2393a2 100644 --- a/redwood-treehouse/src/commonMain/kotlin/app/cash/redwood/treehouse/StateSnapshot.kt +++ b/redwood-treehouse/src/commonMain/kotlin/app/cash/redwood/treehouse/StateSnapshot.kt @@ -18,20 +18,28 @@ package app.cash.redwood.treehouse import androidx.compose.runtime.MutableState import androidx.compose.runtime.mutableStateOf import kotlin.jvm.JvmInline +import kotlinx.serialization.Contextual import kotlinx.serialization.Serializable import kotlinx.serialization.json.JsonArray import kotlinx.serialization.json.JsonElement import kotlinx.serialization.json.JsonPrimitive import kotlinx.serialization.json.booleanOrNull import kotlinx.serialization.json.doubleOrNull +import kotlinx.serialization.json.intOrNull @Serializable public class StateSnapshot( - public val content: Map>, + public val content: Map>, ) { - public fun toValuesMap(): Map>? { + public fun toValuesMap(): Map> { return content.mapValues { entry -> - entry.value.map { mutableStateOf(it.fromJsonElement()) } + entry.value.map { + if (it.isMutableState) { + mutableStateOf(it.value) + } else { + it.value.fromJsonElement() + } + } } } @@ -48,8 +56,8 @@ public fun Map>.toStateSnapshot(): StateSnapshot = StateSnaps mapValues { entry -> entry.value.map { element -> when (element) { - is MutableState<*> -> element.value.toJsonElement() - else -> error("unexpected type: $this") + is MutableState<*> -> Saveable(true, element.value.toJsonElement()) + else -> Saveable(false, element.toJsonElement()) } } }, @@ -58,6 +66,7 @@ public fun Map>.toStateSnapshot(): StateSnapshot = StateSnaps private fun Any?.toJsonElement(): JsonElement { return when (this) { is String -> JsonPrimitive(this) + is Int -> JsonPrimitive(this) is List<*> -> JsonArray(map { it.toJsonElement() }) is JsonElement -> this else -> error("unexpected type: $this") @@ -68,11 +77,9 @@ private fun Any?.toJsonElement(): JsonElement { private fun JsonElement?.fromJsonElement(): Any { return when (this) { is JsonPrimitive -> { - if (this.isString) { - return content - } - return booleanOrNull ?: doubleOrNull ?: error("unexpected type: $this") - // TODO add other primitive types (double, float, long) when needed + if (this.isString) return content + return booleanOrNull ?: doubleOrNull ?: intOrNull ?: error("unexpected type: $this") + // TODO add other primitive types (float, long) when needed } is JsonArray -> listOf({ this.forEach { it.toJsonElement() } }) @@ -81,3 +88,8 @@ private fun JsonElement?.fromJsonElement(): Any { else -> error("unexpected type: $this") } } + +public data class Saveable( + val isMutableState: Boolean, + val value: JsonElement, +) diff --git a/redwood-treehouse/src/commonTest/kotlin/app/cash/redwood/treehouse/StateSnapshotTest.kt b/redwood-treehouse/src/commonTest/kotlin/app/cash/redwood/treehouse/StateSnapshotTest.kt new file mode 100644 index 0000000000..c25c93f534 --- /dev/null +++ b/redwood-treehouse/src/commonTest/kotlin/app/cash/redwood/treehouse/StateSnapshotTest.kt @@ -0,0 +1,72 @@ +/* + * Copyright (C) 2023 Square, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package app.cash.redwood.treehouse + +import androidx.compose.runtime.MutableState +import androidx.compose.runtime.mutableStateOf +import assertk.assertThat +import assertk.assertions.containsOnly +import assertk.assertions.isEqualTo +import kotlin.test.Test +import kotlin.test.assertTrue +import kotlinx.serialization.json.JsonPrimitive + +class StateSnapshotTest { + + @Test + fun toValueMapWorksAsExpected() { + val stateSnapshot = stateSnapshot() + val valuesMap = stateSnapshot.toValuesMap() + assertThat(valuesMap.entries.size).isEqualTo(4) + assertTrue(valuesMap["key1"]!![0] is MutableState<*>) + assertThat((valuesMap["key1"]!![0] as MutableState<*>).value).isEqualTo(JsonPrimitive(1)) + + assertThat(valuesMap["key2"]).isEqualTo(listOf(1.0)) + + assertThat(valuesMap["key3"]!![0] is MutableState<*>) + assertThat((valuesMap["key3"]!![0] as MutableState<*>).value).isEqualTo(JsonPrimitive("str")) + + assertThat(valuesMap["key4"]).isEqualTo(listOf("str")) + } + + @Test + fun toStateSnapshotWorksAsExpected() { + val storedStateSnapshot = storedStateSnapshot() + val stateSnapshot = storedStateSnapshot.toStateSnapshot() + assertThat(stateSnapshot.content).containsOnly( + "key1" to listOf(Saveable(true, JsonPrimitive(1))), + "key2" to listOf(Saveable(false, JsonPrimitive(1))), + "key3" to listOf(Saveable(true, JsonPrimitive("str"))), + "key4" to listOf(Saveable(false, JsonPrimitive("str"))), + ) + } + + private fun stateSnapshot() = StateSnapshot( + mapOf( + "key1" to listOf(Saveable(true, JsonPrimitive(1))), + "key2" to listOf(Saveable(false, JsonPrimitive(1))), + "key3" to listOf(Saveable(true, JsonPrimitive("str"))), + "key4" to listOf(Saveable(false, JsonPrimitive("str"))), + ), + ) + + private fun storedStateSnapshot() = mapOf( + "key1" to listOf(mutableStateOf(1)), + "key2" to listOf(1), + "key3" to listOf(mutableStateOf("str")), + "key4" to listOf("str"), + ) +} From 0bc7ce9171beccb73a9a09cba88083d99247cadb Mon Sep 17 00:00:00 2001 From: "renovate[bot]" <29139614+renovate[bot]@users.noreply.github.com> Date: Tue, 26 Sep 2023 20:34:46 +0000 Subject: [PATCH 3/3] Update dependency org.jetbrains.compose:compose-gradle-plugin to v1.5.2 (#1511) Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> --- gradle/libs.versions.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index a79d15d7dd..c209da159e 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -3,7 +3,7 @@ kotlin = "1.9.10" kotlinx-coroutines = "1.7.3" kotlinx-serialization = "1.6.0" androidx-activity = "1.7.2" -jbCompose = "1.5.1" +jbCompose = "1.5.2" lint = "31.1.1" zipline = "1.3.0" coil = "2.4.0"