I'm writing external declarations for LeafletJS 1.8.0, a JavaScript library, using Kotlin 1.6.21.
The Polyline
class has a function, getLatLngs()
that can return any of these types:
Array<LatLng>
Array<Array<LatLng>>
Array<Array<Array<LatLng>>>
Of course the setter is easy to overload to handle a type-union
open external class Polyline {
open fun setLatLngs(latlngs: Array<LatLng>): Polyline<T>
open fun setLatLngs(latlngs: Array<Array<LatLng>>): Polyline<T>
open fun setLatLngs(latlngs: Array<Array<Array<LatLng>>>): Polyline<T>
}
However it's not possible to overload the getter
open external class Polyline {
// ERROR: Conflicting overloads
open fun getLatLngs(): Array<LatLng>
open fun getLatLngs(): Array<Array<LatLng>>
open fun getLatLngs(): Array<Array<Array<LatLng>>>
}
As a compromise I can set the return type to dynamic
and add a comment so users can see the intention.
open external class Polyline {
open fun getLatLngs(): dynamic /* Array<LatLng> | Array<Array<LatLng>> | Array<Array<Array<LatLng>>> */
}
There's an open ticket KT-13108, and an update in November 2021 indicates direct Kotlin support for type unions won't be available until after Kotlin 1.7 is released.
Is there a better way of implementing the external function so the return type is type safe, or users can see the available types that might be returned, and handle each appropriately? What's the idiomatic practice?
Problem:
You are looking for an idiomatic way to describe union types for external declarations with:
Long story short, for any general representation of a JS union-type in Kotlin, it's not possible to hit all three of these criteria without having more information about the instances of those types (due to type-erasure which I'll explain). But, In your case and the vast majority of cases, there is a nice trick to do this by using Kotlin's extension functions.
Solution:
There are two cases that I'll explain what to do to hit these criteria as best as possible:
Cat | Dog | string
)Array<LatLng>
,Array<Array<LatLng>>
, andArray<Array<Array<LatLng>>>
each use Generics for their types)Union types that don't use Generics:
Say you had the Kotlin external declaration for
AnimalOwner
that is currently using dynamic as an output for itsgetPet
method:AnimalOwner.kt (draft)
One can specify an external interface to represent the output type. Then, using extension functions, one can define how to cast/morph each instance into each type (or return null if it can't):
Pet.kt
Now, we can replace the
dynamic
keyword inAnimalOwner
withPet
(the interface just created):AnimalOwner.kt (revised)
We can now use
AnimalOwner
by calling each extension function and checking if it is null or not:Union types that do use Generics:
This case is a little more complicated due to type-erasure. Let's use a similar example to
AnimalOwner
where now the owner is specifying lists of Dogs, Cats, or a String of animals:AnimalOwner.kt (draft)
At this point, if we attempt to do the same procedure to create an output type as before, we run into a problem when creating casting/morphing functions:
Pets.kt (ERROR)
Specifically, we must change the following code
this as? List<Cat>
andthis as? List<Dog>
because Generics Types likeList<T>
lose information on the generic parameterT
at runtime. This loss of information is called type-erasure (for more information see here). We must replace this withthis as? List<*>
for both extension methods because we can't know generics at runtime. This now creates another problem, as of now we cannot delineate between a list of Dogs and a list of Cats. This is where we require some outside knowledge of instances of these lists and how JavaScriptgetPets()
method treats them. This is project specific so for the sake of this example I am going to pretend I have done some research to determine this outside knowledge we speak of.So let's say we found out that our corresponding JavaScript method for
getPets()
always represents the returning of an empty list as list of Cats. Now we have enough information to correct our code to delineateList<Cats>
andList<Dog>
even though we only have access toList<*>
:Pets.kt (revised)
Now, in
AnimalOwner
, we can change the output type ofgetPets
fromdynamic
toPets
:AnimalOwner.kt (revised)
We can then use
AnimalOwner
the same way as the non-Generic case: