mirror of
https://github.com/facebook/react.git
synced 2026-02-24 12:43:00 +00:00
## Overview Reverts https://github.com/facebook/react/pull/26616 and implements the suggested way instead. This change in #26616 broken the internal sync command, which now results in duplicated `@generated` headers. It also makes it harder to detect changes during the diff train sync. Instead, we will check for changes, and if there are changes sign the files and commit them to the sync branch. ## Strategy The new sync strategy accounts for the generated headers during the sync: - **Revert Version**: Revert the version strings - **Revert @generated**: Re-sign the files (will be the same hash as before if unchanged) - **Check**: Check if there are changes **if not, skip** - **Re-apply Version**: Now add back the new version string - **Re-sign @generated**: And re-generate the headers Then commit to branch. This ensures that if there are no changes, we'll skip. --------- Co-authored-by: Timothy Yung <yungsters@gmail.com>
124 lines
3.5 KiB
JavaScript
124 lines
3.5 KiB
JavaScript
/**
|
|
* 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.
|
|
*
|
|
* @noformat
|
|
* @nolint
|
|
* @flow strict-local
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
import {type ViewConfig} from './ReactNativeTypes';
|
|
import invariant from 'invariant';
|
|
|
|
// Event configs
|
|
export const customBubblingEventTypes: {
|
|
[eventName: string]: $ReadOnly<{
|
|
phasedRegistrationNames: $ReadOnly<{
|
|
captured: string,
|
|
bubbled: string,
|
|
skipBubbling?: ?boolean,
|
|
}>,
|
|
}>,
|
|
} = {};
|
|
export const customDirectEventTypes: {
|
|
[eventName: string]: $ReadOnly<{
|
|
registrationName: string,
|
|
}>,
|
|
} = {};
|
|
|
|
const viewConfigCallbacks = new Map<string, ?() => ViewConfig>();
|
|
const viewConfigs = new Map<string, ViewConfig>();
|
|
|
|
function processEventTypes(viewConfig: ViewConfig): void {
|
|
const {bubblingEventTypes, directEventTypes} = viewConfig;
|
|
|
|
if (__DEV__) {
|
|
if (bubblingEventTypes != null && directEventTypes != null) {
|
|
for (const topLevelType in directEventTypes) {
|
|
invariant(
|
|
bubblingEventTypes[topLevelType] == null,
|
|
'Event cannot be both direct and bubbling: %s',
|
|
topLevelType,
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (bubblingEventTypes != null) {
|
|
for (const topLevelType in bubblingEventTypes) {
|
|
if (customBubblingEventTypes[topLevelType] == null) {
|
|
customBubblingEventTypes[topLevelType] =
|
|
bubblingEventTypes[topLevelType];
|
|
}
|
|
}
|
|
}
|
|
|
|
if (directEventTypes != null) {
|
|
for (const topLevelType in directEventTypes) {
|
|
if (customDirectEventTypes[topLevelType] == null) {
|
|
customDirectEventTypes[topLevelType] = directEventTypes[topLevelType];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Registers a native view/component by name.
|
|
* A callback is provided to load the view config from UIManager.
|
|
* The callback is deferred until the view is actually rendered.
|
|
*/
|
|
export function register(name: string, callback: () => ViewConfig): string {
|
|
invariant(
|
|
!viewConfigCallbacks.has(name),
|
|
'Tried to register two views with the same name %s',
|
|
name,
|
|
);
|
|
invariant(
|
|
typeof callback === 'function',
|
|
'View config getter callback for component `%s` must be a function (received `%s`)',
|
|
name,
|
|
callback === null ? 'null' : typeof callback,
|
|
);
|
|
viewConfigCallbacks.set(name, callback);
|
|
return name;
|
|
}
|
|
|
|
/**
|
|
* Retrieves a config for the specified view.
|
|
* If this is the first time the view has been used,
|
|
* This configuration will be lazy-loaded from UIManager.
|
|
*/
|
|
export function get(name: string): ViewConfig {
|
|
let viewConfig;
|
|
if (!viewConfigs.has(name)) {
|
|
const callback = viewConfigCallbacks.get(name);
|
|
if (typeof callback !== 'function') {
|
|
invariant(
|
|
false,
|
|
'View config getter callback for component `%s` must be a function (received `%s`).%s',
|
|
name,
|
|
callback === null ? 'null' : typeof callback,
|
|
// $FlowFixMe[recursive-definition]
|
|
typeof name[0] === 'string' && /[a-z]/.test(name[0])
|
|
? ' Make sure to start component names with a capital letter.'
|
|
: '',
|
|
);
|
|
}
|
|
viewConfig = callback();
|
|
processEventTypes(viewConfig);
|
|
viewConfigs.set(name, viewConfig);
|
|
|
|
// Clear the callback after the config is set so that
|
|
// we don't mask any errors during registration.
|
|
viewConfigCallbacks.set(name, null);
|
|
} else {
|
|
viewConfig = viewConfigs.get(name);
|
|
}
|
|
invariant(viewConfig, 'View config not found for name %s', name);
|
|
return viewConfig;
|
|
}
|