Skip to content

Commit 06751aa

Browse files
NickGerlemanfacebook-github-bot
authored andcommitted
"experimental_layoutConformance" ViewProp -> "experimental_LayoutConformance" component (#48188)
Summary: Pull Request resolved: #48188 Yoga is full of bugs! Some of these bugs cannot be fixed without breaking large swaths of product code. To get around this, we introduced "errata" to Yoga as a mechanism to preserve bug compatibility, and an `experimental_layoutConformance` prop in React Native to create layout conformance contexts. This has allowed us to create more compliant layout behavior for XPR. This prop was originally designed as a context-like component, so you could set a conformance level at the root of your app, and individual components could change it for compatibility. This was difficult to achieve at the time, without introducing a primitive like `LayoutConformanceView`, which itself participated in the view tree. This prop has not been the desired end-goal, since it does not make clear that it is setting a whole new context, effecting children as well! Now that we've landed support for `display: contents`, we can achieve this desired API pretty easily. **Before** ``` import {View} from 'react-native'; // Root of the app <View {...props} experimental_layoutConformance="strict"> {content} </View> ``` **After** ``` import {View, experimental_LayoutConformance as LayoutConformance} from 'react-native'; // Root of the app <LayoutConformance mode="strict"> <View {...props}> {content} </View> </LayoutConformance> ``` Changelog: [Internal] Reviewed By: javache Differential Revision: D66910054 fbshipit-source-id: e6a304b5c30ad3c5845a7ce2d1021996a74c2f34
1 parent 4adaacb commit 06751aa

28 files changed

+361
-134
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
/**
2+
* Copyright (c) Meta Platforms, Inc. and affiliates.
3+
*
4+
* This source code is licensed under the MIT license found in the
5+
* LICENSE file in the root directory of this source tree.
6+
*
7+
* @format
8+
*/
9+
10+
import type * as React from 'react';
11+
12+
type LayoutConformanceProps = {
13+
/**
14+
* strict: Layout in accordance with W3C spec, even when breaking
15+
* compatibility: Layout with the same behavior as previous versions of React Native
16+
*/
17+
mode: 'strict' | 'compatibility';
18+
children: React.ReactNode;
19+
};
20+
21+
export const experimental_LayoutConformance: React.ComponentType<LayoutConformanceProps>;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
/**
2+
* Copyright (c) Meta Platforms, Inc. and affiliates.
3+
*
4+
* This source code is licensed under the MIT license found in the
5+
* LICENSE file in the root directory of this source tree.
6+
*
7+
* @flow strict-local
8+
* @format
9+
* @oncall react_native
10+
*/
11+
12+
import StyleSheet from '../../StyleSheet/StyleSheet';
13+
import LayoutConformanceNativeComponent from './LayoutConformanceNativeComponent';
14+
import * as React from 'react';
15+
16+
type Props = $ReadOnly<{
17+
/**
18+
* strict: Layout in accordance with W3C spec, even when breaking
19+
* compatibility: Layout with the same behavior as previous versions of React Native
20+
*/
21+
mode: 'strict' | 'compatibility',
22+
23+
children: React.Node,
24+
}>;
25+
26+
// We want a graceful fallback for apps using legacy arch, but need to know
27+
// ahead of time whether the component is available, so we test for global.
28+
// This does not correctly handle mixed arch apps (which is okay, since we just
29+
// degrade the error experience).
30+
const isFabricUIManagerInstalled = global?.nativeFabricUIManager != null;
31+
32+
function LayoutConformance(props: Props): React.Node {
33+
return (
34+
<LayoutConformanceNativeComponent {...props} style={styles.container} />
35+
);
36+
}
37+
38+
function UnimplementedLayoutConformance(props: Props): React.Node {
39+
if (__DEV__) {
40+
const warnOnce = require('../../Utilities/warnOnce');
41+
42+
warnOnce(
43+
'layoutconformance-unsupported',
44+
'"LayoutConformance" is only supported in the New Architecture',
45+
);
46+
}
47+
48+
return props.children;
49+
}
50+
51+
export default (isFabricUIManagerInstalled
52+
? LayoutConformance
53+
: UnimplementedLayoutConformance) as component(...Props);
54+
55+
const styles = StyleSheet.create({
56+
container: {
57+
display: 'contents',
58+
},
59+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
/**
2+
* Copyright (c) Meta Platforms, Inc. and affiliates.
3+
*
4+
* This source code is licensed under the MIT license found in the
5+
* LICENSE file in the root directory of this source tree.
6+
*
7+
* @flow strict-local
8+
* @format
9+
*/
10+
11+
import type {HostComponent} from '../../Renderer/shims/ReactNativeTypes';
12+
import type {ViewProps} from '../View/ViewPropTypes';
13+
14+
import * as NativeComponentRegistry from '../../NativeComponent/NativeComponentRegistry';
15+
16+
type Props = $ReadOnly<{
17+
mode: 'strict' | 'compatibility',
18+
...ViewProps,
19+
}>;
20+
21+
const LayoutConformanceNativeComponent: HostComponent<Props> =
22+
NativeComponentRegistry.get<Props>('LayoutConformance', () => ({
23+
uiViewClassName: 'LayoutConformance',
24+
validAttributes: {
25+
mode: true,
26+
},
27+
}));
28+
29+
export default LayoutConformanceNativeComponent;

packages/react-native/Libraries/Components/View/ViewPropTypes.d.ts

-7
Original file line numberDiff line numberDiff line change
@@ -211,11 +211,4 @@ export interface ViewProps
211211
* Used to reference react managed views from native code.
212212
*/
213213
nativeID?: string | undefined;
214-
215-
/**
216-
* Contols whether this view, and its transitive children, are laid in a way
217-
* consistent with web browsers ('strict'), or consistent with existing
218-
* React Native code which may rely on incorrect behavior ('classic').
219-
*/
220-
experimental_layoutConformance?: 'strict' | 'classic' | undefined;
221214
}

packages/react-native/Libraries/Components/View/ViewPropTypes.js

-9
Original file line numberDiff line numberDiff line change
@@ -578,15 +578,6 @@ export type ViewProps = $ReadOnly<{|
578578
*/
579579
collapsableChildren?: ?boolean,
580580

581-
/**
582-
* Contols whether this view, and its transitive children, are laid in a way
583-
* consistent with web browsers ('strict'), or consistent with existing
584-
* React Native code which may rely on incorrect behavior ('classic').
585-
*
586-
* This prop only works when using Fabric.
587-
*/
588-
experimental_layoutConformance?: ?('strict' | 'classic'),
589-
590581
/**
591582
* Used to locate this view from native classes. Has precedence over `nativeID` prop.
592583
*

packages/react-native/Libraries/NativeComponent/BaseViewConfig.android.js

-2
Original file line numberDiff line numberDiff line change
@@ -293,8 +293,6 @@ const validAttributesForNonEventProps = {
293293

294294
style: ReactNativeStyleAttributes,
295295

296-
experimental_layoutConformance: true,
297-
298296
// ReactClippingViewManager @ReactProps
299297
removeClippedSubviews: true,
300298

packages/react-native/Libraries/NativeComponent/BaseViewConfig.ios.js

-2
Original file line numberDiff line numberDiff line change
@@ -357,8 +357,6 @@ const validAttributesForNonEventProps = {
357357
direction: true,
358358

359359
style: ReactNativeStyleAttributes,
360-
361-
experimental_layoutConformance: true,
362360
};
363361

364362
// Props for bubbling and direct events

packages/react-native/Libraries/__tests__/__snapshots__/public-api-test.js.snap

+20-1
Original file line numberDiff line numberDiff line change
@@ -1887,6 +1887,25 @@ declare export default typeof NativeKeyboardObserver;
18871887
"
18881888
`;
18891889

1890+
exports[`public API should not change unintentionally Libraries/Components/LayoutConformance/LayoutConformance.js 1`] = `
1891+
"type Props = $ReadOnly<{
1892+
mode: \\"strict\\" | \\"compatibility\\",
1893+
children: React.Node,
1894+
}>;
1895+
declare export default component(...Props);
1896+
"
1897+
`;
1898+
1899+
exports[`public API should not change unintentionally Libraries/Components/LayoutConformance/LayoutConformanceNativeComponent.js 1`] = `
1900+
"type Props = $ReadOnly<{
1901+
mode: \\"strict\\" | \\"compatibility\\",
1902+
...ViewProps,
1903+
}>;
1904+
declare const LayoutConformanceNativeComponent: HostComponent<Props>;
1905+
declare export default typeof LayoutConformanceNativeComponent;
1906+
"
1907+
`;
1908+
18901909
exports[`public API should not change unintentionally Libraries/Components/Pressable/Pressable.js 1`] = `
18911910
"type ViewStyleProp = $ElementType<React.ElementConfig<typeof View>, \\"style\\">;
18921911
export type StateCallbackType = $ReadOnly<{|
@@ -4268,7 +4287,6 @@ export type ViewProps = $ReadOnly<{|
42684287
\\"aria-hidden\\"?: ?boolean,
42694288
collapsable?: ?boolean,
42704289
collapsableChildren?: ?boolean,
4271-
experimental_layoutConformance?: ?(\\"strict\\" | \\"classic\\"),
42724290
id?: string,
42734291
testID?: ?string,
42744292
nativeID?: ?string,
@@ -9603,6 +9621,7 @@ declare module.exports: {
96039621
get Image(): Image,
96049622
get ImageBackground(): ImageBackground,
96059623
get InputAccessoryView(): InputAccessoryView,
9624+
get experimental_LayoutConformance(): LayoutConformance,
96069625
get KeyboardAvoidingView(): KeyboardAvoidingView,
96079626
get Modal(): Modal,
96089627
get Pressable(): Pressable,

packages/react-native/React/Views/RCTViewManager.m

-7
Original file line numberDiff line numberDiff line change
@@ -424,13 +424,6 @@ - (void)updateAccessibilityTraitsForRole:(RCTView *)view withDefaultView:(RCTVie
424424
// filtered by view configs.
425425
}
426426

427-
RCT_CUSTOM_VIEW_PROPERTY(experimental_layoutConformance, NSString *, RCTView)
428-
{
429-
// Property is only to be used in the new renderer.
430-
// It is necessary to add it here, otherwise it gets
431-
// filtered by view configs.
432-
}
433-
434427
typedef NSArray *FilterArray; // Custom type to make the StaticViewConfigValidator Happy
435428
RCT_CUSTOM_VIEW_PROPERTY(filter, FilterArray, RCTView)
436429
{

packages/react-native/ReactAndroid/api/ReactAndroid.api

-1
Original file line numberDiff line numberDiff line change
@@ -4137,7 +4137,6 @@ public class com/facebook/react/uimanager/LayoutShadowNode : com/facebook/react/
41374137
public fun setInsetBlock (ILcom/facebook/react/bridge/Dynamic;)V
41384138
public fun setInsetInline (ILcom/facebook/react/bridge/Dynamic;)V
41394139
public fun setJustifyContent (Ljava/lang/String;)V
4140-
public fun setLayoutConformance (Ljava/lang/String;)V
41414140
public fun setMarginBlock (ILcom/facebook/react/bridge/Dynamic;)V
41424141
public fun setMarginInline (ILcom/facebook/react/bridge/Dynamic;)V
41434142
public fun setMargins (ILcom/facebook/react/bridge/Dynamic;)V

packages/react-native/ReactAndroid/src/main/java/com/facebook/react/uimanager/LayoutShadowNode.java

-5
Original file line numberDiff line numberDiff line change
@@ -972,9 +972,4 @@ public void setShouldNotifyPointerLeave(boolean value) {
972972
public void setShouldNotifyPointerMove(boolean value) {
973973
// Do Nothing: Align with static ViewConfigs
974974
}
975-
976-
@ReactProp(name = "experimental_layoutConformance")
977-
public void setLayoutConformance(String value) {
978-
// Do Nothing: Align with static ViewConfigs
979-
}
980975
}

packages/react-native/ReactAndroid/src/main/jni/react/fabric/CoreComponentsRegistry.cpp

+3
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,7 @@
2222
#include <react/renderer/components/text/ParagraphComponentDescriptor.h>
2323
#include <react/renderer/components/text/RawTextComponentDescriptor.h>
2424
#include <react/renderer/components/text/TextComponentDescriptor.h>
25+
#include <react/renderer/components/view/LayoutConformanceComponentDescriptor.h>
2526
#include <react/renderer/components/view/ViewComponentDescriptor.h>
2627

2728
namespace facebook::react::CoreComponentsRegistry {
@@ -66,6 +67,8 @@ sharedProviderRegistry() {
6667
AndroidDrawerLayoutComponentDescriptor>());
6768
providerRegistry->add(concreteComponentDescriptorProvider<
6869
DebuggingOverlayComponentDescriptor>());
70+
providerRegistry->add(concreteComponentDescriptorProvider<
71+
LayoutConformanceComponentDescriptor>());
6972

7073
return providerRegistry;
7174
}();

packages/react-native/ReactCommon/react/renderer/components/view/BaseViewProps.cpp

+1-11
Original file line numberDiff line numberDiff line change
@@ -343,16 +343,7 @@ BaseViewProps::BaseViewProps(
343343
rawProps,
344344
"removeClippedSubviews",
345345
sourceProps.removeClippedSubviews,
346-
false)),
347-
experimental_layoutConformance(
348-
ReactNativeFeatureFlags::enableCppPropsIteratorSetter()
349-
? sourceProps.experimental_layoutConformance
350-
: convertRawProp(
351-
context,
352-
rawProps,
353-
"experimental_layoutConformance",
354-
sourceProps.experimental_layoutConformance,
355-
{})) {}
346+
false)) {}
356347

357348
#define VIEW_EVENT_CASE(eventType) \
358349
case CONSTEXPR_RAW_PROPS_KEY_HASH("on" #eventType): { \
@@ -398,7 +389,6 @@ void BaseViewProps::setProp(
398389
RAW_SET_PROP_SWITCH_CASE_BASIC(collapsable);
399390
RAW_SET_PROP_SWITCH_CASE_BASIC(collapsableChildren);
400391
RAW_SET_PROP_SWITCH_CASE_BASIC(removeClippedSubviews);
401-
RAW_SET_PROP_SWITCH_CASE_BASIC(experimental_layoutConformance);
402392
RAW_SET_PROP_SWITCH_CASE_BASIC(cursor);
403393
RAW_SET_PROP_SWITCH_CASE_BASIC(outlineColor);
404394
RAW_SET_PROP_SWITCH_CASE_BASIC(outlineOffset);

packages/react-native/ReactCommon/react/renderer/components/view/BaseViewProps.h

-2
Original file line numberDiff line numberDiff line change
@@ -108,8 +108,6 @@ class BaseViewProps : public YogaStylableProps, public AccessibilityProps {
108108

109109
bool removeClippedSubviews{false};
110110

111-
LayoutConformance experimental_layoutConformance{};
112-
113111
#pragma mark - Convenience Methods
114112

115113
CascadedBorderWidths getBorderWidths() const;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
/*
2+
* Copyright (c) Meta Platforms, Inc. and affiliates.
3+
*
4+
* This source code is licensed under the MIT license found in the
5+
* LICENSE file in the root directory of this source tree.
6+
*/
7+
8+
#pragma once
9+
10+
#include <react/renderer/components/view/LayoutConformanceShadowNode.h>
11+
#include <react/renderer/core/ConcreteComponentDescriptor.h>
12+
13+
namespace facebook::react {
14+
15+
using LayoutConformanceComponentDescriptor =
16+
ConcreteComponentDescriptor<LayoutConformanceShadowNode>;
17+
18+
} // namespace facebook::react
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
/*
2+
* Copyright (c) Meta Platforms, Inc. and affiliates.
3+
*
4+
* This source code is licensed under the MIT license found in the
5+
* LICENSE file in the root directory of this source tree.
6+
*/
7+
8+
#pragma once
9+
10+
#include <react/renderer/components/view/YogaStylableProps.h>
11+
#include <react/renderer/components/view/propsConversions.h>
12+
13+
namespace facebook::react {
14+
15+
struct LayoutConformanceProps final : public YogaStylableProps {
16+
/**
17+
* Whether to layout the subtree with strict conformance to W3C standard
18+
* (YGErrataNone) or for compatibility with legacy RN bugs (YGErrataAll)
19+
*/
20+
LayoutConformance mode{LayoutConformance::Strict};
21+
22+
LayoutConformanceProps() = default;
23+
LayoutConformanceProps(
24+
const PropsParserContext& context,
25+
const LayoutConformanceProps& sourceProps,
26+
const RawProps& rawProps)
27+
: YogaStylableProps(context, sourceProps, rawProps),
28+
mode{convertRawProp(
29+
context,
30+
rawProps,
31+
"mode",
32+
mode,
33+
LayoutConformance::Strict)} {}
34+
};
35+
36+
} // namespace facebook::react
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
/*
2+
* Copyright (c) Meta Platforms, Inc. and affiliates.
3+
*
4+
* This source code is licensed under the MIT license found in the
5+
* LICENSE file in the root directory of this source tree.
6+
*/
7+
8+
#pragma once
9+
10+
#include <react/renderer/components/view/ConcreteViewShadowNode.h>
11+
#include <react/renderer/components/view/LayoutConformanceProps.h>
12+
#include <react/renderer/components/view/YogaLayoutableShadowNode.h>
13+
14+
namespace facebook::react {
15+
16+
constexpr const char LayoutConformanceShadowNodeComponentName[] =
17+
"LayoutConformance";
18+
19+
using LayoutConformanceShadowNode = ConcreteShadowNode<
20+
LayoutConformanceShadowNodeComponentName,
21+
YogaLayoutableShadowNode,
22+
LayoutConformanceProps>;
23+
24+
} // namespace facebook::react

packages/react-native/ReactCommon/react/renderer/components/view/YogaLayoutableShadowNode.cpp

+6-7
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@
1111
#include <react/debug/flags.h>
1212
#include <react/debug/react_native_assert.h>
1313
#include <react/featureflags/ReactNativeFeatureFlags.h>
14+
#include <react/renderer/components/view/LayoutConformanceShadowNode.h>
1415
#include <react/renderer/components/view/ViewProps.h>
1516
#include <react/renderer/components/view/ViewShadowNode.h>
1617
#include <react/renderer/components/view/conversions.h>
@@ -501,15 +502,13 @@ void YogaLayoutableShadowNode::configureYogaTree(
501502
}
502503

503504
YGErrata YogaLayoutableShadowNode::resolveErrata(YGErrata defaultErrata) const {
504-
if (auto viewShadowNode = dynamic_cast<const ViewShadowNode*>(this)) {
505-
const auto& props = viewShadowNode->getConcreteProps();
506-
switch (props.experimental_layoutConformance) {
507-
case LayoutConformance::Classic:
508-
return YGErrataAll;
505+
if (auto layoutConformanceNode =
506+
dynamic_cast<const LayoutConformanceShadowNode*>(this)) {
507+
switch (layoutConformanceNode->getConcreteProps().mode) {
509508
case LayoutConformance::Strict:
510509
return YGErrataNone;
511-
case LayoutConformance::Undefined:
512-
return defaultErrata;
510+
case LayoutConformance::Compatibility:
511+
return YGErrataAll;
513512
}
514513
}
515514

0 commit comments

Comments
 (0)