Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix(Android): onContentSizeChanged is called multiple times If TextInput is in the Modal on Fabric #47492

Open
wants to merge 7 commits into
base: main
Choose a base branch
from

Conversation

coado
Copy link
Contributor

@coado coado commented Nov 7, 2024

Summary:

Fixes #47186

The modal size is updated asynchronously, so the Yoga doesn't know about its size on the initial render. The effect is that children of the modal are sized differently before and after the state is updated. Whenever the size of the TextInput changes it invokes onContentSizeChangedcallback in onLayout and we want it to happen only once.

The fix introduces setting display frame sizes before the initial render in ModalHostViewState similarly to how it is done on iOS. I am not sure if the new getDisplaySizes function should be added to FabricUIManager so I would love to get some guidance here. Also I found that there is getModalHostSize function that is not used anywhere. I've tried to use it but it returns incorrect values after rotating the phone.

The same issue happens on the old architecture, but I will try to make a separate PR with the fix.

Changelog:

[ANDROID] [FIXED] - onContentSizeChanged callback is invoked only once on TextInput located inside the Modal during the mounting on Fabric.

Test Plan:

I've tested with enabled/disabled statusBarTranslucent, with rotating the phone, and on split screen

code
import {Button, Modal, SafeAreaView, StyleSheet, TextInput} from 'react-native';
import React, {useState} from 'react';

export default function App() {
  const [modal, setModal] = useState(false);

  return (
    <SafeAreaView style={styles.container}>
      <Button onPress={() => setModal(true)} title="Show modal" />
      <Modal
        // statusBarTranslucent={true}
        visible={modal}
      >
        <TextInput
          multiline
          onContentSizeChange={e =>
            console.log("inside: ", e.nativeEvent.contentSize.height)
          }
          style={[styles.textInput, styles.textInputModal]}
        />
        <Button onPress={() => setModal(false)} title="Hide modal" />
      </Modal>
    </SafeAreaView>
  );
}

const styles = StyleSheet.create({
  container: {
    backgroundColor: '#d1f0f1',
    flex: 1,
    justifyContent: 'center',
    padding: 8,
  },
  textInput: {
    backgroundColor: 'gray',
  },
  textInputModal: {
    marginTop: 100,
  },
});

@facebook-github-bot facebook-github-bot added CLA Signed This label is managed by the Facebook bot. Authors need to sign the CLA before a PR can be reviewed. p: Software Mansion Partner: Software Mansion Partner Shared with Meta Applied via automation to indicate that an Issue or Pull Request has been shared with the team. labels Nov 7, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
CLA Signed This label is managed by the Facebook bot. Authors need to sign the CLA before a PR can be reviewed. p: Software Mansion Partner: Software Mansion Partner Shared with Meta Applied via automation to indicate that an Issue or Pull Request has been shared with the team.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

TextInput onContentSizeChange triggers twice inside a Modal
2 participants