Run-time Types
Types can be represented at run-time.
To create a type value, use the constructor function Type<T>(), which accepts the static type as a type argument.
This is similar to e.g. T.self in Swift, T::class/KClass<T> in Kotlin, and T.class/Class<T> in Java.
For example, to represent the type Int at run-time:
_10let intType: Type = Type<Int>()
This works for both built-in and user-defined types. For example, to get the type value for a resource:
_10resource Collectible {}_10_10let collectibleType = Type<@Collectible>()_10_10// `collectibleType` has type `Type`
Type values are comparable.
_10_10Type<Int>() == Type<Int>()_10_10Type<Int>() != Type<String>()
The method fun isSubtype(of: Type): Bool can be used to compare the run-time types of values.
_10Type<Int>().isSubtype(of: Type<Int>()) // true_10_10Type<Int>().isSubtype(of: Type<String>()) // false_10_10Type<Int>().isSubtype(of: Type<Int?>()) // true
To get the run-time type's fully qualified type identifier, use the let identifier: String field:
_10let type = Type<Int>()_10type.identifier  // is "Int"
_10// in account 0x1_10_10struct Test {}_10_10let type = Type<Test>()_10type.identifier  // is "A.0000000000000001.Test"
Getting the Type from a Value
The method fun getType(): Type can be used to get the runtime type of a value.
_10let something = "hello"_10_10let type: Type = something.getType()_10// `type` is `Type<String>()`
This method returns the concrete run-time type of the object, not the static type.
_10// Declare a variable named `something` that has the *static* type `AnyResource`_10// and has a resource of type `Collectible`_10//_10let something: @AnyResource <- create Collectible()_10_10// The resource's concrete run-time type is `Collectible`_10//_10let type: Type = something.getType()_10// `type` is `Type<@Collectible>()`
Constructing a Run-time Type
Run-time types can also be constructed from type identifier strings using built-in constructor functions.
_10fun CompositeType(_ identifier: String): Type?_10fun InterfaceType(_ identifier: String): Type?_10fun RestrictedType(identifier: String?, restrictions: [String]): Type?
Given a type identifier (as well as a list of identifiers for restricting interfaces
in the case of RestrictedType), these functions will look up nominal types and
produce their run-time equivalents. If the provided identifiers do not correspond
to any types, or (in the case of RestrictedType) the provided combination of
identifiers would not type-check statically, these functions will produce nil.
_10struct Test {}_10struct interface I {}_10let type: Type = CompositeType("A.0000000000000001.Test")_10// `type` is `Type<Test>`_10_10let type2: Type = RestrictedType(_10    identifier: type.identifier,_10    restrictions: ["A.0000000000000001.I"]_10)_10// `type2` is `Type<Test{I}>`
Other built-in functions will construct compound types from other run-types.
_10fun OptionalType(_ type: Type): Type_10fun VariableSizedArrayType(_ type: Type): Type_10fun ConstantSizedArrayType(type: Type, size: Int): Type_10fun FunctionType(parameters: [Type], return: Type): Type_10// returns `nil` if `key` is not valid dictionary key type_10fun DictionaryType(key: Type, value: Type): Type?_10// returns `nil` if `type` is not a reference type_10fun CapabilityType(_ type: Type): Type?_10fun ReferenceType(authorized: bool, type: Type): Type
Asserting the Type of a Value
The method fun isInstance(_ type: Type): Bool can be used to check if a value has a certain type,
using the concrete run-time type, and considering subtyping rules,
_19// Declare a variable named `collectible` that has the *static* type `Collectible`_19// and has a resource of type `Collectible`_19//_19let collectible: @Collectible <- create Collectible()_19_19// The resource is an instance of type `Collectible`,_19// because the concrete run-time type is `Collectible`_19//_19collectible.isInstance(Type<@Collectible>())  // is `true`_19_19// The resource is an instance of type `AnyResource`,_19// because the concrete run-time type `Collectible` is a subtype of `AnyResource`_19//_19collectible.isInstance(Type<@AnyResource>())  // is `true`_19_19// The resource is *not* an instance of type `String`,_19// because the concrete run-time type `Collectible` is *not* a subtype of `String`_19//_19collectible.isInstance(Type<String>())  // is `false`
Note that the concrete run-time type of the object is used, not the static type.
_19// Declare a variable named `something` that has the *static* type `AnyResource`_19// and has a resource of type `Collectible`_19//_19let something: @AnyResource <- create Collectible()_19_19// The resource is an instance of type `Collectible`,_19// because the concrete run-time type is `Collectible`_19//_19something.isInstance(Type<@Collectible>())  // is `true`_19_19// The resource is an instance of type `AnyResource`,_19// because the concrete run-time type `Collectible` is a subtype of `AnyResource`_19//_19something.isInstance(Type<@AnyResource>())  // is `true`_19_19// The resource is *not* an instance of type `String`,_19// because the concrete run-time type `Collectible` is *not* a subtype of `String`_19//_19something.isInstance(Type<String>())  // is `false`
For example, this allows implementing a marketplace sale resource:
_67access(all) resource SimpleSale {_67_67    /// The resource for sale._67    /// Once the resource is sold, the field becomes `nil`._67    ///_67    access(all) var resourceForSale: @AnyResource?_67_67    /// The price that is wanted for the purchase of the resource._67    ///_67    access(all) let priceForResource: UFix64_67_67    /// The type of currency that is required for the purchase._67    ///_67    access(all) let requiredCurrency: Type_67    access(all) let paymentReceiver: Capability<&{FungibleToken.Receiver}>_67_67    /// `paymentReceiver` is the capability that will be borrowed_67    /// once a valid purchase is made._67    /// It is expected to target a resource that allows depositing the paid amount_67    /// (a vault which has the type in `requiredCurrency`)._67    ///_67    init(_67        resourceForSale: @AnyResource,_67        priceForResource: UFix64,_67        requiredCurrency: Type,_67        paymentReceiver: Capability<&{FungibleToken.Receiver}>_67    ) {_67        self.resourceForSale <- resourceForSale_67        self.priceForResource = priceForResource_67        self.requiredCurrency = requiredCurrency_67        self.paymentReceiver = paymentReceiver_67    }_67_67    destroy() {_67        // When this sale resource is destroyed,_67        // also destroy the resource for sale._67        // Another option could be to transfer it back to the seller._67        destroy self.resourceForSale_67    }_67_67    /// buyObject allows purchasing the resource for sale by providing_67    /// the required funds._67    /// If the purchase succeeds, the resource for sale is returned._67    /// If the purchase fails, the program aborts._67    ///_67    access(all) fun buyObject(with funds: @FungibleToken.Vault): @AnyResource {_67        pre {_67            // Ensure the resource is still up for sale_67            self.resourceForSale != nil: "The resource has already been sold"_67            // Ensure the paid funds have the right amount_67            funds.balance >= self.priceForResource: "Payment has insufficient amount"_67            // Ensure the paid currency is correct_67            funds.isInstance(self.requiredCurrency): "Incorrect payment currency"_67        }_67_67        // Transfer the paid funds to the payment receiver_67        // by borrowing the payment receiver capability of this sale resource_67        // and depositing the payment into it_67_67        let receiver = self.paymentReceiver.borrow()_67            ?? panic("failed to borrow payment receiver capability")_67_67        receiver.deposit(from: <-funds)_67        let resourceForSale <- self.resourceForSale <- nil_67        return <-resourceForSale_67    }_67}