mirror of
https://github.com/facebook/react-native.git
synced 2025-11-01 09:14:26 +00:00
e7a4dbcefc
Summary: Add annotations to function parameters required for Flow's Local Type Inference project. This codemod prepares the codebase to match Flow's new typechecking algorithm. The new algorithm will make Flow more reliable and predicatable. Reviewed By: evanyeung Differential Revision: D37353648 fbshipit-source-id: e5a0c685ced85a8ff353d578b373f836b376bb28
226 lines
6.4 KiB
JavaScript
226 lines
6.4 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.
|
|
*
|
|
* @flow strict-local
|
|
* @format
|
|
*/
|
|
|
|
import type {PressEvent} from '../Types/CoreEventTypes';
|
|
|
|
import Platform from '../Utilities/Platform';
|
|
import * as PressabilityDebug from '../Pressability/PressabilityDebug';
|
|
import usePressability from '../Pressability/usePressability';
|
|
import StyleSheet from '../StyleSheet/StyleSheet';
|
|
import processColor from '../StyleSheet/processColor';
|
|
import TextAncestor from './TextAncestor';
|
|
import {NativeText, NativeVirtualText} from './TextNativeComponent';
|
|
import {type TextProps} from './TextProps';
|
|
import * as React from 'react';
|
|
import {useContext, useMemo, useState} from 'react';
|
|
|
|
/**
|
|
* Text is the fundamental component for displaying text.
|
|
*
|
|
* @see https://reactnative.dev/docs/text
|
|
*/
|
|
const Text: React.AbstractComponent<
|
|
TextProps,
|
|
React.ElementRef<typeof NativeText | typeof NativeVirtualText>,
|
|
> = React.forwardRef((props: TextProps, forwardedRef) => {
|
|
const {
|
|
accessible,
|
|
allowFontScaling,
|
|
ellipsizeMode,
|
|
onLongPress,
|
|
onPress,
|
|
onPressIn,
|
|
onPressOut,
|
|
onResponderGrant,
|
|
onResponderMove,
|
|
onResponderRelease,
|
|
onResponderTerminate,
|
|
onResponderTerminationRequest,
|
|
onStartShouldSetResponder,
|
|
pressRetentionOffset,
|
|
suppressHighlighting,
|
|
...restProps
|
|
} = props;
|
|
|
|
const [isHighlighted, setHighlighted] = useState(false);
|
|
|
|
const _disabled =
|
|
restProps.disabled != null
|
|
? restProps.disabled
|
|
: props.accessibilityState?.disabled;
|
|
const _accessibilityState =
|
|
_disabled !== props.accessibilityState?.disabled
|
|
? {...props.accessibilityState, disabled: _disabled}
|
|
: props.accessibilityState;
|
|
|
|
const isPressable =
|
|
(onPress != null ||
|
|
onLongPress != null ||
|
|
onStartShouldSetResponder != null) &&
|
|
_disabled !== true;
|
|
|
|
const initialized = useLazyInitialization(isPressable);
|
|
const config = useMemo(
|
|
() =>
|
|
initialized
|
|
? {
|
|
disabled: !isPressable,
|
|
pressRectOffset: pressRetentionOffset,
|
|
onLongPress,
|
|
onPress,
|
|
onPressIn(event: PressEvent) {
|
|
setHighlighted(!suppressHighlighting);
|
|
onPressIn?.(event);
|
|
},
|
|
onPressOut(event: PressEvent) {
|
|
setHighlighted(false);
|
|
onPressOut?.(event);
|
|
},
|
|
onResponderTerminationRequest_DEPRECATED:
|
|
onResponderTerminationRequest,
|
|
onStartShouldSetResponder_DEPRECATED: onStartShouldSetResponder,
|
|
}
|
|
: null,
|
|
[
|
|
initialized,
|
|
isPressable,
|
|
pressRetentionOffset,
|
|
onLongPress,
|
|
onPress,
|
|
onPressIn,
|
|
onPressOut,
|
|
onResponderTerminationRequest,
|
|
onStartShouldSetResponder,
|
|
suppressHighlighting,
|
|
],
|
|
);
|
|
|
|
const eventHandlers = usePressability(config);
|
|
const eventHandlersForText = useMemo(
|
|
() =>
|
|
eventHandlers == null
|
|
? null
|
|
: {
|
|
onResponderGrant(event: PressEvent) {
|
|
eventHandlers.onResponderGrant(event);
|
|
if (onResponderGrant != null) {
|
|
onResponderGrant(event);
|
|
}
|
|
},
|
|
onResponderMove(event: PressEvent) {
|
|
eventHandlers.onResponderMove(event);
|
|
if (onResponderMove != null) {
|
|
onResponderMove(event);
|
|
}
|
|
},
|
|
onResponderRelease(event: PressEvent) {
|
|
eventHandlers.onResponderRelease(event);
|
|
if (onResponderRelease != null) {
|
|
onResponderRelease(event);
|
|
}
|
|
},
|
|
onResponderTerminate(event: PressEvent) {
|
|
eventHandlers.onResponderTerminate(event);
|
|
if (onResponderTerminate != null) {
|
|
onResponderTerminate(event);
|
|
}
|
|
},
|
|
onClick: eventHandlers.onClick,
|
|
onResponderTerminationRequest:
|
|
eventHandlers.onResponderTerminationRequest,
|
|
onStartShouldSetResponder: eventHandlers.onStartShouldSetResponder,
|
|
},
|
|
[
|
|
eventHandlers,
|
|
onResponderGrant,
|
|
onResponderMove,
|
|
onResponderRelease,
|
|
onResponderTerminate,
|
|
],
|
|
);
|
|
|
|
// TODO: Move this processing to the view configuration.
|
|
const selectionColor =
|
|
restProps.selectionColor == null
|
|
? null
|
|
: processColor(restProps.selectionColor);
|
|
|
|
let style = restProps.style;
|
|
if (__DEV__) {
|
|
if (PressabilityDebug.isEnabled() && onPress != null) {
|
|
style = StyleSheet.compose(restProps.style, {
|
|
color: 'magenta',
|
|
});
|
|
}
|
|
}
|
|
|
|
let numberOfLines = restProps.numberOfLines;
|
|
if (numberOfLines != null && !(numberOfLines >= 0)) {
|
|
console.error(
|
|
`'numberOfLines' in <Text> must be a non-negative number, received: ${numberOfLines}. The value will be set to 0.`,
|
|
);
|
|
numberOfLines = 0;
|
|
}
|
|
|
|
const hasTextAncestor = useContext(TextAncestor);
|
|
|
|
const _accessible = Platform.select({
|
|
ios: accessible !== false,
|
|
default: accessible,
|
|
});
|
|
|
|
return hasTextAncestor ? (
|
|
<NativeVirtualText
|
|
{...restProps}
|
|
{...eventHandlersForText}
|
|
isHighlighted={isHighlighted}
|
|
isPressable={isPressable}
|
|
numberOfLines={numberOfLines}
|
|
selectionColor={selectionColor}
|
|
style={style}
|
|
ref={forwardedRef}
|
|
/>
|
|
) : (
|
|
<TextAncestor.Provider value={true}>
|
|
<NativeText
|
|
{...restProps}
|
|
{...eventHandlersForText}
|
|
disabled={_disabled}
|
|
accessible={_accessible}
|
|
accessibilityState={_accessibilityState}
|
|
allowFontScaling={allowFontScaling !== false}
|
|
ellipsizeMode={ellipsizeMode ?? 'tail'}
|
|
isHighlighted={isHighlighted}
|
|
numberOfLines={numberOfLines}
|
|
selectionColor={selectionColor}
|
|
style={style}
|
|
ref={forwardedRef}
|
|
/>
|
|
</TextAncestor.Provider>
|
|
);
|
|
});
|
|
|
|
Text.displayName = 'Text';
|
|
|
|
/**
|
|
* Returns false until the first time `newValue` is true, after which this will
|
|
* always return true. This is necessary to lazily initialize `Pressability` so
|
|
* we do not eagerly create one for every pressable `Text` component.
|
|
*/
|
|
function useLazyInitialization(newValue: boolean): boolean {
|
|
const [oldValue, setValue] = useState(newValue);
|
|
if (!oldValue && newValue) {
|
|
setValue(newValue);
|
|
}
|
|
return oldValue;
|
|
}
|
|
|
|
module.exports = Text;
|