134 lines
4.7 KiB
JavaScript
134 lines
4.7 KiB
JavaScript
/**
|
|
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*
|
|
* @flow
|
|
*/
|
|
|
|
import type {AnyNativeEvent} from './legacy-events/PluginModuleType';
|
|
import type {Fiber} from 'react-reconciler/src/ReactInternalTypes';
|
|
import type {LegacyPluginModule} from './legacy-events/PluginModuleType';
|
|
import type {ReactSyntheticEvent} from './legacy-events/ReactSyntheticEventType';
|
|
import type {
|
|
RNTopLevelEventType,
|
|
TopLevelType,
|
|
} from './legacy-events/TopLevelEventTypes';
|
|
|
|
import {registrationNameModules} from './legacy-events/EventPluginRegistry';
|
|
import {batchedUpdates} from './legacy-events/ReactGenericBatching';
|
|
import accumulateInto from './legacy-events/accumulateInto';
|
|
|
|
import {plugins} from './legacy-events/EventPluginRegistry';
|
|
import getListeners from './ReactNativeGetListeners';
|
|
import {runEventsInBatch} from './legacy-events/EventBatching';
|
|
|
|
import {RawEventEmitter} from 'react-native/Libraries/ReactPrivate/ReactNativePrivateInterface';
|
|
|
|
export {getListeners, registrationNameModules as registrationNames};
|
|
|
|
/**
|
|
* Allows registered plugins an opportunity to extract events from top-level
|
|
* native browser events.
|
|
*
|
|
* @return {*} An accumulation of synthetic events.
|
|
* @internal
|
|
*/
|
|
function extractPluginEvents(
|
|
topLevelType: TopLevelType,
|
|
targetInst: null | Fiber,
|
|
nativeEvent: AnyNativeEvent,
|
|
nativeEventTarget: null | EventTarget,
|
|
): Array<ReactSyntheticEvent> | ReactSyntheticEvent | null {
|
|
let events = null;
|
|
const legacyPlugins = ((plugins: any): Array<LegacyPluginModule<Event>>);
|
|
for (let i = 0; i < legacyPlugins.length; i++) {
|
|
// Not every plugin in the ordering may be loaded at runtime.
|
|
const possiblePlugin: LegacyPluginModule<AnyNativeEvent> = legacyPlugins[i];
|
|
if (possiblePlugin) {
|
|
const extractedEvents = possiblePlugin.extractEvents(
|
|
topLevelType,
|
|
targetInst,
|
|
nativeEvent,
|
|
nativeEventTarget,
|
|
);
|
|
if (extractedEvents) {
|
|
events = accumulateInto(events, extractedEvents);
|
|
}
|
|
}
|
|
}
|
|
return events;
|
|
}
|
|
|
|
function runExtractedPluginEventsInBatch(
|
|
topLevelType: TopLevelType,
|
|
targetInst: null | Fiber,
|
|
nativeEvent: AnyNativeEvent,
|
|
nativeEventTarget: null | EventTarget,
|
|
) {
|
|
const events = extractPluginEvents(
|
|
topLevelType,
|
|
targetInst,
|
|
nativeEvent,
|
|
nativeEventTarget,
|
|
);
|
|
runEventsInBatch(events);
|
|
}
|
|
|
|
export function dispatchEvent(
|
|
target: null | Object,
|
|
topLevelType: RNTopLevelEventType,
|
|
nativeEvent: AnyNativeEvent,
|
|
) {
|
|
const targetFiber = (target: null | Fiber);
|
|
|
|
let eventTarget = null;
|
|
if (targetFiber != null) {
|
|
const stateNode = targetFiber.stateNode;
|
|
// Guard against Fiber being unmounted
|
|
if (stateNode != null) {
|
|
eventTarget = stateNode.canonical;
|
|
}
|
|
}
|
|
|
|
batchedUpdates(function() {
|
|
// Emit event to the RawEventEmitter. This is an unused-by-default EventEmitter
|
|
// that can be used to instrument event performance monitoring (primarily - could be useful
|
|
// for other things too).
|
|
//
|
|
// NOTE: this merely emits events into the EventEmitter below.
|
|
// If *you* do not add listeners to the `RawEventEmitter`,
|
|
// then all of these emitted events will just blackhole and are no-ops.
|
|
// It is available (although not officially supported... yet) if you want to collect
|
|
// perf data on event latency in your application, and could also be useful for debugging
|
|
// low-level events issues.
|
|
//
|
|
// If you do not have any event perf monitoring and are extremely concerned about event perf,
|
|
// it is safe to disable these "emit" statements; it will prevent checking the size of
|
|
// an empty array twice and prevent two no-ops. Practically the overhead is so low that
|
|
// we don't think it's worth thinking about in prod; your perf issues probably lie elsewhere.
|
|
//
|
|
// We emit two events here: one for listeners to this specific event,
|
|
// and one for the catchall listener '*', for any listeners that want
|
|
// to be notified for all events.
|
|
// Note that extracted events are *not* emitted,
|
|
// only events that have a 1:1 mapping with a native event, at least for now.
|
|
const event = {eventName: topLevelType, nativeEvent};
|
|
// $FlowFixMe[class-object-subtyping] found when upgrading Flow
|
|
RawEventEmitter.emit(topLevelType, event);
|
|
// $FlowFixMe[class-object-subtyping] found when upgrading Flow
|
|
RawEventEmitter.emit('*', event);
|
|
|
|
// Heritage plugin event system
|
|
runExtractedPluginEventsInBatch(
|
|
topLevelType,
|
|
targetFiber,
|
|
nativeEvent,
|
|
eventTarget,
|
|
);
|
|
});
|
|
// React Native doesn't use ReactControlledComponent but if it did, here's
|
|
// where it would do it.
|
|
}
|