mirror of
https://github.com/facebook/react.git
synced 2025-11-01 09:12:30 +00:00
05c283c3c3
* Implement addEventListener and removeEventListener on Fabric HostComponent * add files * re-add CustomEvent * fix flow * Need to get CustomEvent from an import since it won't exist on the global scope by default * yarn prettier-all * use a mangled name consistently to refer to imperatively registered event handlers * yarn prettier-all * fuzzy null check * fix capture phase event listener logic * early exit from getEventListeners more often * make some optimizations to getEventListeners and the bridge plugin * fix accumulateInto logic * fix accumulateInto * Simplifying getListeners at the expense of perf for the non-hot path * feedback * fix impl of getListeners to correctly remove function * pass all args in to event listeners
647 lines
18 KiB
JavaScript
647 lines
18 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 {ReactNodeList, OffscreenMode} from 'shared/ReactTypes';
|
|
import type {ElementRef} from 'react';
|
|
import type {
|
|
HostComponent,
|
|
MeasureInWindowOnSuccessCallback,
|
|
MeasureLayoutOnSuccessCallback,
|
|
MeasureOnSuccessCallback,
|
|
NativeMethods,
|
|
ViewConfig,
|
|
TouchedViewDataAtPoint,
|
|
} from './ReactNativeTypes';
|
|
|
|
import {mountSafeCallback_NOT_REALLY_SAFE} from './NativeMethodsMixinUtils';
|
|
import {create, diff} from './ReactNativeAttributePayload';
|
|
|
|
import {dispatchEvent} from './ReactFabricEventEmitter';
|
|
|
|
import {
|
|
DefaultEventPriority,
|
|
DiscreteEventPriority,
|
|
} from 'react-reconciler/src/ReactEventPriorities';
|
|
|
|
// Modules provided by RN:
|
|
import {
|
|
ReactNativeViewConfigRegistry,
|
|
TextInputState,
|
|
deepFreezeAndThrowOnMutationInDev,
|
|
} from 'react-native/Libraries/ReactPrivate/ReactNativePrivateInterface';
|
|
|
|
const {
|
|
createNode,
|
|
cloneNode,
|
|
cloneNodeWithNewChildren,
|
|
cloneNodeWithNewChildrenAndProps,
|
|
cloneNodeWithNewProps,
|
|
createChildSet: createChildNodeSet,
|
|
appendChild: appendChildNode,
|
|
appendChildToSet: appendChildNodeToSet,
|
|
completeRoot,
|
|
registerEventHandler,
|
|
measure: fabricMeasure,
|
|
measureInWindow: fabricMeasureInWindow,
|
|
measureLayout: fabricMeasureLayout,
|
|
unstable_DefaultEventPriority: FabricDefaultPriority,
|
|
unstable_DiscreteEventPriority: FabricDiscretePriority,
|
|
unstable_getCurrentEventPriority: fabricGetCurrentEventPriority,
|
|
} = nativeFabricUIManager;
|
|
|
|
const {get: getViewConfigForType} = ReactNativeViewConfigRegistry;
|
|
|
|
// Counter for uniquely identifying views.
|
|
// % 10 === 1 means it is a rootTag.
|
|
// % 2 === 0 means it is a Fabric tag.
|
|
// This means that they never overlap.
|
|
let nextReactTag = 2;
|
|
|
|
type Node = Object;
|
|
export type Type = string;
|
|
export type Props = Object;
|
|
export type Instance = {
|
|
node: Node,
|
|
canonical: ReactFabricHostComponent,
|
|
...
|
|
};
|
|
export type TextInstance = {node: Node, ...};
|
|
export type HydratableInstance = Instance | TextInstance;
|
|
export type PublicInstance = ReactFabricHostComponent;
|
|
export type Container = number;
|
|
export type ChildSet = Object;
|
|
export type HostContext = $ReadOnly<{|
|
|
isInAParentText: boolean,
|
|
|}>;
|
|
export type UpdatePayload = Object;
|
|
|
|
export type TimeoutHandle = TimeoutID;
|
|
export type NoTimeout = -1;
|
|
|
|
export type RendererInspectionConfig = $ReadOnly<{|
|
|
// Deprecated. Replaced with getInspectorDataForViewAtPoint.
|
|
getInspectorDataForViewTag?: (tag: number) => Object,
|
|
getInspectorDataForViewAtPoint?: (
|
|
inspectedView: Object,
|
|
locationX: number,
|
|
locationY: number,
|
|
callback: (viewData: TouchedViewDataAtPoint) => mixed,
|
|
) => void,
|
|
|}>;
|
|
|
|
// TODO?: find a better place for this type to live
|
|
export type EventListenerOptions = $ReadOnly<{|
|
|
capture?: boolean,
|
|
once?: boolean,
|
|
passive?: boolean,
|
|
signal: mixed, // not yet implemented
|
|
|}>;
|
|
export type EventListenerRemoveOptions = $ReadOnly<{|
|
|
capture?: boolean,
|
|
|}>;
|
|
|
|
// TODO?: this will be changed in the future to be w3c-compatible and allow "EventListener" objects as well as functions.
|
|
export type EventListener = Function;
|
|
|
|
type InternalEventListeners = {
|
|
[string]: {|
|
|
listener: EventListener,
|
|
options: EventListenerOptions,
|
|
invalidated: boolean,
|
|
|}[],
|
|
};
|
|
|
|
// TODO: Remove this conditional once all changes have propagated.
|
|
if (registerEventHandler) {
|
|
/**
|
|
* Register the event emitter with the native bridge
|
|
*/
|
|
registerEventHandler(dispatchEvent);
|
|
}
|
|
|
|
/**
|
|
* This is used for refs on host components.
|
|
*/
|
|
class ReactFabricHostComponent {
|
|
_nativeTag: number;
|
|
viewConfig: ViewConfig;
|
|
currentProps: Props;
|
|
_internalInstanceHandle: Object;
|
|
_eventListeners: ?InternalEventListeners;
|
|
|
|
constructor(
|
|
tag: number,
|
|
viewConfig: ViewConfig,
|
|
props: Props,
|
|
internalInstanceHandle: Object,
|
|
) {
|
|
this._nativeTag = tag;
|
|
this.viewConfig = viewConfig;
|
|
this.currentProps = props;
|
|
this._internalInstanceHandle = internalInstanceHandle;
|
|
}
|
|
|
|
blur() {
|
|
TextInputState.blurTextInput(this);
|
|
}
|
|
|
|
focus() {
|
|
TextInputState.focusTextInput(this);
|
|
}
|
|
|
|
measure(callback: MeasureOnSuccessCallback) {
|
|
const {stateNode} = this._internalInstanceHandle;
|
|
if (stateNode != null) {
|
|
fabricMeasure(
|
|
stateNode.node,
|
|
mountSafeCallback_NOT_REALLY_SAFE(this, callback),
|
|
);
|
|
}
|
|
}
|
|
|
|
measureInWindow(callback: MeasureInWindowOnSuccessCallback) {
|
|
const {stateNode} = this._internalInstanceHandle;
|
|
if (stateNode != null) {
|
|
fabricMeasureInWindow(
|
|
stateNode.node,
|
|
mountSafeCallback_NOT_REALLY_SAFE(this, callback),
|
|
);
|
|
}
|
|
}
|
|
|
|
measureLayout(
|
|
relativeToNativeNode: number | ElementRef<HostComponent<mixed>>,
|
|
onSuccess: MeasureLayoutOnSuccessCallback,
|
|
onFail?: () => void /* currently unused */,
|
|
) {
|
|
if (
|
|
typeof relativeToNativeNode === 'number' ||
|
|
!(relativeToNativeNode instanceof ReactFabricHostComponent)
|
|
) {
|
|
if (__DEV__) {
|
|
console.error(
|
|
'Warning: ref.measureLayout must be called with a ref to a native component.',
|
|
);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
const toStateNode = this._internalInstanceHandle.stateNode;
|
|
const fromStateNode =
|
|
relativeToNativeNode._internalInstanceHandle.stateNode;
|
|
|
|
if (toStateNode != null && fromStateNode != null) {
|
|
fabricMeasureLayout(
|
|
toStateNode.node,
|
|
fromStateNode.node,
|
|
mountSafeCallback_NOT_REALLY_SAFE(this, onFail),
|
|
mountSafeCallback_NOT_REALLY_SAFE(this, onSuccess),
|
|
);
|
|
}
|
|
}
|
|
|
|
setNativeProps(nativeProps: Object) {
|
|
if (__DEV__) {
|
|
console.error(
|
|
'Warning: setNativeProps is not currently supported in Fabric',
|
|
);
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
// This API (addEventListener, removeEventListener) attempts to adhere to the
|
|
// w3 Level2 Events spec as much as possible, treating HostComponent as a DOM node.
|
|
//
|
|
// Unless otherwise noted, these methods should "just work" and adhere to the W3 specs.
|
|
// If they deviate in a way that is not explicitly noted here, you've found a bug!
|
|
//
|
|
// See:
|
|
// * https://www.w3.org/TR/DOM-Level-2-Events/events.html
|
|
// * https://developer.mozilla.org/en-US/docs/Web/API/EventTarget/addEventListener
|
|
// * https://developer.mozilla.org/en-US/docs/Web/API/EventTarget/removeEventListener
|
|
//
|
|
// And notably, not implemented (yet?):
|
|
// * https://developer.mozilla.org/en-US/docs/Web/API/EventTarget/dispatchEvent
|
|
//
|
|
//
|
|
// Deviations from spec/TODOs:
|
|
// (1) listener must currently be a function, we do not support EventListener objects yet.
|
|
// (2) we do not support the `signal` option / AbortSignal yet
|
|
addEventListener_unstable(
|
|
eventType: string,
|
|
listener: EventListener,
|
|
options: EventListenerOptions | boolean,
|
|
) {
|
|
if (typeof eventType !== 'string') {
|
|
throw new Error('addEventListener_unstable eventType must be a string');
|
|
}
|
|
if (typeof listener !== 'function') {
|
|
throw new Error('addEventListener_unstable listener must be a function');
|
|
}
|
|
|
|
// The third argument is either boolean indicating "captures" or an object.
|
|
const optionsObj =
|
|
typeof options === 'object' && options !== null ? options : {};
|
|
const capture =
|
|
(typeof options === 'boolean' ? options : optionsObj.capture) || false;
|
|
const once = optionsObj.once || false;
|
|
const passive = optionsObj.passive || false;
|
|
const signal = null; // TODO: implement signal/AbortSignal
|
|
|
|
const eventListeners: InternalEventListeners = this._eventListeners || {};
|
|
if (this._eventListeners == null) {
|
|
this._eventListeners = eventListeners;
|
|
}
|
|
|
|
const namedEventListeners = eventListeners[eventType] || [];
|
|
if (eventListeners[eventType] == null) {
|
|
eventListeners[eventType] = namedEventListeners;
|
|
}
|
|
|
|
namedEventListeners.push({
|
|
listener: listener,
|
|
invalidated: false,
|
|
options: {
|
|
capture: capture,
|
|
once: once,
|
|
passive: passive,
|
|
signal: signal,
|
|
},
|
|
});
|
|
}
|
|
|
|
// See https://developer.mozilla.org/en-US/docs/Web/API/EventTarget/removeEventListener
|
|
removeEventListener_unstable(
|
|
eventType: string,
|
|
listener: EventListener,
|
|
options: EventListenerRemoveOptions | boolean,
|
|
) {
|
|
// eventType and listener must be referentially equal to be removed from the listeners
|
|
// data structure, but in "options" we only check the `capture` flag, according to spec.
|
|
// That means if you add the same function as a listener with capture set to true and false,
|
|
// you must also call removeEventListener twice with capture set to true/false.
|
|
const optionsObj =
|
|
typeof options === 'object' && options !== null ? options : {};
|
|
const capture =
|
|
(typeof options === 'boolean' ? options : optionsObj.capture) || false;
|
|
|
|
// If there are no event listeners or named event listeners, we can bail early - our
|
|
// job is already done.
|
|
const eventListeners = this._eventListeners;
|
|
if (!eventListeners) {
|
|
return;
|
|
}
|
|
const namedEventListeners = eventListeners[eventType];
|
|
if (!namedEventListeners) {
|
|
return;
|
|
}
|
|
|
|
// TODO: optimize this path to make remove cheaper
|
|
eventListeners[eventType] = namedEventListeners.filter(listenerObj => {
|
|
return !(
|
|
listenerObj.listener === listener &&
|
|
listenerObj.options.capture === capture
|
|
);
|
|
});
|
|
}
|
|
}
|
|
|
|
// eslint-disable-next-line no-unused-expressions
|
|
(ReactFabricHostComponent.prototype: $ReadOnly<{...NativeMethods, ...}>);
|
|
|
|
export * from 'react-reconciler/src/ReactFiberHostConfigWithNoMutation';
|
|
export * from 'react-reconciler/src/ReactFiberHostConfigWithNoHydration';
|
|
export * from 'react-reconciler/src/ReactFiberHostConfigWithNoScopes';
|
|
export * from 'react-reconciler/src/ReactFiberHostConfigWithNoTestSelectors';
|
|
export * from 'react-reconciler/src/ReactFiberHostConfigWithNoMicrotasks';
|
|
|
|
export function appendInitialChild(
|
|
parentInstance: Instance,
|
|
child: Instance | TextInstance,
|
|
): void {
|
|
appendChildNode(parentInstance.node, child.node);
|
|
}
|
|
|
|
export function createInstance(
|
|
type: string,
|
|
props: Props,
|
|
rootContainerInstance: Container,
|
|
hostContext: HostContext,
|
|
internalInstanceHandle: Object,
|
|
): Instance {
|
|
const tag = nextReactTag;
|
|
nextReactTag += 2;
|
|
|
|
const viewConfig = getViewConfigForType(type);
|
|
|
|
if (__DEV__) {
|
|
for (const key in viewConfig.validAttributes) {
|
|
if (props.hasOwnProperty(key)) {
|
|
deepFreezeAndThrowOnMutationInDev(props[key]);
|
|
}
|
|
}
|
|
}
|
|
|
|
const updatePayload = create(props, viewConfig.validAttributes);
|
|
|
|
const node = createNode(
|
|
tag, // reactTag
|
|
viewConfig.uiViewClassName, // viewName
|
|
rootContainerInstance, // rootTag
|
|
updatePayload, // props
|
|
internalInstanceHandle, // internalInstanceHandle
|
|
);
|
|
|
|
const component = new ReactFabricHostComponent(
|
|
tag,
|
|
viewConfig,
|
|
props,
|
|
internalInstanceHandle,
|
|
);
|
|
|
|
return {
|
|
node: node,
|
|
canonical: component,
|
|
};
|
|
}
|
|
|
|
export function createTextInstance(
|
|
text: string,
|
|
rootContainerInstance: Container,
|
|
hostContext: HostContext,
|
|
internalInstanceHandle: Object,
|
|
): TextInstance {
|
|
if (__DEV__) {
|
|
if (!hostContext.isInAParentText) {
|
|
console.error('Text strings must be rendered within a <Text> component.');
|
|
}
|
|
}
|
|
|
|
const tag = nextReactTag;
|
|
nextReactTag += 2;
|
|
|
|
const node = createNode(
|
|
tag, // reactTag
|
|
'RCTRawText', // viewName
|
|
rootContainerInstance, // rootTag
|
|
{text: text}, // props
|
|
internalInstanceHandle, // instance handle
|
|
);
|
|
|
|
return {
|
|
node: node,
|
|
};
|
|
}
|
|
|
|
export function finalizeInitialChildren(
|
|
parentInstance: Instance,
|
|
type: string,
|
|
props: Props,
|
|
rootContainerInstance: Container,
|
|
hostContext: HostContext,
|
|
): boolean {
|
|
return false;
|
|
}
|
|
|
|
export function getRootHostContext(
|
|
rootContainerInstance: Container,
|
|
): HostContext {
|
|
return {isInAParentText: false};
|
|
}
|
|
|
|
export function getChildHostContext(
|
|
parentHostContext: HostContext,
|
|
type: string,
|
|
rootContainerInstance: Container,
|
|
): HostContext {
|
|
const prevIsInAParentText = parentHostContext.isInAParentText;
|
|
const isInAParentText =
|
|
type === 'AndroidTextInput' || // Android
|
|
type === 'RCTMultilineTextInputView' || // iOS
|
|
type === 'RCTSinglelineTextInputView' || // iOS
|
|
type === 'RCTText' ||
|
|
type === 'RCTVirtualText';
|
|
|
|
// TODO: If this is an offscreen host container, we should reuse the
|
|
// parent context.
|
|
|
|
if (prevIsInAParentText !== isInAParentText) {
|
|
return {isInAParentText};
|
|
} else {
|
|
return parentHostContext;
|
|
}
|
|
}
|
|
|
|
export function getPublicInstance(instance: Instance): * {
|
|
return instance.canonical;
|
|
}
|
|
|
|
export function prepareForCommit(containerInfo: Container): null | Object {
|
|
// Noop
|
|
return null;
|
|
}
|
|
|
|
export function prepareUpdate(
|
|
instance: Instance,
|
|
type: string,
|
|
oldProps: Props,
|
|
newProps: Props,
|
|
rootContainerInstance: Container,
|
|
hostContext: HostContext,
|
|
): null | Object {
|
|
const viewConfig = instance.canonical.viewConfig;
|
|
const updatePayload = diff(oldProps, newProps, viewConfig.validAttributes);
|
|
// TODO: If the event handlers have changed, we need to update the current props
|
|
// in the commit phase but there is no host config hook to do it yet.
|
|
// So instead we hack it by updating it in the render phase.
|
|
instance.canonical.currentProps = newProps;
|
|
return updatePayload;
|
|
}
|
|
|
|
export function resetAfterCommit(containerInfo: Container): void {
|
|
// Noop
|
|
}
|
|
|
|
export function shouldSetTextContent(type: string, props: Props): boolean {
|
|
// TODO (bvaughn) Revisit this decision.
|
|
// Always returning false simplifies the createInstance() implementation,
|
|
// But creates an additional child Fiber for raw text children.
|
|
// No additional native views are created though.
|
|
// It's not clear to me which is better so I'm deferring for now.
|
|
// More context @ github.com/facebook/react/pull/8560#discussion_r92111303
|
|
return false;
|
|
}
|
|
|
|
export function getCurrentEventPriority(): * {
|
|
const currentEventPriority = fabricGetCurrentEventPriority
|
|
? fabricGetCurrentEventPriority()
|
|
: null;
|
|
|
|
if (currentEventPriority != null) {
|
|
switch (currentEventPriority) {
|
|
case FabricDiscretePriority:
|
|
return DiscreteEventPriority;
|
|
case FabricDefaultPriority:
|
|
default:
|
|
return DefaultEventPriority;
|
|
}
|
|
}
|
|
|
|
return DefaultEventPriority;
|
|
}
|
|
|
|
// The Fabric renderer is secondary to the existing React Native renderer.
|
|
export const isPrimaryRenderer = false;
|
|
|
|
// The Fabric renderer shouldn't trigger missing act() warnings
|
|
export const warnsIfNotActing = false;
|
|
|
|
export const scheduleTimeout = setTimeout;
|
|
export const cancelTimeout = clearTimeout;
|
|
export const noTimeout = -1;
|
|
|
|
// -------------------
|
|
// Persistence
|
|
// -------------------
|
|
|
|
export const supportsPersistence = true;
|
|
|
|
export function cloneInstance(
|
|
instance: Instance,
|
|
updatePayload: null | Object,
|
|
type: string,
|
|
oldProps: Props,
|
|
newProps: Props,
|
|
internalInstanceHandle: Object,
|
|
keepChildren: boolean,
|
|
recyclableInstance: null | Instance,
|
|
): Instance {
|
|
const node = instance.node;
|
|
let clone;
|
|
if (keepChildren) {
|
|
if (updatePayload !== null) {
|
|
clone = cloneNodeWithNewProps(node, updatePayload);
|
|
} else {
|
|
clone = cloneNode(node);
|
|
}
|
|
} else {
|
|
if (updatePayload !== null) {
|
|
clone = cloneNodeWithNewChildrenAndProps(node, updatePayload);
|
|
} else {
|
|
clone = cloneNodeWithNewChildren(node);
|
|
}
|
|
}
|
|
return {
|
|
node: clone,
|
|
canonical: instance.canonical,
|
|
};
|
|
}
|
|
|
|
// TODO: These two methods should be replaced with `createOffscreenInstance` and
|
|
// `cloneOffscreenInstance`. I did it this way for now because the offscreen
|
|
// instance is stored on an extra HostComponent fiber instead of the
|
|
// OffscreenComponent fiber, and I didn't want to add an extra check to the
|
|
// generic HostComponent path. Instead we should use the OffscreenComponent
|
|
// fiber, but currently Fabric expects a 1:1 correspondence between Fabric
|
|
// instances and host fibers, so I'm leaving this optimization for later once
|
|
// we can confirm this won't break any downstream expectations.
|
|
export function getOffscreenContainerType(): string {
|
|
return 'RCTView';
|
|
}
|
|
|
|
export function getOffscreenContainerProps(
|
|
mode: OffscreenMode,
|
|
children: ReactNodeList,
|
|
): Props {
|
|
if (mode === 'hidden') {
|
|
return {
|
|
children,
|
|
style: {display: 'none'},
|
|
};
|
|
} else {
|
|
return {
|
|
children,
|
|
style: {
|
|
flex: 1,
|
|
},
|
|
};
|
|
}
|
|
}
|
|
|
|
export function cloneHiddenInstance(
|
|
instance: Instance,
|
|
type: string,
|
|
props: Props,
|
|
internalInstanceHandle: Object,
|
|
): Instance {
|
|
const viewConfig = instance.canonical.viewConfig;
|
|
const node = instance.node;
|
|
const updatePayload = create(
|
|
{style: {display: 'none'}},
|
|
viewConfig.validAttributes,
|
|
);
|
|
return {
|
|
node: cloneNodeWithNewProps(node, updatePayload),
|
|
canonical: instance.canonical,
|
|
};
|
|
}
|
|
|
|
export function cloneHiddenTextInstance(
|
|
instance: Instance,
|
|
text: string,
|
|
internalInstanceHandle: Object,
|
|
): TextInstance {
|
|
throw new Error('Not yet implemented.');
|
|
}
|
|
|
|
export function createContainerChildSet(container: Container): ChildSet {
|
|
return createChildNodeSet(container);
|
|
}
|
|
|
|
export function appendChildToContainerChildSet(
|
|
childSet: ChildSet,
|
|
child: Instance | TextInstance,
|
|
): void {
|
|
appendChildNodeToSet(childSet, child.node);
|
|
}
|
|
|
|
export function finalizeContainerChildren(
|
|
container: Container,
|
|
newChildren: ChildSet,
|
|
): void {
|
|
completeRoot(container, newChildren);
|
|
}
|
|
|
|
export function replaceContainerChildren(
|
|
container: Container,
|
|
newChildren: ChildSet,
|
|
): void {}
|
|
|
|
export function getInstanceFromNode(node: any) {
|
|
throw new Error('Not yet implemented.');
|
|
}
|
|
|
|
export function beforeActiveInstanceBlur(internalInstanceHandle: Object) {
|
|
// noop
|
|
}
|
|
|
|
export function afterActiveInstanceBlur() {
|
|
// noop
|
|
}
|
|
|
|
export function preparePortalMount(portalInstance: Instance): void {
|
|
// noop
|
|
}
|
|
|
|
export function detachDeletedInstance(node: Instance): void {
|
|
// noop
|
|
}
|