mirror of
https://github.com/jlengrand/kotlin.git
synced 2026-03-10 08:31:29 +00:00
Parcelize: Handle class hierarchies of Parcelers (KT-46567)
This commit is contained in:
committed by
Alexander Udalov
parent
709c127f1b
commit
bf7db84451
@@ -59,7 +59,11 @@ fun IrBuilderWithScope.parcelerCreate(parceler: IrClass, parcel: IrValueDeclarat
|
||||
|
||||
// object P: Parceler<T> { fun newArray(size: Int): Array<T> }
|
||||
fun IrBuilderWithScope.parcelerNewArray(parceler: IrClass?, size: IrValueDeclaration): IrExpression? =
|
||||
parceler?.parcelerSymbolByName("newArray")?.let { newArraySymbol ->
|
||||
parceler?.parcelerSymbolByName("newArray")?.takeIf {
|
||||
// The `newArray` method in `kotlinx.parcelize.Parceler` is stubbed out and we
|
||||
// have to produce a new implementation, unless the user overrides it.
|
||||
!it.owner.isFakeOverride || it.owner.resolveFakeOverride()?.parentClassOrNull?.fqNameWhenAvailable != PARCELER_FQNAME
|
||||
}?.let { newArraySymbol ->
|
||||
irCall(newArraySymbol).apply {
|
||||
dispatchReceiver = irGetObject(parceler.symbol)
|
||||
putValueArgument(0, irGet(size))
|
||||
@@ -101,7 +105,7 @@ fun IrBuilderWithScope.parcelableCreatorCreateFromParcel(creator: IrExpression,
|
||||
// has already done the work.
|
||||
private fun IrClass.parcelerSymbolByName(name: String): IrSimpleFunctionSymbol? =
|
||||
functions.firstOrNull { function ->
|
||||
!function.isFakeOverride && function.name.asString() == name && function.overridesFunctionIn(PARCELER_FQNAME)
|
||||
function.name.asString() == name && function.overridesFunctionIn(PARCELER_FQNAME)
|
||||
}?.symbol
|
||||
|
||||
fun IrSimpleFunction.overridesFunctionIn(fqName: FqName): Boolean =
|
||||
|
||||
@@ -190,6 +190,11 @@ public class ParcelBoxTestGenerated extends AbstractParcelBoxTest {
|
||||
runTest("plugins/android-extensions/android-extensions-compiler/testData/parcel/box/kt41553_2.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("kt46567.kt")
|
||||
public void testKt46567() throws Exception {
|
||||
runTest("plugins/android-extensions/android-extensions-compiler/testData/parcel/box/kt46567.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("listKinds.kt")
|
||||
public void testListKinds() throws Exception {
|
||||
runTest("plugins/android-extensions/android-extensions-compiler/testData/parcel/box/listKinds.kt");
|
||||
@@ -250,6 +255,11 @@ public class ParcelBoxTestGenerated extends AbstractParcelBoxTest {
|
||||
runTest("plugins/android-extensions/android-extensions-compiler/testData/parcel/box/newArray.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("newArrayParceler.kt")
|
||||
public void testNewArrayParceler() throws Exception {
|
||||
runTest("plugins/android-extensions/android-extensions-compiler/testData/parcel/box/newArrayParceler.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("nullableTypes.kt")
|
||||
public void testNullableTypes() throws Exception {
|
||||
runTest("plugins/android-extensions/android-extensions-compiler/testData/parcel/box/nullableTypes.kt");
|
||||
|
||||
@@ -190,6 +190,11 @@ public class ParcelIrBoxTestGenerated extends AbstractParcelIrBoxTest {
|
||||
runTest("plugins/android-extensions/android-extensions-compiler/testData/parcel/box/kt41553_2.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("kt46567.kt")
|
||||
public void testKt46567() throws Exception {
|
||||
runTest("plugins/android-extensions/android-extensions-compiler/testData/parcel/box/kt46567.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("listKinds.kt")
|
||||
public void testListKinds() throws Exception {
|
||||
runTest("plugins/android-extensions/android-extensions-compiler/testData/parcel/box/listKinds.kt");
|
||||
@@ -250,6 +255,11 @@ public class ParcelIrBoxTestGenerated extends AbstractParcelIrBoxTest {
|
||||
runTest("plugins/android-extensions/android-extensions-compiler/testData/parcel/box/newArray.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("newArrayParceler.kt")
|
||||
public void testNewArrayParceler() throws Exception {
|
||||
runTest("plugins/android-extensions/android-extensions-compiler/testData/parcel/box/newArrayParceler.kt");
|
||||
}
|
||||
|
||||
@TestMetadata("nullableTypes.kt")
|
||||
public void testNullableTypes() throws Exception {
|
||||
runTest("plugins/android-extensions/android-extensions-compiler/testData/parcel/box/nullableTypes.kt");
|
||||
|
||||
64
plugins/android-extensions/android-extensions-compiler/testData/parcel/box/kt46567.kt
vendored
Normal file
64
plugins/android-extensions/android-extensions-compiler/testData/parcel/box/kt46567.kt
vendored
Normal file
@@ -0,0 +1,64 @@
|
||||
// WITH_RUNTIME
|
||||
|
||||
@file:JvmName("TestKt")
|
||||
package test
|
||||
|
||||
import kotlinx.android.parcel.*
|
||||
import android.os.Parcel
|
||||
import android.os.Parcelable
|
||||
import java.util.Arrays
|
||||
|
||||
/**
|
||||
* Generic pair parceler
|
||||
* Create concrete object to use (see below)
|
||||
*/
|
||||
open class PairParceler<F: Any, S: Any>(private val firstParceler: Parceler<F>, private val secondParceler: Parceler<S>): Parceler<Pair<F, S>> {
|
||||
/**
|
||||
* Reads the [T] instance state from the [parcel], constructs the new [T] instance and returns it.
|
||||
*/
|
||||
override fun create(parcel: Parcel): Pair<F, S> =
|
||||
firstParceler.create(parcel) to secondParceler.create(parcel)
|
||||
|
||||
/**
|
||||
* Writes the [T] instance state to the [parcel].
|
||||
*/
|
||||
override fun Pair<F, S>.write(parcel: Parcel, flags: Int) {
|
||||
with(firstParceler) { this@write.first.write(parcel, 0) }
|
||||
with(secondParceler) { this@write.second.write(parcel, 0) }
|
||||
}
|
||||
}
|
||||
|
||||
object IntParceler: Parceler<Int> {
|
||||
/**
|
||||
* Reads the [T] instance state from the [parcel], constructs the new [T] instance and returns it.
|
||||
*/
|
||||
override fun create(parcel: Parcel): Int = parcel.readInt()
|
||||
|
||||
/**
|
||||
* Writes the [T] instance state to the [parcel].
|
||||
*/
|
||||
override fun Int.write(parcel: Parcel, flags: Int) {
|
||||
parcel.writeInt(this)
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* [Int] to [Int] pair parceler
|
||||
*/
|
||||
object IntToIntParceler: PairParceler<Int, Int>(IntParceler, IntParceler)
|
||||
|
||||
@Parcelize
|
||||
@TypeParceler<Pair<Int, Int>, IntToIntParceler>
|
||||
class A(val pair: Pair<Int, Int>): Parcelable
|
||||
|
||||
fun box() = parcelTest { parcel ->
|
||||
val a1 = A(1 to 2)
|
||||
a1.writeToParcel(parcel, 0)
|
||||
|
||||
val bytes = parcel.marshall()
|
||||
parcel.unmarshall(bytes, 0, bytes.size)
|
||||
parcel.setDataPosition(0)
|
||||
|
||||
val a2 = readFromParcel<A>(parcel)
|
||||
assert(a1.pair == a2.pair)
|
||||
}
|
||||
45
plugins/android-extensions/android-extensions-compiler/testData/parcel/box/newArrayParceler.kt
vendored
Normal file
45
plugins/android-extensions/android-extensions-compiler/testData/parcel/box/newArrayParceler.kt
vendored
Normal file
@@ -0,0 +1,45 @@
|
||||
// WITH_RUNTIME
|
||||
// IGNORE_BACKEND: JVM
|
||||
|
||||
@file:JvmName("TestKt")
|
||||
package test
|
||||
|
||||
import kotlinx.android.parcel.*
|
||||
import android.os.Parcel
|
||||
import android.os.Parcelable
|
||||
|
||||
abstract class UserParceler : Parceler<User> {
|
||||
override fun User.write(parcel: Parcel, flags: Int) {
|
||||
parcel.writeString(name)
|
||||
}
|
||||
|
||||
override fun newArray(size: Int): Array<User> {
|
||||
return Array(size + 1) { User(null) }
|
||||
}
|
||||
}
|
||||
|
||||
@Parcelize
|
||||
class User(val name: String?) : Parcelable {
|
||||
companion object : UserParceler() {
|
||||
override fun create(parcel: Parcel) = User(parcel.readString())
|
||||
}
|
||||
}
|
||||
|
||||
fun box() = parcelTest { parcel ->
|
||||
val user = User("John")
|
||||
val user2 = User("Joe")
|
||||
val array = arrayOf(user, user2)
|
||||
parcel.writeTypedArray(array, 0)
|
||||
|
||||
val bytes = parcel.marshall()
|
||||
parcel.unmarshall(bytes, 0, bytes.size)
|
||||
parcel.setDataPosition(0)
|
||||
|
||||
val creator = User::class.java.getDeclaredField("CREATOR").get(null) as Parcelable.Creator<User>
|
||||
val result = parcel.createTypedArray(creator)
|
||||
|
||||
assert(result.size == 3)
|
||||
assert(result[0].name == user.name)
|
||||
assert(result[1].name == user2.name)
|
||||
assert(result[2].name == null)
|
||||
}
|
||||
@@ -167,4 +167,4 @@ public final class test/Foo : java/lang/Object, android/os/Parcelable {
|
||||
RETURN
|
||||
LABEL (L1)
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user