A well-tested, adaptable, lightweight <Popover>
component for react-native. Tested and working on iOS and Android. May work on Web, but not officially supported.
It is written entirely in TypeScript, but uses React Native's native driver for responsive animations, even when the JS thread is busy.
The <Popover>
is able to handle dynamic content and adapt to screen size changes while showing, and will move out of the way for on-screen keyboards automatically.
- Features
- Demo
- Installation
- Usage
- Props
- Usage with Safe Area Context
- Troubleshooting
- Upgrading
- Contributing
- Credits
- Extremely simple but also highly customizable
- Moves to avoid keyboard
- Ability to show from a view, from a rect, or float in center of screen
- Adapts to changing content size
- Automatically detects best placement on screen
- Moves to stay visible on orientation change or when entering split-screen mode
- Great for use in Tablets: you can put entire views that you would normally show in a modal (on a smaller device) into a popover, optionally give it an anchor point, and have it float on top of all of the other views.
You can play around with the various features using the Expo test app. Source Code: react-native-popover-view-test-app
npm i react-native-popover-view
or
yarn add react-native-popover-view
For the simplest usage, just pass your Touchable
into the from
prop. The Popover
will automatically be shown when the Touchable
is pressed.
import React from 'react';
import Popover from 'react-native-popover-view';
function App() {
return (
<Popover
from={(
<TouchableOpacity>
<Text>Press here to open popover!</Text>
</TouchableOpacity>
)}>
<Text>This is the contents of the popover</Text>
</Popover>
);
}
Note that if you pass an onPress
or ref
prop to the Touchable
it will be overwritten.
For more advanced usage, pass in a function that returns any React element. You control which element the popover anchors on (using the sourceRef
) and when the popover will be shown (using the showPopover
callback). In this example, the Popover
will appear to originate from the text inside the popover, and will only be shown when the Touchable
is held down.
import React from 'react';
import Popover from 'react-native-popover-view';
function App() {
return (
<Popover
from={(sourceRef, showPopover) => (
<View>
<TouchableOpacity onLongPress={showPopover}>
<Text ref={sourceRef}>Press here to open popover!</Text>
</TouchableOpacity>
</View>
)}>
<Text>This is the contents of the popover</Text>
</Popover>
);
}
You can control visibility yourself instead of letting the Popover
manage it automatically by using the isVisible
and onRequestClose
prop. This would allow you to manually dismiss the Popover
. onRequestClose
is called when the user taps outside the Popover
. If you want to force the user to tap a button inside the Popover
to dismiss, you could omit onRequestClose
and change the state manually.
import React, { useState, useEffect } from 'react';
import Popover from 'react-native-popover-view';
function App() {
const [showPopover, setShowPopover] = useState(false);
useEffect(() => {
setTimeout(() => setShowPopover(false), 2000);
}, []);
return (
<Popover
isVisible={showPopover}
onRequestClose={() => setShowPopover(false)}
from={(
<TouchableOpacity onPress={() => setShowPopover(true)}>
<Text>Press here to open popover!</Text>
</TouchableOpacity>
)}>
<Text>This popover will be dismissed automatically after 2 seconds</Text>
</Popover>
);
}
If you need even more control (e.g. having the Popover
and Touchable
in complete different parts of the node hierarchy), you can just pass in a normal ref
.
import React, { useRef, useState } from 'react';
import Popover from 'react-native-popover-view';
function App() {
const touchable = useRef();
const [showPopover, setShowPopover] = useState(false);
return (
<>
<TouchableOpacity ref={touchable} onPress={() => setShowPopover(true)}>
<Text>Press here to open popover!</Text>
</TouchableOpacity>
<Popover from={touchable} isVisible={showPopover} onRequestClose={() => setShowPopover(false)}>
<Text>This is the contents of the popover</Text>
</Popover>
</>
);
}
If you already know the exact location of the place you want the Popover
to anchor, you can create a Rect(x, y, width, height)
object, and show from that Rect
. Note that Rect(x, y, 0, 0)
is equivalent to showing from the point (x, y)
.
import React, { useState } from 'react';
import Popover, { Rect } from 'react-native-popover-view';
function App() {
const [showPopover, setShowPopover] = useState(false);
return (
<>
<TouchableOpacity onPress={() => setShowPopover(true)}>
<Text>Press here to open popover!</Text>
</TouchableOpacity>
<Popover from={new Rect(5, 100, 20, 40)} isVisible={showPopover} onRequestClose={() => setShowPopover(false)}>
<Text>This is the contents of the popover</Text>
</Popover>
</>
);
}
If you just want the popover to be centered on the screen, not anchored to anything, you can omit the from
prop altogether.
import React, { useState } from 'react';
import Popover from 'react-native-popover-view';
function App() {
const [showPopover, setShowPopover] = useState(false);
return (
<>
<TouchableOpacity onPress={() => setShowPopover(true)}>
<Text>Press here to open popover!</Text>
</TouchableOpacity>
<Popover isVisible={showPopover} onRequestClose={() => setShowPopover(false)}>
<Text>This popover will stay centered on the screen, even when the device is rotated!</Text>
</Popover>
</>
);
}
Normally, the Popover
will automatically pick the direction it pops out based on where it would fit best on the screen, even showing centered and unanchored if the contents would be compressed otherwise. If you would like to force a direction, you can pass in the placement
prop.
import React from 'react';
import Popover, { PopoverPlacement } from 'react-native-popover-view';
function App() {
return (
<Popover
placement={PopoverPlacement.BOTTOM}
from={(
<TouchableOpacity>
<Text>Press here to open popover!</Text>
</TouchableOpacity>
)}>
<Text>This is the contents of the popover</Text>
</Popover>
);
}
Normally, the popover creates a background that dims the content behind it. You can also show a tooltip without fading the background. Read more about the available modes below. Note that when using TOOLTIP
mode, you must control the visiblity manually (onRequestClose
will never be called).
import React, { useRef, useState } from 'react';
import Popover, { PopoverMode, PopoverPlacement } from 'react-native-popover-view';
function App() {
const [showPopover, setShowPopover] = useState(false);
return (
<Popover
mode={PopoverMode.TOOLTIP}
placement={PopoverPlacement.TOP}
isVisible={showPopover}
from={(
<TouchableOpacity onPress={() => setShowPopover(true)}>
<Text>Press here to open popover!</Text>
</TouchableOpacity>
)}>
<>
<Text>This is the contents of the popover</Text>
<TouchableOpacity onPress={() => setShowPopover(false)}>
<Text>Dismiss</Text>
</TouchableOpacity>
</>
</Popover>
);
}
If you are not using functional components and hooks yet, you can still use class components in almost every case outlined above. Here is an example of using a class component and a ref
, which is slightly different when using class components.
import React, { createRef } from 'react';
import Popover from 'react-native-popover-view';
class App extends React.Component {
constructor(props) {
super(props);
this.touchable = createRef();
this.state = {
showPopover: false
}
}
render() {
return (
<>
<TouchableOpacity ref={this.touchable} onPress={() => this.setState({ showPopover: true })}>
<Text>Press here to open popover!</Text>
</TouchableOpacity>
<Popover
from={this.touchable}
isVisible={this.state.showPopover}
onRequestClose={() => this.setState({ showPopover: false })}>
<Text>This is the contents of the popover</Text>
</Popover>
</>
);
}
}
All props are optional
Prop | Type | Default | Description |
---|---|---|---|
isVisible | bool | false | Show/Hide the popover |
mode | string | 'rn-modal' | One of: 'rn-modal', 'js-modal', 'tooltip'. See Mode section below for details. |
from | element OR | Yes | null |
displayArea | rect | Area where the popover is allowed to be displayed. By default, this will be automatically calculated to be the size of the display, or the size of the parent component if mode is not 'rn-modal'. | |
placement | string | 'auto' | How to position the popover - top | bottom | left | right | center | auto. When 'auto' is specified, it will determine the ideal placement so that the popover is fully visible within displayArea . |
animationConfig | object | An object containing any configuration options that can be passed to Animated.timing (e.g. { duration: 600, easing: Easing.inOut(Easing.quad) } ). The configuration options you pass will override the defaults for all animations. |
|
verticalOffset | number | 0 | The amount to vertically shift the popover on the screen. In certain Android configurations, you may need to apply a verticalOffset of -StatusBar.currentHeight for the popover to originate from the correct place. |
statusBarTranslucent | bool | For 'rn-modal' mode only. Determines whether the background should go under the status bar. Passed through to RN Modal component, see their docs as well. |
|
displayAreaInsets | object | Insets to apply to the display area. The Popover will not be allowed to go beyond the display area minus the insets. | |
popoverStyle | object | The style of the popover itself. You can override the borderRadius , backgroundColor , or any other style prop for a View . |
|
backgroundStyle | object | The style of the background that fades in. | |
arrowStyle | object | The style of the arrow that points to the rect. Supported options are width , height , and backgroundColor . You can use {backgroundColor: 'transparent'} to hide the arrow completely. |
|
arrowShift | number | 0 | How much to shift the arrow to either side, as a multiplier. -1 will shift it all the way to the left (or top) corner of the source view, while 1 will shift all the way to the right (or bottom) corner. A value of 0.5 or -0.8 will shift it partly to one side. |
onOpenStart | function | Callback to be fired when the open animation starts (before animation) | |
onOpenComplete | function | Callback to be fired when the open animation ends (after animation) | |
onRequestClose | function | Callback to be fired when the user taps outside the popover (on the background) or taps the "hardware" back button on Android | |
onCloseStart | function | Callback to be fired when the popover starts closing (before animation) | |
onCloseComplete | function | Callback to be fired when the popover is finished closing (after animation) | |
debug | bool | false | Set this to true to turn on debug logging to the console. This is useful for figuring out why a Popover isn't showing. |
If no from
is provided, the popover will float in the center of the screen.
The Popover can show in three modes:
- Popover.MODE.RN_MODAL ('rn-modal')
- Popover.MODE.JS_MODAL ('js-modal')
- Popover.MODE.TOOLTIP ('tooltip')
Shows the popover full screen in a React Native Modal Component. The upside is that it is guaranteed to show on top of all other views, regardless of where the Popover
component is placed in the view hierarchy. The downside is that you can only have one Modal shown at any one time, so this won't work for nested popovers or if you use a Modal for other reasons. Taps to the area outside the popover will trigger the onRequestClose
callback.
Shows the popover in the space provided, filling the Popover
component's parent. This looks identical to rn-modal
if the Popover
component's parent is a top-level view, but may look weird if the Popover
is nested inside a few views with their own padding and margins. With careful component hierarchy design, this can allow for nested popovers and popovers in other Modals. Taps to the area outside the popover will trigger the onRequestClose
callback.
Shows the Popover
without taking over the screen, no background is faded in and taps to the area around the popover fall through to those views (as expected). The onRequestClose
callback will never be called, so the Popover
will have to be dismissed some other way.
Some devices have notches or other screen features that create zones where you might want to avoid showing a Popover
. To do so, follow the instructions to setup react-native-safe-area-context
, then use the provided hooks to pass the safe area insets straight to the displayAreaInsets
prop:
import { useSafeAreaInsets } from 'react-native-safe-area-context';
import Popover from 'react-native-popover-view';
function PopoverSafeWrapper(props) {
const insets = useSafeAreaInsets();
return (
<Popover {...props} displayAreaInsets={insets} />
);
}
In all cases, start by passing the debug={true}
prop to the Popover, to see if the debug output can help you figure out the issue.
If on an Android device, try adding these props to the component whose ref
you passed in to from
:
renderToHardwareTextureAndroid={true}
collapsable={false}
See facebook/react-native#3282 and SteffeyDev#28 for more info.
Depending on how your app is configured, you may need to use the following verticalOffset
prop to correctly position the popover on Android:
import { Platform, StatusBar, ... } from 'react-native';
...
<Popover
{...otherProps}
verticalOffset={Platform.OS === 'android' ? -StatusBar.currentHeight : 0 }
/>
Removed internal safe area view; if you want the Popover to avoid showing behind notches on some devices, follow the instructions: Usage with Safe Area Context.
fromRect
andfromView
have been consolidated into a singlefrom
prop, where you can pass a Rect or a Ref. All Refs passed in must now be aRefObject
created fromReact.createRef
orReact.useRef
. All Rects passed in must be an actual Rect object (e.g.from={new Rect(x, y, width, height)}
).from
prop now supports additional modes, including passing in a React element for simpler usage. See new examples and usage notes above.fromDynamicRect
has been removed.
onClose
has been renamed toonRequestClose
(for clarity and consistency with normal RN Modals)- New
mode
prop replacesshowInModal
. ReplaceshowInModal={false}
withmode={Popover.MODE.JS_MODAL}
doneClosingCallback
has been renamedonCloseComplete
- New
backgroundStyle
prop replacesshowBackground
. ReplaceshowBackground={false}
withbackgroundStyle={{ backgroundColor: 'transparent' }}
- Fix in version 2.0 means that
verticalOffset
may no longer be needed, so if you are using it be sure to test
This version moved the react-navigation portion of this project to it's own repository: react-navigation-popover.
To use with react-navigation, install that npm package change import { createPopoverStackNavigator } from 'react-native-popover-view'
to import createPopoverStackNavigator from 'react-navigation-popover'
.
The only breaking change in version 1.0 was renaming PopoverStackNavigator
to createPopoverStackNavigator
, to match the react-navigation
other navigation functions.
Version 0.6 brought some large changes, increasing efficiency, stability, and flexibility. For React Navigation users, there is a new prop, showInPopover
, that you might want to pass to createPopoverStackNavigator
if you want to customize when to show stack views in a Popover. This replaces PopoverNavigation.shouldShowInPopover
. See the new setup instructions below for details.
Pull requests are welcome; if you find that you are having to bend over backwards to make this work for you, feel free to open an issue or PR! Of course, try to keep the same coding style if possible and I'll try to get back to you as soon as I can.
Use yarn build
to build the dist
folder (compile TypeScript to JavaScript), and use yarn watch
to continuously build on save.
This is a fork of react-native-popover, originally created by Jean Regisser jean.regisser@gmail.com (https://github.com/jeanregisser) but since abandoned.
I have rebuilt most of the library from the ground up for improved handling of changing screen sizes on tablets (split-screen mode), a redesigned automatic placement algorithm, TypeScript, and ES6 compatibility.
Similar forks exist on Github (such as react-native-modal-popover), but this is the first to be published on NPM as far as I know.
MIT Licensed