From 26d5056c8eb4a11c4a6aa44afcae8b49c0646d85 Mon Sep 17 00:00:00 2001 From: Jake Wharton Date: Tue, 1 Oct 2024 08:10:36 -0400 Subject: [PATCH] Use a single toString implementation for protocol nodes (#2345) This saves even more. For emoji sample: - Before: 11299596 - After: 10953834 --- .../api/redwood-protocol-host.api | 3 ++- .../api/redwood-protocol-host.klib.api | 4 ++- .../protocol/host/HostProtocolAdapter.kt | 4 +-- .../redwood/protocol/host/ProtocolNode.kt | 11 +++++++- .../protocol/host/ChildrenNodeIndexTest.kt | 3 +-- .../tooling/codegen/protocolHostGeneration.kt | 25 +++++++------------ .../redwood/treehouse/FakeProtocolNode.kt | 4 +-- 7 files changed, 29 insertions(+), 25 deletions(-) diff --git a/redwood-protocol-host/api/redwood-protocol-host.api b/redwood-protocol-host/api/redwood-protocol-host.api index 7ed924dade..aa3fe0b2a8 100644 --- a/redwood-protocol-host/api/redwood-protocol-host.api +++ b/redwood-protocol-host/api/redwood-protocol-host.api @@ -44,9 +44,10 @@ public abstract class app/cash/redwood/protocol/host/ProtocolNode { public abstract fun detach ()V public final fun getId-0HhLjSo ()I public abstract fun getWidget ()Lapp/cash/redwood/widget/Widget; + public abstract fun getWidgetName ()Ljava/lang/String; public abstract fun getWidgetTag-BlhN7y0 ()I public final fun setId-ou3jOuA (I)V - public abstract fun toString ()Ljava/lang/String; + public final fun toString ()Ljava/lang/String; public final fun updateModifier (Lapp/cash/redwood/Modifier;)V public fun visitIds (Lapp/cash/redwood/protocol/host/IdVisitor;)V } diff --git a/redwood-protocol-host/api/redwood-protocol-host.klib.api b/redwood-protocol-host/api/redwood-protocol-host.klib.api index d0cbc89516..bb00a887e5 100644 --- a/redwood-protocol-host/api/redwood-protocol-host.klib.api +++ b/redwood-protocol-host/api/redwood-protocol-host.klib.api @@ -42,6 +42,8 @@ abstract class <#A: kotlin/Any> app.cash.redwood.protocol.host/ProtocolNode { // abstract val widget // app.cash.redwood.protocol.host/ProtocolNode.widget|{}widget[0] abstract fun (): app.cash.redwood.widget/Widget<#A> // app.cash.redwood.protocol.host/ProtocolNode.widget.|(){}[0] + abstract val widgetName // app.cash.redwood.protocol.host/ProtocolNode.widgetName|{}widgetName[0] + abstract fun (): kotlin/String // app.cash.redwood.protocol.host/ProtocolNode.widgetName.|(){}[0] abstract val widgetTag // app.cash.redwood.protocol.host/ProtocolNode.widgetTag|{}widgetTag[0] abstract fun (): app.cash.redwood.protocol/WidgetTag // app.cash.redwood.protocol.host/ProtocolNode.widgetTag.|(){}[0] @@ -52,7 +54,7 @@ abstract class <#A: kotlin/Any> app.cash.redwood.protocol.host/ProtocolNode { // abstract fun apply(app.cash.redwood.protocol/PropertyChange, app.cash.redwood.protocol.host/UiEventSink) // app.cash.redwood.protocol.host/ProtocolNode.apply|apply(app.cash.redwood.protocol.PropertyChange;app.cash.redwood.protocol.host.UiEventSink){}[0] abstract fun children(app.cash.redwood.protocol/ChildrenTag): app.cash.redwood.protocol.host/ProtocolChildren<#A>? // app.cash.redwood.protocol.host/ProtocolNode.children|children(app.cash.redwood.protocol.ChildrenTag){}[0] abstract fun detach() // app.cash.redwood.protocol.host/ProtocolNode.detach|detach(){}[0] - abstract fun toString(): kotlin/String // app.cash.redwood.protocol.host/ProtocolNode.toString|toString(){}[0] + final fun toString(): kotlin/String // app.cash.redwood.protocol.host/ProtocolNode.toString|toString(){}[0] final fun updateModifier(app.cash.redwood/Modifier) // app.cash.redwood.protocol.host/ProtocolNode.updateModifier|updateModifier(app.cash.redwood.Modifier){}[0] open fun visitIds(app.cash.redwood.protocol.host/IdVisitor) // app.cash.redwood.protocol.host/ProtocolNode.visitIds|visitIds(app.cash.redwood.protocol.host.IdVisitor){}[0] } diff --git a/redwood-protocol-host/src/commonMain/kotlin/app/cash/redwood/protocol/host/HostProtocolAdapter.kt b/redwood-protocol-host/src/commonMain/kotlin/app/cash/redwood/protocol/host/HostProtocolAdapter.kt index f66d6e15e7..2f5086a938 100644 --- a/redwood-protocol-host/src/commonMain/kotlin/app/cash/redwood/protocol/host/HostProtocolAdapter.kt +++ b/redwood-protocol-host/src/commonMain/kotlin/app/cash/redwood/protocol/host/HostProtocolAdapter.kt @@ -396,6 +396,8 @@ private class RootProtocolNode( Widget { override val widgetTag: WidgetTag get() = UnknownWidgetTag + override val widgetName: String get() = "RootProtocolNode" + private val children = ProtocolChildren(children) override fun apply(change: PropertyChange, eventSink: UiEventSink) { @@ -424,8 +426,6 @@ private class RootProtocolNode( override fun detach() { children.detach() } - - override fun toString() = "RootProtocolNode" } private const val REUSE_MODIFIER_TAG = -4_543_827 diff --git a/redwood-protocol-host/src/commonMain/kotlin/app/cash/redwood/protocol/host/ProtocolNode.kt b/redwood-protocol-host/src/commonMain/kotlin/app/cash/redwood/protocol/host/ProtocolNode.kt index 05260397c9..4a54fae09e 100644 --- a/redwood-protocol-host/src/commonMain/kotlin/app/cash/redwood/protocol/host/ProtocolNode.kt +++ b/redwood-protocol-host/src/commonMain/kotlin/app/cash/redwood/protocol/host/ProtocolNode.kt @@ -37,6 +37,8 @@ public abstract class ProtocolNode( ) { public abstract val widgetTag: WidgetTag + public abstract val widgetName: String + public abstract val widget: Widget /** The index of [widget] within its parent [container]. */ @@ -78,7 +80,14 @@ public abstract class ProtocolNode( public abstract fun detach() /** Human-readable name of this node along with [id] and [widgetTag]. */ - public abstract override fun toString(): String + public final override fun toString(): String = buildString { + append(widgetName) + append("(id=") + append(id.value) + append(", tag=") + append(widgetTag.value) + append(")") + } } /** @suppress */ diff --git a/redwood-protocol-host/src/commonTest/kotlin/app/cash/redwood/protocol/host/ChildrenNodeIndexTest.kt b/redwood-protocol-host/src/commonTest/kotlin/app/cash/redwood/protocol/host/ChildrenNodeIndexTest.kt index 9dacfb3a1d..db9087458b 100644 --- a/redwood-protocol-host/src/commonTest/kotlin/app/cash/redwood/protocol/host/ChildrenNodeIndexTest.kt +++ b/redwood-protocol-host/src/commonTest/kotlin/app/cash/redwood/protocol/host/ChildrenNodeIndexTest.kt @@ -128,6 +128,7 @@ class ChildrenNodeIndexTest { @OptIn(RedwoodCodegenApi::class) private class WidgetNode(override val widget: StringWidget) : ProtocolNode(Id(1)) { override val widgetTag: WidgetTag get() = WidgetTag(1) + override val widgetName: String get() = "WidgetNode" override fun apply(change: PropertyChange, eventSink: UiEventSink) { throw UnsupportedOperationException() @@ -139,8 +140,6 @@ private class WidgetNode(override val widget: StringWidget) : ProtocolNode { diff --git a/redwood-tooling-codegen/src/main/kotlin/app/cash/redwood/tooling/codegen/protocolHostGeneration.kt b/redwood-tooling-codegen/src/main/kotlin/app/cash/redwood/tooling/codegen/protocolHostGeneration.kt index ddf53707e9..26e9228529 100644 --- a/redwood-tooling-codegen/src/main/kotlin/app/cash/redwood/tooling/codegen/protocolHostGeneration.kt +++ b/redwood-tooling-codegen/src/main/kotlin/app/cash/redwood/tooling/codegen/protocolHostGeneration.kt @@ -330,6 +330,15 @@ internal fun generateProtocolNode( ) .build(), ) + .addProperty( + PropertySpec.builder("widgetName", STRING, OVERRIDE) + .getter( + FunSpec.getterBuilder() + .addStatement("return %S", widget.type.flatName) + .build(), + ) + .build(), + ) .addProperty( PropertySpec.builder("_widget", widgetType.copy(nullable = true), PRIVATE) .mutable(true) @@ -528,22 +537,6 @@ internal fun generateProtocolNode( .addStatement("_widget = null") .build(), ) - .addFunction( - FunSpec.builder("toString") - .addModifiers(OVERRIDE) - .returns(STRING) - // This explicit string builder usage allows sharing of strings in dex. - // See https://jakewharton.com/the-economics-of-generated-code/#string-duplication. - .beginControlFlow("return buildString") - .addStatement("append(%S)", type.simpleName) - .addStatement("""append("(id=")""") - .addStatement("append(id.value)") - .addStatement("""append(", tag=")""") - .addStatement("append(%L)", widget.tag) - .addStatement("append(')')") - .endControlFlow() - .build(), - ) .build(), ) } diff --git a/redwood-treehouse-host/src/commonTest/kotlin/app/cash/redwood/treehouse/FakeProtocolNode.kt b/redwood-treehouse-host/src/commonTest/kotlin/app/cash/redwood/treehouse/FakeProtocolNode.kt index ac89c94601..c5b69d0581 100644 --- a/redwood-treehouse-host/src/commonTest/kotlin/app/cash/redwood/treehouse/FakeProtocolNode.kt +++ b/redwood-treehouse-host/src/commonTest/kotlin/app/cash/redwood/treehouse/FakeProtocolNode.kt @@ -35,6 +35,8 @@ internal class FakeProtocolNode( id: Id, override val widgetTag: WidgetTag, ) : ProtocolNode(id) { + override val widgetName: String get() = "FakeProtocolNode" + override val widget = FakeWidget() override fun apply(change: PropertyChange, eventSink: UiEventSink) { @@ -50,6 +52,4 @@ internal class FakeProtocolNode( override fun detach() { } - - override fun toString() = "FakeProtocolNode(id=${id.value}, tag=${widgetTag.value})" }