Single and Multiple React Native Modal Picker With Pagination!
Install
npm i react-native-paginated-modal-picker
Single and Multi Picker Usage
import React, { useState, useEffect } from 'react';
import { View, Text, Dimensions, Image } from 'react-native';
import {
  PaginatedModalPicker,
  PaginatedMultipleModalPicker,
} from 'react-native-paginated-modal-picker';
import axios from 'axios';
const { width, height } = Dimensions.get('window');
export default function App() {
  const [users, setUsers] = useState({
    data: [],
    pageNumber: 1,
    isLoading: false,
  });
  const pages = 2;
  const [selectedUser, setSelectedUser] = useState({
    name: '',
    id: '',
  });
  const [selectedUsers, setSelectedUsers] = useState({
    names: [],
    ids: [],
  });
  const getCountriesData = async (page) => {
    const response = await axios.get(
      `https://reqres.in/api/users?page=${page}`,
      {
        headers: {
          'Content-Type': 'application/json',
        },
      }
    );
    return response.data.data;
  };
  useEffect(() => {
    getCountriesData(users.pageNumber).then((response) => {
      const usersLabeled = response.map((data) =>
        Object.assign({}, { label: data.first_name, value: data.id })
      );
      setUsers({
        ...users,
        data: [...users.data, ...usersLabeled],
        isLoading: false,
      });
    });
  }, [users.pageNumber]);
  return (
    <View
      style={{ flex: 1, alignItems: 'center', justifyContent: 'space-around' }}>
      <PaginatedModalPicker
        textInputStyle={{ paddingLeft: 5 }}
        // modalStyle={{ height: height * 0.4 }}
        flatListStyle={{ padding: 6 }}
        listItemStyle={{
          borderRadius: 5,
          borderColor: 'black',
          borderWidth: 0.4,
          marginTop: 20,
        }}
        listItemTextStyle={{ color: 'blue' }}
        placeholderTextColor='#9b9b9b'
        placeholder={'Select a User'}
        isRequestingComponent={<Text>Loading Component</Text>}
        ListEmptyComponent={<Text>Empty Component</Text>}
        isWorking={true}
        data={users.data}
        // modalType={'FullScreen'}
        backDropOpacity={0.6}
        onSelect={(item) => {
          setSelectedUser({ id: item.value, name: item.label });
        }}
        value={selectedUser.name}
        onEndReached={() =>
          pages > users.pageNumber &&
          setUsers({
            ...users,
            pageNumber: users.pageNumber + 1,
            isLoading: true,
          })
        }
        isRequesting={users.isLoading}
        onEndReachedThreshold={0}
      />
      <PaginatedMultipleModalPicker
        textInputStyle={{ paddingLeft: 5 }}
        // modalStyle={{ height: height * 0.4 }}
        flatListStyle={{ padding: 6 }}
        listItemStyle={{
          borderRadius: 5,
          borderColor: 'black',
          borderWidth: 0.4,
          marginTop: 20,
        }}
        listItemTextStyle={{ color: 'blue' }}
        //new props start
        saveButtonStyle={{ backgroundColor: '#1259a1' }}
        saveButtonTitleStyle={{ fontSize: 17 }}
        saveButtonTitle='Confirm'
        selectedItemImageStyle={{ width: 13, height: 13 }}
        selectedItemIconComponent={
          <Image
            source={require('./images/correct.png')}
            style={{ height: 20, width: 20 }}
          />
        }
        //new props end
        placeholderTextColor='#9b9b9b'
        placeholder={'Select Multiple Users'}
        isRequestingComponent={<Text>Loading Component</Text>}
        ListEmptyComponent={<Text>Empty Component</Text>}
        isWorking={true}
        data={users.data}
        // modalType={'FullScreen'}
        backDropOpacity={0.6}
        onSelect={(item) => {
          setSelectedUsers({ ids: item.values, names: item.labels });
        }}
        value={selectedUsers?.names?.toString()}
        onEndReached={() =>
          pages > users.pageNumber &&
          setUsers({
            ...users,
            pageNumber: users.pageNumber + 1,
            isLoading: true,
          })
        }
        isRequesting={users.isLoading}
        onEndReachedThreshold={0}
      />
    </View>
  );
}
Properties
Single Picker
Props
| Prop | Default | Type | Description | 
|---|
| data | [{label:'', value:''}] | array | an array pagination items | 
| placeholder | Click Me! | string | place holder for textinput | 
| isWorking | true | bool | enable/disable textinput | 
| isRequesting | false | bool | shows flatlist footer component when requesting data with pagination | 
| onSelect |  | call back function | used to get the selected label and value | 
| value |  | string | value of text input after onSelect is fired | 
| modalType | Custom | string | when modalType is 'FullScreen' it become in fullscreen view | 
| onEndReached |  | function | triggered when end of flatlist reached incase of pagination | 
| onEndReachedThreshold | number | number | represents the number of screen lengths you should be from the bottom before it fires the event | 
Components
| Components | Description | 
|---|
| ListEmptyComponent | ListEmptyComponent | 
| isRequestingComponent | ListFooterComponent | 
Styles
| Prop | Default | Type | Description | 
|---|
| textInputStyle | { borderRadius: 10, borderWidth: 1, width: width * 0.8, height: height * 0.05, padding: 10} | style | textInputStyle | 
| modalStyle | { borderRadius: 10, width: width * 0.8, height: height * 0.3, backgroundColor: 'white', alignSelf: 'center'} | style | modalStyle | 
| contentContainerStyle | {} | style | contentContainerStyle for flatlist | 
| flatListStyle | {} | style | style for flatlist | 
| placeholderTextColor |  | hexadecimal | placeholderTextColor | 
| listItemStyle | {padding: 10} | style | style for each item in flatlist | 
| listItemTextStyle | {fontSize: 16} | style | listItemTextStyle | 
| backDropOpacity | 0.65 | number | backDropOpacity of modal | 
Multi Picker ( Can use same Props, Components and Styles of Single Picker in addition to: )
Props
| Prop | Default | Type | Description | 
|---|
| saveButtonTitle | Save | string | title of save button | 
Styles
| Prop | Default | Type | Description | 
|---|
| saveButtonStyle | {borderRadius: 30, minHeight: height * 0.04, width: width * 0.3, backgroundColor: 'black', justifyContent: "center", alignItems: 'center', alignSelf: "center", marginTop: 20, marginHorizontal: width * 0.015} | style | saveButtonStyle | 
| saveButtonTitleStyle | {color: "white", fontSize: width * 0.04} | style | saveButtonTitleStyle | 
| selectedItemIconStyle | {height: 15, width: 15, alignSelf: 'center'} | style | selectedItemIconStyle | 
Components
| Components | Description | 
|---|
| selectedItemIconComponent | selectedItemIconComponent | 
Author
š¤ Ahmed Halbas
š¤ Contributing
Contributions, issues and feature requests are welcome!Feel free to check issues page.
Show your support
Give a āļø if this project helped you!