xref: /expo/docs/pages/modules/module-api.mdx (revision 3c8ea696)
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