UNPKG

react-native

Version:

A framework for building native apps using React

129 lines (113 loc) 4.04 kB
/** * Copyright (c) Meta Platforms, Inc. and affiliates. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. * * @flow strict-local * @format */ 'use strict'; import type { EventSubscription, IEventEmitter, } from '../vendor/emitter/EventEmitter'; import Platform from '../Utilities/Platform'; import RCTDeviceEventEmitter from './RCTDeviceEventEmitter'; import invariant from 'invariant'; interface NativeModule { addListener(eventType: string): void; removeListeners(count: number): void; } export type {EventSubscription}; // $FlowFixMe[unclear-type] unclear type of events type UnsafeObject = Object; /** * `NativeEventEmitter` is intended for use by Native Modules to emit events to * JavaScript listeners. If a `NativeModule` is supplied to the constructor, it * will be notified (via `addListener` and `removeListeners`) when the listener * count changes to manage "native memory". * * Currently, all native events are fired via a global `RCTDeviceEventEmitter`. * This means event names must be globally unique, and it means that call sites * can theoretically listen to `RCTDeviceEventEmitter` (although discouraged). */ export default class NativeEventEmitter< TEventToArgsMap: $ReadOnly< Record<string, $ReadOnlyArray<UnsafeObject>>, > = $ReadOnly<Record<string, $ReadOnlyArray<UnsafeObject>>>, > implements IEventEmitter<TEventToArgsMap> { _nativeModule: ?NativeModule; constructor(nativeModule: ?NativeModule) { if (Platform.OS === 'ios') { invariant( nativeModule != null, '`new NativeEventEmitter()` requires a non-null argument.', ); } const hasAddListener = // $FlowFixMe[method-unbinding] added when improving typing for this parameters !!nativeModule && typeof nativeModule.addListener === 'function'; const hasRemoveListeners = // $FlowFixMe[method-unbinding] added when improving typing for this parameters !!nativeModule && typeof nativeModule.removeListeners === 'function'; if (nativeModule && hasAddListener && hasRemoveListeners) { this._nativeModule = nativeModule; } else if (nativeModule != null) { if (!hasAddListener) { console.warn( '`new NativeEventEmitter()` was called with a non-null argument without the required `addListener` method.', ); } if (!hasRemoveListeners) { console.warn( '`new NativeEventEmitter()` was called with a non-null argument without the required `removeListeners` method.', ); } } } addListener<TEvent: $Keys<TEventToArgsMap>>( eventType: TEvent, listener: (...args: $ElementType<TEventToArgsMap, TEvent>) => mixed, context?: mixed, ): EventSubscription { this._nativeModule?.addListener(eventType); let subscription: ?EventSubscription = RCTDeviceEventEmitter.addListener( eventType, listener, context, ); return { remove: () => { if (subscription != null) { this._nativeModule?.removeListeners(1); // $FlowFixMe[incompatible-use] subscription.remove(); subscription = null; } }, }; } emit<TEvent: $Keys<TEventToArgsMap>>( eventType: TEvent, ...args: $ElementType<TEventToArgsMap, TEvent> ): void { // Generally, `RCTDeviceEventEmitter` is directly invoked. But this is // included for completeness. RCTDeviceEventEmitter.emit(eventType, ...args); } removeAllListeners<TEvent: $Keys<TEventToArgsMap>>( eventType?: ?TEvent, ): void { invariant( eventType != null, '`NativeEventEmitter.removeAllListener()` requires a non-null argument.', ); this._nativeModule?.removeListeners(this.listenerCount(eventType)); RCTDeviceEventEmitter.removeAllListeners(eventType); } listenerCount<TEvent: $Keys<TEventToArgsMap>>(eventType: TEvent): number { return RCTDeviceEventEmitter.listenerCount(eventType); } }