Skip to content

AppAndFlow/react-native-magic-scroll

Repository files navigation

magic-scroll

About

App & Flow is a Montreal-based, close-knit team that specializes in React Native and Expo development. We work with multiple YC-backed startups and are recommended by Expo. Need a hand? Let’s build together. [email protected]

npm (scoped)

Why react-native-magic-scroll?

The goal of the library is to seamlessly and precisely handle your keyboard, scrollview and inputs when interacting with forms. While other solutions offer plug-and-play functionalities, we wanted to have something more precise and with more flexibility so that it can be used in any situation.

Examples

We recreated two flows from popular apps to showcase our library in action. The demo app code is available here.

Twitch's sign up Shop's check out
RPReplay_Final1717175614.mov
RPReplay_Final1717177734.mov

Installation

react-native-magic-scroll

yarn add @appandflow/react-native-magic-scroll
npm i @appandflow/react-native-magic-scroll

Dependencies

To use our library, you will need to install these two dependencies into your project.

1) React Native Reanimated

npx expo install react-native-reanimated

Learn more about this dependency here.

2) SafeAreaContext

npx expo install react-native-safe-area-context

Learn more about this dependency here.

Android

On Android, make sure to set android:windowSoftInputMode in your AndroidManifest.xml to pan

Expo

// app.json
"android": {
  ...rest,
  "softwareKeyboardLayoutMode": "pan"
}

Basic Usage

Wrap your screen within our ScrollView.

import { MagicScroll } from '@appandflow/react-native-magic-scroll';
// rest of your imports

const YourScreen = () => {
  return (
    <MagicScroll.ScrollView>
      // Your form
    </MagicScroll.ScrollView>
  );
};

export default YourScreen;

You then use our TextInputs for the form itself, that you place inside the MagicScroll.ScrollView. Easily "chain" your inputs (so that the return keyboard button hops to the next desired input) by using the MagicScroll.TextInput name and chainTo props, like so:

import { MagicScroll } from '@appandflow/react-native-magic-scroll';
// rest of your imports

const textInputStyle = {
  height: 50,
  backgroundColor: '#ddd',
  borderRadius: 10,
  marginTop: 8,
};


const YourScreen = () => {
  return (
    <MagicScroll.ScrollView>
      <MagicScroll.TextInput
        // This is the name of this text input, used by the `chainTo` prop
        name="email"
        // This is where you can design your a custom label above the input
        renderTop={() => <Text>Email</Text>}
        // This is where you can design your custom label below the input
        renderBottom={() => <Text>Must be unique</Text>}
        // This is the function that will make the text input named "password" focused when pressing the Enter or Return key on the device's keyboard
        chainTo="password"
        textInputProps={{
          style: textInputStyle,
        }}
      />
      <MagicScroll.TextInput
        name="password"
        renderTop={() => <Text>Password</Text>}
        textInputProps={{
          secureTextEntry: true,
          style: textInputStyle,
        }}
      />
    </MagicScroll.ScrollView>
  );
};

Advanced

As mentioned in the introduction, the drawbacks of a plug-and-play library are its limitations when deviating from standard functionality. That's precisely why our library allows for customization, enabling you to tailor its usage to suit your specific needs and use cases.

Tips

It's a great idea to wrap our MagicScroll.TextInput within your own for re-usability!

Here's an example

import { MagicScroll } from '@appandflow/react-native-magic-scroll';

// rest of your imports

interface Props {
  label?: string;
  isPassword?: boolean;
  name?: string;
  description?: string;
  chainTo?: string;
}

const YourCustomInput = (props: Props) => {
  return (
    <MagicScroll.TextInput
      name={props.name}
      chainTo={props.chainTo}
      renderTop={() => <Text>{props.label}</Text>}
      renderBottom={() => <Text>{props.description}</Text>}
      textInputProps={{
        secureTextEntry: props.isPassword,
        style: {
          height: 50,
          backgroundColor: '#ddd',
          borderRadius: 10,
          marginTop: 8,
        },
      }}
    />
  );
};

Props (Optional)

All of these props are optional. It is, however, recommended to use them to get the most out of the library.

MagicScroll.ScrollView:

Name Description Values
additionalPadding adds extra padding between your text input and the keyboard number
scrollViewProps contains all props of the scrollview from React's Reanimated library props

MagicScroll.TextInput:

Name Description Values
chainTo a string containing the name of the next text input that will be focused when pressing the "Enter Key" string
containerStyle contains all Style props of the View from React Native props
name a string to name the current text input, used in the "chainTo" props mentionned above string
renderBottom() a function that renders components to display custom text under the text input renderBottom={() => <Text>bottomText</Text>}
renderTop() a function that renders components to display custom text above the text input renderTop={() => <Text>topText</Text>}
textInputProps contains all props of the TextInput component from React Native props