mirror of
https://github.com/facebook/react.git
synced 2025-11-01 09:12:30 +00:00
ba6fea1bf5
Summary: This makes a few changes to React Core, most notably `ReactEventEmitter` and `ReactEventTopLevelCallback`. - Changed `ReactEventEmitter` to use `EventListener` (instead of `NormalizedEventListener`). - Deleted `NormalizedEventListener` (which was previously broken). - Created `getEventTarget` which is used to get a normalized `target` from a native event. - Changed `ReactEventTopLevelCallback` to use `getEventTarget`. - Renamed `abstractEventType` to `reactEventType` in `AbstractEvent`. - Reanmed `abstractTargetID` to `reactTargetID` in `AbstractEvent`. - Removed `originatingTopLevelEventType` from `AbstractEvent` (unused and violates encapsulation). - Removed `nativeEvent.target === window` check when refreshing authoritative scroll values (unnecessary). This actually fixes React because `NormalizedEventListener` does not currently do what it promises to do (which is normalizing `target` on the native event). The `target` event is read-only on native events. This also revises documentation and adds `@typechecks` to a few modules. NOTE: Most importantly, this sets the stage for replacing `AbstractEvent` with `ReactEvent` and subclasses, piecemeal.
182 lines
5.8 KiB
JavaScript
182 lines
5.8 KiB
JavaScript
/**
|
|
* Copyright 2013 Facebook, Inc.
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*
|
|
* @providesModule EventPropagators
|
|
*/
|
|
|
|
"use strict";
|
|
|
|
var CallbackRegistry = require('CallbackRegistry');
|
|
var EventConstants = require('EventConstants');
|
|
|
|
var accumulate = require('accumulate');
|
|
var forEachAccumulated = require('forEachAccumulated');
|
|
var getListener = CallbackRegistry.getListener;
|
|
var PropagationPhases = EventConstants.PropagationPhases;
|
|
|
|
/**
|
|
* Injected dependencies:
|
|
*/
|
|
|
|
/**
|
|
* - `InstanceHandle`: [required] Module that performs logical traversals of DOM
|
|
* hierarchy given ids of the logical DOM elements involved.
|
|
*/
|
|
var injection = {
|
|
InstanceHandle: null,
|
|
injectInstanceHandle: function(InjectedInstanceHandle) {
|
|
injection.InstanceHandle = InjectedInstanceHandle;
|
|
if (__DEV__) {
|
|
injection.validate();
|
|
}
|
|
},
|
|
validate: function() {
|
|
var invalid = !injection.InstanceHandle||
|
|
!injection.InstanceHandle.traverseTwoPhase ||
|
|
!injection.InstanceHandle.traverseEnterLeave;
|
|
if (invalid) {
|
|
throw new Error('InstanceHandle not injected before use!');
|
|
}
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Some event types have a notion of different registration names for different
|
|
* "phases" of propagation. This finds listeners by a given phase.
|
|
*/
|
|
function listenerAtPhase(id, abstractEvent, propagationPhase) {
|
|
var registrationName =
|
|
abstractEvent.reactEventType.phasedRegistrationNames[propagationPhase];
|
|
return getListener(id, registrationName);
|
|
}
|
|
|
|
/**
|
|
* Tags an `AbstractEvent` with dispatched listeners. Creating this function
|
|
* here, allows us to not have to bind or create functions for each event.
|
|
* Mutating the event's members allows us to not have to create a wrapping
|
|
* "dispatch" object that pairs the event with the listener.
|
|
*/
|
|
function accumulateDirectionalDispatches(domID, upwards, abstractEvent) {
|
|
if (__DEV__) {
|
|
if (!domID) {
|
|
throw new Error('Dispatching id must not be null');
|
|
}
|
|
injection.validate();
|
|
}
|
|
var phase = upwards ? PropagationPhases.bubbled : PropagationPhases.captured;
|
|
var listener = listenerAtPhase(domID, abstractEvent, phase);
|
|
if (listener) {
|
|
abstractEvent._dispatchListeners =
|
|
accumulate(abstractEvent._dispatchListeners, listener);
|
|
abstractEvent._dispatchIDs = accumulate(abstractEvent._dispatchIDs, domID);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Collect dispatches (must be entirely collected before dispatching - see unit
|
|
* tests). Lazily allocate the array to conserve memory. We must loop through
|
|
* each event and perform the traversal for each one. We can not perform a
|
|
* single traversal for the entire collection of events because each event may
|
|
* have a different target.
|
|
*/
|
|
function accumulateTwoPhaseDispatchesSingle(abstractEvent) {
|
|
if (abstractEvent && abstractEvent.reactEventType.phasedRegistrationNames) {
|
|
injection.InstanceHandle.traverseTwoPhase(
|
|
abstractEvent.reactTargetID,
|
|
accumulateDirectionalDispatches,
|
|
abstractEvent
|
|
);
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Accumulates without regard to direction, does not look for phased
|
|
* registration names. Same as `accumulateDirectDispatchesSingle` but without
|
|
* requiring that the `reactTargetID` be the same as the dispatched ID.
|
|
*/
|
|
function accumulateDispatches(id, ignoredDirection, abstractEvent) {
|
|
if (abstractEvent && abstractEvent.reactEventType.registrationName) {
|
|
var registrationName = abstractEvent.reactEventType.registrationName;
|
|
var listener = getListener(id, registrationName);
|
|
if (listener) {
|
|
abstractEvent._dispatchListeners =
|
|
accumulate(abstractEvent._dispatchListeners, listener);
|
|
abstractEvent._dispatchIDs = accumulate(abstractEvent._dispatchIDs, id);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Accumulates dispatches on an `AbstractEvent`, but only for the
|
|
* `reactTargetID`.
|
|
* @param {AbstractEvent} abstractEvent
|
|
*/
|
|
function accumulateDirectDispatchesSingle(abstractEvent) {
|
|
if (abstractEvent && abstractEvent.reactEventType.registrationName) {
|
|
accumulateDispatches(abstractEvent.reactTargetID, null, abstractEvent);
|
|
}
|
|
}
|
|
|
|
function accumulateTwoPhaseDispatches(abstractEvents) {
|
|
if (__DEV__) {
|
|
injection.validate();
|
|
}
|
|
forEachAccumulated(abstractEvents, accumulateTwoPhaseDispatchesSingle);
|
|
}
|
|
|
|
function accumulateEnterLeaveDispatches(leave, enter, fromID, toID) {
|
|
if (__DEV__) {
|
|
injection.validate();
|
|
}
|
|
injection.InstanceHandle.traverseEnterLeave(
|
|
fromID,
|
|
toID,
|
|
accumulateDispatches,
|
|
leave,
|
|
enter
|
|
);
|
|
}
|
|
|
|
|
|
function accumulateDirectDispatches(abstractEvents) {
|
|
if (__DEV__) {
|
|
injection.validate();
|
|
}
|
|
forEachAccumulated(abstractEvents, accumulateDirectDispatchesSingle);
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
* A small set of propagation patterns, each of which will accept a small amount
|
|
* of information, and generate a set of "dispatch ready event objects" - which
|
|
* are sets of events that have already been annotated with a set of dispatched
|
|
* listener functions/ids. The API is designed this way to discourage these
|
|
* propagation strategies from actually executing the dispatches, since we
|
|
* always want to collect the entire set of dispatches before executing event a
|
|
* single one.
|
|
*
|
|
* @constructor EventPropagators
|
|
*/
|
|
var EventPropagators = {
|
|
accumulateTwoPhaseDispatches: accumulateTwoPhaseDispatches,
|
|
accumulateDirectDispatches: accumulateDirectDispatches,
|
|
accumulateEnterLeaveDispatches: accumulateEnterLeaveDispatches,
|
|
injection: injection
|
|
};
|
|
|
|
module.exports = EventPropagators;
|