1--- 2title: Module API Reference 3description: An API reference of Expo modules API. 4--- 5 6import { CodeBlocksTable } from '~/components/plugins/CodeBlocksTable'; 7import { APIBox } from '~/components/plugins/APIBox'; 8import { PlatformTags } from '~/ui/components/Tag'; 9import { APIMethod } from '~/components/plugins/api/APISectionMethods'; 10 11The native modules API is an abstraction layer on top of [JSI](https://reactnative.dev/architecture/glossary#javascript-interfaces-jsi) and other low-level primitives that React Native is built upon. It is built with modern languages (Swift and Kotlin) and provides an easy to use and convenient API that is consistent across platforms where possible. 12 13## Definition Components 14 15As you might have noticed in the snippets on the [Get Started](./get-started.mdx) page, each module class must implement the `definition` function. 16The module definition consists of the DSL components that describe the module's functionality and behavior. 17 18<APIBox header="Name"> 19 20Sets the name of the module that JavaScript code will use to refer to the module. Takes a string as an argument. Can be inferred from module's class name, but it's recommended to set it explicitly for clarity. 21 22```swift Swift / Kotlin 23Name("MyModuleName") 24``` 25 26</APIBox> 27<APIBox header="Constants"> 28 29Sets constant properties on the module. Can take a dictionary or a closure that returns a dictionary. 30 31<CodeBlocksTable> 32 33```swift 34// Created from the dictionary 35Constants([ 36 "PI": Double.pi 37]) 38 39// or returned by the closure 40Constants { 41 return [ 42 "PI": Double.pi 43 ] 44} 45``` 46 47```kotlin 48// Passed as arguments 49Constants( 50 "PI" to kotlin.math.PI 51) 52 53// or returned by the closure 54Constants { 55 return@Constants mapOf( 56 "PI" to kotlin.math.PI 57 ) 58} 59``` 60 61</CodeBlocksTable> 62</APIBox> 63<APIBox header="Function"> 64 65Defines a native synchronous function that will be exported to JavaScript. Synchronous means that when the function is executed in JavaScript, its native code is run on the same thread and blocks further execution of the script until the native function returns. 66 67#### Arguments 68 69- **name**: `String` — Name of the function that you'll call from JavaScript. 70- **body**: `(args...) -> ReturnType` — The closure to run when the function is called. 71 72The function can receive up to 8 arguments. This is due to the limitations of generics in both Swift and Kotlin, because this component must be implemented separately for each arity. 73 74See the [Argument Types](#argument-types) section for more details on what types can be used in the function body. 75 76<CodeBlocksTable> 77 78```swift 79Function("syncFunction") { (message: String) in 80 return message 81} 82``` 83 84```kotlin 85Function("syncFunction") { message: String -> 86 return@Function message 87} 88``` 89 90</CodeBlocksTable> 91 92```js JavaScript 93import { requireNativeModule } from 'expo-modules-core'; 94 95// Assume that we have named the module "MyModule" 96const MyModule = requireNativeModule('MyModule'); 97 98function getMessage() { 99 return MyModule.syncFunction('bar'); 100} 101``` 102 103</APIBox> 104<APIBox header="AsyncFunction"> 105 106Defines a JavaScript function that always returns a `Promise` and whose native code is by default dispatched on the different thread than the JavaScript runtime runs on. 107 108#### Arguments 109 110- **name**: `String` — Name of the function that you'll call from JavaScript. 111- **body**: `(args...) -> ReturnType` — The closure to run when the function is called. 112 113If the type of the last argument is `Promise`, the function will wait for the promise to be resolved or rejected before the response is passed back to JavaScript. Otherwise, the function is immediately resolved with the returned value or rejected if it throws an exception. 114The function can receive up to 8 arguments (including the promise). 115 116See the [Argument Types](#argument-types) section for more details on what types can be used in the function body. 117 118It is recommended to use `AsyncFunction` over `Function` when it: 119 120- does I/O bound tasks such as sending network requests or interacting with the file system 121- needs to be run on different thread, e.g. the main UI thread for UI-related tasks 122- is an extensive or long-lasting operation that would block the JavaScript thread which in turn would reduce the responsiveness of the application 123 124<CodeBlocksTable> 125 126```swift 127AsyncFunction("asyncFunction") { (message: String, promise: Promise) in 128 DispatchQueue.main.asyncAfter(deadline: .now() + 3.0) { 129 promise.resolve(message) 130 } 131} 132``` 133 134```kotlin 135AsyncFunction("asyncFunction") { message: String, promise: Promise -> 136 launch(Dispatchers.Main) { 137 promise.resolve(message) 138 } 139} 140``` 141 142</CodeBlocksTable> 143 144```js JavaScript 145import { requireNativeModule } from 'expo-modules-core'; 146 147// Assume that we have named the module "MyModule" 148const MyModule = requireNativeModule('MyModule'); 149 150async function getMessageAsync() { 151 return await MyModule.asyncFunction('bar'); 152} 153``` 154 155--- 156 157#### Kotlin coroutines <PlatformTags prefix="" platforms={['android']} /> 158 159`AsyncFunction` can receive a suspendable body on Android. However, it has to be passed in the infix notation after the `Coroutine` block. You can read more about suspendable functions and coroutines on [coroutine overview](https://kotlinlang.org/docs/coroutines-overview.html). 160 161`AsyncFunction` with suspendable body can't receive `Promise` as an argument. It uses a suspension mechanism to execute asynchronous calls. 162The function is immediately resolved with the returned value of the provided suspendable block or rejected if it throws an exception. The function can receive up to 8 arguments. 163 164By default, suspend functions are dispatched on the module's coroutine scope. Moreover, every other suspendable function called from the body block is run within the same scope. 165This scope's lifecycle is bound to the module's lifecycle - all unfinished suspend functions will be canceled when the module is deallocated. 166 167```kotlin Kotlin 168AsyncFunction("suspendFunction") Coroutine { message: String -> 169 launch { 170 return@Coroutine message 171 } 172} 173``` 174 175</APIBox> 176<APIBox header="Events"> 177 178Defines event names that the module can send to JavaScript. 179 180> **Note**: This component can be used inside of the [`View`](#view) block to define callback names. See [`View callbacks`](#view-callbacks) 181 182<CodeBlocksTable> 183 184```swift 185Events("onCameraReady", "onPictureSaved", "onBarCodeScanned") 186``` 187 188```kotlin 189Events("onCameraReady", "onPictureSaved", "onBarCodeScanned") 190``` 191 192</CodeBlocksTable> 193 194See [Sending events](#sending-events) to learn how to send events from the native code to JavaScript/TypeScript. 195 196</APIBox> 197<APIBox header="Property"> 198 199Defines a new property directly on the JavaScript object that represents a native module. It is the same as calling [`Object.defineProperty`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/defineProperty) on the module object. 200 201To declare a read-only property, you can use a shorthanded syntax that requires two arguments: 202 203- **name**: `String` — Name of the property that you'll use from JavaScript. 204- **getter**: `() -> PropertyType` — The closure to run when the getter for a property was called. 205 206<CodeBlocksTable> 207 208```swift 209Property("foo") { 210 return "bar" 211} 212``` 213 214```kotlin 215Property("foo") { 216 return@Property "bar" 217} 218``` 219 220</CodeBlocksTable> 221 222In the case of the mutable property, both the getter and the setter closure are needed (using the syntax below is also possible to declare a property with only a setter): 223 224- **name**: `String` — Name of the property that you'll use from JavaScript. 225- **getter**: `() -> ProperyType` — The closure to run when the getter for a property was called. 226- **setter**: `(newValue: PropertyType) -> void` — The closure to run when the setter for a property was called. 227 228<CodeBlocksTable> 229 230```swift 231Property("foo") 232 .get { return "bar" } 233 .set { (newValue: String) in 234 // do something with new value 235 } 236``` 237 238```kotlin 239Property("foo") 240 .get { return@get "bar" } 241 .set { newValue: String -> 242 // do something with new value 243 } 244``` 245 246</CodeBlocksTable> 247 248```js JavaScript 249import { requireNativeModule } from 'expo-modules-core'; 250 251// Assume that we have named the module "MyModule" 252const MyModule = requireNativeModule('MyModule'); 253 254// Obtain the property value 255MyModule.foo; 256 257// Set a new value 258MyModule.foo = 'foobar'; 259``` 260 261</APIBox> 262<APIBox header="ViewManager"> 263 264> **warning** **Deprecated**: To better integrate with [React Native's new architecture (Fabric)](https://reactnative.dev/architecture/fabric-renderer) and its recycling mechanism, as of SDK 47 the `ViewManager` component is deprecated in favor of [`View`](#view) with a view class passed as the first argument. This component will be removed in SDK 48. 265 266Enables the module to be used as a view manager. The view manager definition is built from the definition components used in the closure passed to `ViewManager`. Definition components that are accepted as part of the view manager definition: [`View`](#view), [`Prop`](#prop). 267 268<CodeBlocksTable> 269 270```swift 271ViewManager { 272 View { 273 MyNativeView() 274 } 275 276 Prop("isHidden") { (view: UIView, hidden: Bool) in 277 view.isHidden = hidden 278 } 279} 280``` 281 282```kotlin 283ViewManager { 284 View { context -> 285 MyNativeView(context) 286 } 287 288 Prop("isHidden") { view: View, hidden: Bool -> 289 view.isVisible = !hidden 290 } 291} 292``` 293 294</CodeBlocksTable> 295</APIBox> 296<APIBox header="View"> 297 298Enables the module to be used as a native view. Definition components that are accepted as part of the view definition: [`Prop`](#prop), [`Events`](#events). 299 300#### Arguments 301 302- **viewType** — The class of the native view that will be rendered. Note: On Android, the provided class must inherit from the [`ExpoView`](#expoview), on iOS it's optional. See [`Extending ExpoView`](#extending--expoview). 303- **definition**: `() -> ViewDefinition` — A builder of the view definition. 304 305<CodeBlocksTable> 306 307```swift 308View(UITextView.self) { 309 Prop("text") { ... } 310} 311``` 312 313```kotlin 314View(TextView::class) { 315 Prop("text") { ... } 316} 317``` 318 319</CodeBlocksTable> 320 321> Support for rendering SwiftUI views is planned. For now, you can use [`UIHostingController`](https://developer.apple.com/documentation/swiftui/uihostingcontroller) and add its content view to your UIKit view. 322 323</APIBox> 324<APIBox header="Prop"> 325 326Defines a setter for the view prop of given name. 327 328#### Arguments 329 330- **name**: `String` — Name of view prop that you want to define a setter. 331- **setter**: `(view: ViewType, value: ValueType) -> ()` — Closure that is invoked when the view rerenders. 332 333This property can only be used within a [`ViewManager`](#viewmanager) closure. 334 335<CodeBlocksTable> 336 337```swift 338Prop("background") { (view: UIView, color: UIColor) in 339 view.backgroundColor = color 340} 341``` 342 343```kotlin 344Prop("background") { view: View, @ColorInt color: Int -> 345 view.setBackgroundColor(color) 346} 347``` 348 349</CodeBlocksTable> 350 351> **Note** Props of function type (callbacks) are not supported yet. 352 353</APIBox> 354<APIBox header="OnCreate"> 355 356Defines module's lifecycle listener that is called right after module initialization. If you need to set up something when the module gets initialized, use this instead of module's class initializer. 357 358</APIBox> 359<APIBox header="OnDestroy"> 360 361Defines module's lifecycle listener that is called when the module is about to be deallocated. Use it instead of module's class destructor. 362 363</APIBox> 364<APIBox header="OnStartObserving"> 365 366Defines the function that is invoked when the first event listener is added. 367 368</APIBox> 369<APIBox header="OnStopObserving"> 370 371Defines the function that is invoked when all event listeners are removed. 372 373</APIBox> 374<APIBox header="OnAppContextDestroys"> 375 376Defines module's lifecycle listener that is called when the app context owning the module is about to be deallocated. 377 378</APIBox> 379<APIBox header="OnAppEntersForeground" platforms={["ios"]}> 380 381Defines the listener that is called when the app is about to enter the foreground mode. 382 383> **Note** This function is not available on Android — you may want to use [`OnActivityEntersForeground`](#onactivityentersforeground) instead. 384 385</APIBox> 386<APIBox header="OnAppEntersBackground" platforms={["ios"]}> 387 388Defines the listener that is called when the app enters the background mode. 389 390> **Note** This function is not available on Android — you may want to use [`OnActivityEntersBackground`](#onactivityentersbackground) instead. 391 392</APIBox> 393<APIBox header="OnAppBecomesActive" platforms={["ios"]}> 394 395Defines the listener that is called when the app becomes active again (after `OnAppEntersForeground`). 396 397> **Note** This function is not available on Android — you may want to use [`OnActivityEntersForeground`](#onactivityentersforeground) instead. 398 399</APIBox> 400<APIBox header="OnActivityEntersForeground" platforms={["android"]}> 401 402Defines the activity lifecycle listener that is called right after the activity is resumed. 403 404> **Note** This function is not available on iOS — you may want to use [`OnAppEntersForeground`](#onappentersforeground) instead. 405 406</APIBox> 407<APIBox header="OnActivityEntersBackground" platforms={["android"]}> 408 409Defines the activity lifecycle listener that is called right after the activity is paused. 410 411> **Note** This function is not available on iOS — you may want to use [`OnAppEntersBackground`](#onappentersbackground) instead. 412 413</APIBox> 414<APIBox header="OnActivityDestroys" platforms={["android"]}> 415 416Defines the activity lifecycle listener that is called when the activity owning the JavaScript context is about to be destroyed. 417 418> **Note** This function is not available on iOS — you may want to use [`OnAppEntersBackground`](#onappentersbackground) instead. 419 420</APIBox> 421 422## Argument Types 423 424Fundamentally, only primitive and serializable data can be passed back and forth between the runtimes. However, usually native modules need to receive custom data structures — more sophisticated than just the dictionary/map where the values are of unknown (`Any`) type and so each value has to be validated and casted on its own. The Expo Modules API provides protocols to make it more convenient to work with data objects, to provide automatic validation, and finally, to ensure native type-safety on each object member. 425 426<APIBox header="Primitives"> 427 428All functions and view prop setters accept all common primitive types in Swift and Kotlin as the arguments. This includes arrays, dictionaries/maps and optionals of these primitive types. 429 430| Language | Supported primitive types | 431| -------- | ------------------------------------------------------------------------------------------------------------------------------ | 432| Swift | `Bool`, `Int`, `Int8`, `Int16`, `Int32`, `Int64`, `UInt`, `UInt8`, `UInt16`, `UInt32`, `UInt64`, `Float32`, `Double`, `String` | 433| Kotlin | `Boolean`, `Int`, `UInt`, `Float`, `Double`, `String`, `Pair` | 434 435</APIBox> 436<APIBox header="Convertibles"> 437 438_Convertibles_ are native types that can be initialized from certain specific kinds of data received from JavaScript. Such types are allowed to be used as an argument type in `Function`'s body. For example, when the `CGPoint` type is used as a function argument type, its instance can be created from an array of two numbers `(x, y)` or a JavaScript object with numeric `x` and `y` properties. 439 440--- 441 442Some common iOS types from `CoreGraphics` and `UIKit` system frameworks are already made convertible. 443 444| Native iOS Type | TypeScript | 445| ----------------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | 446| `URL` | `string` with a URL. When scheme is not provided, it's assumed to be a file URL. | 447| `CGFloat` | `number` | 448| `CGPoint` | `{ x: number, y: number }` or `number[]` with _x_ and _y_ coords | 449| `CGSize` | `{ width: number, height: number }` or `number[]` with _width_ and _height_ | 450| `CGVector` | `{ dx: number, dy: number }` or `number[]` with _dx_ and _dy_ vector differentials | 451| `CGRect` | `{ x: number, y: number, width: number, height: number }` or `number[]` with _x_, _y_, _width_ and _height_ values | 452| `CGColor`<br/>`UIColor` | Color hex strings (`#RRGGBB`, `#RRGGBBAA`, `#RGB`, `#RGBA`), named colors following the [CSS3/SVG specification](https://www.w3.org/TR/css-color-3/#svg-color) or `"transparent"` | 453 454--- 455 456Similarly, some common Android types from packages like `java.io`, `java.net`, or `android.graphics` are also made convertible. 457 458| Native Android Type | TypeScript | 459| ----------------------------------------------------------------------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | 460| `java.net.URL` | `string` with a URL. Note that the scheme has to be provided | 461| `android.net.Uri`<br/>`java.net.URI` | `string` with a URI. Note that the scheme has to be provided | 462| `java.io.File`<br/>`java.nio.file.Path` (is only available on Android API 26) | `string` with a path to the file | 463| `android.graphics.Color` | Color hex strings (`#RRGGBB`, `#RRGGBBAA`, `#RGB`, `#RGBA`), named colors following the [CSS3/SVG specification](https://www.w3.org/TR/css-color-3/#svg-color) or `"transparent"` | 464| `kotlin.Pair<A, B>` | Array with two values, where the first one is of type _A_ and the second is of type _B_ | 465 466</APIBox> 467<APIBox header="Records"> 468 469_Record_ is a convertible type and an equivalent of the dictionary (Swift) or map (Kotlin), but represented as a struct where each field can have its own type and provide a default value. 470It is a better way to represent a JavaScript object with the native type-safety. 471 472<CodeBlocksTable> 473 474```swift 475struct FileReadOptions: Record { 476 @Field 477 var encoding: String = "utf8" 478 479 @Field 480 var position: Int = 0 481 482 @Field 483 var length: Int? 484} 485 486// Now this record can be used as an argument of the functions or the view prop setters. 487Function("readFile") { (path: String, options: FileReadOptions) -> String in 488 // Read the file using given `options` 489} 490``` 491 492```kotlin 493class FileReadOptions : Record { 494 @Field 495 val encoding: String = "utf8" 496 497 @Field 498 val position: Int = 0 499 500 @Field 501 val length: Int? 502} 503 504// Now this record can be used as an argument of the functions or the view prop setters. 505Function("readFile") { path: String, options: FileReadOptions -> 506 // Read the file using given `options` 507} 508``` 509 510</CodeBlocksTable> 511</APIBox> 512<APIBox header="Enums"> 513 514With enums we can go even further with the above example (with `FileReadOptions` record) and limit supported encodings to `"utf8"` and `"base64"`. To use an enum as an argument or record field, it must represent a primitive value (e.g. `String`, `Int`) and conform to `Enumerable`. 515 516<CodeBlocksTable> 517 518```swift 519enum FileEncoding: String, Enumerable { 520 case utf8 521 case base64 522} 523 524struct FileReadOptions: Record { 525 @Field 526 var encoding: FileEncoding = .utf8 527 // ... 528} 529``` 530 531```kotlin 532// Note: the constructor must have an argument called value. 533enum class FileEncoding(val value: String) : Enumerable { 534 utf8("utf8"), 535 base64("base64") 536} 537 538class FileReadOptions : Record { 539 @Field 540 val encoding: FileEncoding = FileEncoding.utf8 541 // ... 542} 543``` 544 545</CodeBlocksTable> 546</APIBox> 547<APIBox header="Eithers"> 548 549There are some use cases where you want to pass various types for a single function argument. This is where Either types might come in handy. 550They act as a container for a value of one of a couple of types. 551 552<CodeBlocksTable> 553 554```swift 555Function("foo") { (bar: Either<String, Int>) in 556 if let bar: String = bar.get() { 557 // `bar` is a String 558 } 559 if let bar: Int = bar.get() { 560 // `bar` is an Int 561 } 562} 563``` 564 565```kotlin 566Function("foo") { bar: Either<String, Int> -> 567 bar.get(String::class).let { 568 // `it` is a String 569 } 570 bar.get(Int::class).let { 571 // `it` is an Int 572 } 573} 574``` 575 576</CodeBlocksTable> 577 578The implementation for three Either types is currently provided out of the box, allowing you to use up to four different subtypes. 579 580- `Either<FirstType, SecondType>` — A container for one of two types. 581- `EitherOfThree<FirstType, SecondType, ThirdType>` — A container for one of three types. 582- `EitherOfFour<FirstType, SecondType, ThirdType, FourthType>` — A container for one of four types. 583 584> Either types are available as of SDK 47. 585 586</APIBox> 587 588## Native Classes 589 590<APIBox header="Module"> 591 592A base class for a native module. 593 594#### Properties 595 596<APIMethod 597 name="appContext" 598 comment="Provides access to the [`AppContext`](#appcontext)." 599 returnTypeName="AppContext" 600 isProperty={true} 601 isReturnTypeReference={true} 602/> 603 604#### Methods 605 606<APIMethod 607 name="sendEvent" 608 comment="Sends an event with a given name and a payload to JavaScript. See [`Sending events`](#sending-events)" 609 returnTypeName="void" 610 parameters={[ 611 { 612 name: 'eventName', 613 comment: 'The name of the JavaScript event', 614 typeName: 'string', 615 }, 616 { 617 name: 'payload', 618 comment: 'The event payload', 619 typeName: 'Android: Map<String, Any?> | Bundle\niOS: [String: Any?]', 620 }, 621 ]} 622/> 623 624</APIBox> 625 626<APIBox header="AppContext"> 627 628The app context is an interface to a single Expo app. 629 630#### Properties 631 632<APIMethod 633 name="constants" 634 comment="Provides access to app's constants from legacy module registry." 635 returnTypeName="Android: ConstantsInterface? iOS: EXConstantsInterface?" 636 isProperty={true} 637/> 638 639<APIMethod 640 name="permissions" 641 comment="Provides access to the permissions manager from legacy module registry." 642 returnTypeName="Android: Permissions? iOS: EXPermissionsInterface?" 643 isProperty={true} 644/> 645 646<APIMethod 647 name="imageLoader" 648 comment="Provides access to the image loader from the legacy module registry." 649 returnTypeName="Android: ImageLoaderInterface? iOS: EXImageLoaderInterface?" 650 isProperty={true} 651/> 652 653<APIMethod 654 name="barcodeScanner" 655 comment="Provides access to the bar code scanner manager from the legacy module registry." 656 returnTypeName="ImageLoaderInterface?" 657 isProperty={true} 658 platforms={['Android']} 659/> 660 661<APIMethod 662 name="camera" 663 comment="Provides access to the camera view manager from the legacy module registry." 664 returnTypeName="CameraViewInterface?" 665 isProperty={true} 666 platforms={['Android']} 667/> 668 669<APIMethod 670 name="font" 671 comment="Provides access to the font manager from the legacy module registry." 672 returnTypeName="FontManagerInterface?" 673 isProperty={true} 674 platforms={['Android']} 675/> 676 677<APIMethod 678 name="sensor" 679 comment="Provides access to the sensor manager from the legacy module registry." 680 returnTypeName="SensorServiceInterface?" 681 isProperty={true} 682 platforms={['Android']} 683/> 684 685<APIMethod 686 name="taskManager" 687 comment="Provides access to the task manager from the legacy module registry." 688 returnTypeName="TaskManagerInterface?" 689 isProperty={true} 690 platforms={['Android']} 691/> 692 693<APIMethod 694 name="activityProvider" 695 comment="Provides access to the activity provider from the legacy module registry." 696 returnTypeName="ActivityProvider?" 697 isProperty={true} 698 platforms={['Android']} 699/> 700 701<APIMethod 702 name="reactContext" 703 comment="Provides access to the react application context." 704 returnTypeName="Context?" 705 isProperty={true} 706 platforms={['Android']} 707/> 708 709<APIMethod 710 name="hasActiveReactInstance" 711 comment="Checks if there is an not-null, alive react native instance." 712 returnTypeName="Boolean" 713 isProperty={true} 714 platforms={['Android']} 715/> 716 717<APIMethod 718 name="utilities" 719 comment="Provides access to the utilities from legacy module registry." 720 returnTypeName="EXUtilitiesInterface?" 721 isProperty={true} 722 platforms={['iOS']} 723/> 724 725</APIBox> 726 727<APIBox header="ExpoView"> 728 729A base class that should be used by all exported views. 730 731On iOS, `ExpoView` extends the `RCTView` which handles some styles (e.g. borders) and accessibility. 732 733#### Properties 734 735<APIMethod 736 name="appContext" 737 comment="Provides access to the [`AppContext`](#appcontext)." 738 returnTypeName="AppContext" 739 isProperty={true} 740 isReturnTypeReference={true} 741/> 742 743#### Extending `ExpoView` 744 745To export your view using the [`View`](#view) component, your custom class must inherit from the `ExpoView`. By doing that you will get access to the [`AppContext`](#appcontext) object. It's the only way of communicating with other modules and the JavaScript runtime. Also, you can't change constructor parameters, because provided view will be initialized by `expo-modules-core`. 746 747<CodeBlocksTable> 748 749```swift 750class LinearGradientView: ExpoView {} 751 752public class LinearGradientModule: Module { 753 public func definition() -> ModuleDefinition { 754 View(LinearGradientView.self) { 755 // ... 756 } 757 } 758} 759``` 760 761```kotlin 762class LinearGradientView( 763 context: Context, 764 appContext: AppContext, 765) : ExpoView(context, appContext) 766 767class LinearGradientModule : Module() { 768 override fun definition() = ModuleDefinition { 769 View(LinearGradientView::class) { 770 // ... 771 } 772 } 773} 774``` 775 776</CodeBlocksTable> 777 778</APIBox> 779 780## Guides 781 782<APIBox header="Sending events"> 783 784While JavaScript/TypeScript to Native communication is mostly covered by native functions, you might also want to let the JavaScript/TypeScript code know about certain system events, for example, when the clipboard content changes. 785 786To do this, in the module definition, you need to provide the event names that the module can send using the [Events](#events) definition component. After that, you can use the `sendEvent(eventName, payload)` function on the module instance to send the actual event with some payload. For example, a minimal clipboard implementation that sends native events may look like this: 787 788<CodeBlocksTable> 789 790```swift 791let CLIPBOARD_CHANGED_EVENT_NAME = "onClipboardChanged" 792 793public class ClipboardModule: Module { 794 public func definition() -> ModuleDefinition { 795 Events(CLIPBOARD_CHANGED_EVENT_NAME) 796 797 OnStartObserving { 798 NotificationCenter.default.addObserver( 799 self, 800 selector: #selector(self.clipboardChangedListener), 801 name: UIPasteboard.changedNotification, 802 object: nil 803 ) 804 } 805 806 OnStopObserving { 807 NotificationCenter.default.removeObserver( 808 self, 809 name: UIPasteboard.changedNotification, 810 object: nil 811 ) 812 } 813 } 814 815 @objc 816 private func clipboardChangedListener() { 817 sendEvent(CLIPBOARD_CHANGED_EVENT_NAME, [ 818 "contentTypes": availableContentTypes() 819 ]) 820 } 821} 822``` 823 824```kotlin 825const val CLIPBOARD_CHANGED_EVENT_NAME = "onClipboardChanged" 826 827class ClipboardModule : Module() { 828 override fun definition() = ModuleDefinition { 829 Events(CLIPBOARD_CHANGED_EVENT_NAME) 830 831 OnStartObserving { 832 clipboardManager?.addPrimaryClipChangedListener(listener) 833 } 834 835 OnStopObserving { 836 clipboardManager?.removePrimaryClipChangedListener(listener) 837 } 838 } 839 840 private val clipboardManager: ClipboardManager? 841 get() = appContext.reactContext?.getSystemService(Context.CLIPBOARD_SERVICE) as? ClipboardManager 842 843 private val listener = ClipboardManager.OnPrimaryClipChangedListener { 844 clipboardManager?.primaryClipDescription?.let { clip -> 845 [email protected]( 846 CLIPBOARD_CHANGED_EVENT_NAME, 847 bundleOf( 848 "contentTypes" to availableContentTypes(clip) 849 ) 850 ) 851 } 852 } 853} 854``` 855 856</CodeBlocksTable> 857 858To subscribe to these events in JavaScript/TypeScript, you need to wrap the native module with `EventEmitter` class as shown: 859 860```ts TypeScript 861import { requireNativeModule, EventEmitter, Subscription } from 'expo-modules-core'; 862 863const ClipboardModule = requireNativeModule('Clipboard'); 864const emitter = new EventEmitter(ClipboardModule); 865 866export function addClipboardListener(listener: (event) => void): Subscription { 867 return emitter.addListener('onClipboardChanged', listener); 868} 869``` 870 871</APIBox> 872 873<APIBox header="View callbacks"> 874 875Some events are connected to a certain view. For example, the touch event should be sent only to the underlying JavaScript view which was pressed. In that case, you can't use `sendEvent` described in [`Sending events`](#sending-events). The `expo-modules-core` introduces a view callbacks mechanism to handle view-bound events. 876 877To use it, in the view definition, you need to provide the event names that the view can send using the [Events](#events) definition component. After that, you need to declare a property of type `EventDispatcher` in your view class. The name of the declared property has to be the same as the name exported in the `Events` component. Later, you can call it as a function and pass a payload of type `[String: Any?]` on iOS and `Map<String, Any?>` on Android. 878 879> **Note**: On Android, it's possible to specify the payload type. In case of types that don't convert into objects, the payload will be encapsulated and stored under the `payload` key: `{payload: <provided value>}`. 880 881<CodeBlocksTable> 882 883```swift 884class CameraViewModule: Module { 885 public func definition() -> ModuleDefinition { 886 View(CameraView.self) { 887 Events( 888 "onCameraReady" 889 ) 890 891 // ... 892 } 893 } 894} 895 896class CameraView: ExpoView { 897 let onCameraReady = EventDispatcher() 898 899 func callOnCameraReady() { 900 onCameraReady([ 901 "message": "Camera was mounted" 902 ]); 903 } 904} 905``` 906 907```kotlin 908class CameraViewModule : Module() { 909 override fun definition() = ModuleDefinition { 910 View(ExpoCameraView::class) { 911 Events( 912 "onCameraReady" 913 ) 914 915 // ... 916 } 917 } 918} 919 920class CameraView( 921 context: Context, 922 appContext: AppContext 923) : ExpoView(context, appContext) { 924 val onCameraReady by EventDispatcher() 925 926 fun callOnCameraReady() { 927 onCameraReady(mapOf( 928 "message" to "Camera was mounted" 929 )); 930 } 931} 932``` 933 934</CodeBlocksTable> 935 936To subscribe to these events in JavaScript/TypeScript, you need to pass a function to the native view as shown: 937 938```ts TypeScript 939import { requireNativeViewManager } from 'expo-modules-core'; 940 941const CameraView = requireNativeViewManager('CameraView'); 942 943export default function MainView() { 944 const onCameraReady = event => { 945 console.log(event.nativeEvent); 946 }; 947 948 return <CameraView onCameraReady={onCameraReady} />; 949} 950``` 951 952Provided payload is available under the `nativeEvent` key. 953 954</APIBox> 955 956## Examples 957 958<CodeBlocksTable> 959 960```swift 961public class MyModule: Module { 962 public func definition() -> ModuleDefinition { 963 Name("MyFirstExpoModule") 964 965 Function("hello") { (name: String) in 966 return "Hello \(name)!" 967 } 968 } 969} 970``` 971 972```kotlin 973class MyModule : Module() { 974 override fun definition() = ModuleDefinition { 975 Name("MyFirstExpoModule") 976 977 Function("hello") { name: String -> 978 return "Hello $name!" 979 } 980 } 981} 982``` 983 984</CodeBlocksTable> 985 986For more examples from real modules, you can refer to Expo modules that already use this API on GitHub: 987 988- `expo-battery` ([Swift](https://github.com/expo/expo/tree/main/packages/expo-battery/ios)) 989- `expo-cellular` ([Swift](https://github.com/expo/expo/tree/main/packages/expo-cellular/ios), [Kotlin](https://github.com/expo/expo/tree/main/packages/expo-cellular/android/src/main/java/expo/modules/cellular)) 990- `expo-clipboard` ([Swift](https://github.com/expo/expo/tree/main/packages/expo-clipboard/ios), [Kotlin](https://github.com/expo/expo/tree/main/packages/expo-clipboard/android/src/main/java/expo/modules/clipboard)) 991- `expo-crypto` ([Swift](https://github.com/expo/expo/tree/main/packages/expo-crypto/ios), [Kotlin](https://github.com/expo/expo/tree/main/packages/expo-crypto/android/src/main/java/expo/modules/crypto)) 992- `expo-device` ([Swift](https://github.com/expo/expo/tree/main/packages/expo-device/ios)) 993- `expo-haptics` ([Swift](https://github.com/expo/expo/tree/main/packages/expo-haptics/ios)) 994- `expo-image-manipulator` ([Swift](https://github.com/expo/expo/tree/main/packages/expo-image-manipulator/ios)) 995- `expo-image-picker` ([Swift](https://github.com/expo/expo/tree/main/packages/expo-image-picker/ios), [Kotlin](https://github.com/expo/expo/tree/main/packages/expo-image-picker/android/src/main/java/expo/modules/imagepicker)) 996- `expo-linear-gradient` ([Swift](https://github.com/expo/expo/tree/main/packages/expo-linear-gradient/ios), [Kotlin](https://github.com/expo/expo/tree/main/packages/expo-linear-gradient/android/src/main/java/expo/modules/lineargradient)) 997- `expo-localization` ([Swift](https://github.com/expo/expo/tree/main/packages/expo-localization/ios), [Kotlin](https://github.com/expo/expo/tree/main/packages/expo-localization/android/src/main/java/expo/modules/localization)) 998- `expo-store-review` ([Swift](https://github.com/expo/expo/tree/main/packages/expo-store-review/ios)) 999- `expo-system-ui` ([Swift](https://github.com/expo/expo/tree/main/packages/expo-system-ui/ios/ExpoSystemUI)) 1000- `expo-video-thumbnails` ([Swift](https://github.com/expo/expo/tree/main/packages/expo-video-thumbnails/ios)) 1001- `expo-web-browser` ([Swift](https://github.com/expo/expo/blob/main/packages/expo-web-browser/ios), [Kotlin](https://github.com/expo/expo/blob/main/packages/expo-web-browser/android/src/main/java/expo/modules/webbrowser)) 1002