import React from 'react'; import { DeviceEventEmitter, EventSubscription, NativeEventEmitter, StyleSheet, View, } from 'react-native'; import PortalManager from './portal-manager'; export type PortalHostProps = { children: React.ReactNode; }; export type Operation = | { type: 'mount'; key: number; children: React.ReactNode } | { type: 'update'; key: number; children: React.ReactNode } | { type: 'unmount'; key: number }; export type PortalMethods = { mount: (children: React.ReactNode) => number; update: (key: number, children: React.ReactNode) => void; unmount: (key: number) => void; }; export const PortalContext = React.createContext(null as any); // events const addType = 'NUTUI_RN_ADD_PORTAL'; const removeType = 'NUTUI_RN_REMOVE_PORTAL'; // fix react native web does not support DeviceEventEmitter const TopViewEventEmitter = DeviceEventEmitter || new NativeEventEmitter(); class PortalGuard { private nextKey = 10000; add = (e: React.ReactNode) => { const key = this.nextKey++; TopViewEventEmitter.emit(addType, e, key); return key; }; remove = (key: number) => TopViewEventEmitter.emit(removeType, key); } /** * portal */ export const portal = new PortalGuard(); /** * Portal host renders all of its children `Portal` elements. * For example, you can wrap a screen in `Portal.Host` to render items above the screen. * If you're using the `Provider` component, it already includes `Portal.Host`. * * ## Usage * ```js * import * as React from 'react'; * import { Text } from 'react-native'; * import { Portal } from '@nutui/rn'; * * export default class MyComponent extends React.Component { * render() { * return ( * * Content of the app * * ); * } * } * ``` * * Here any `Portal` elements under `` are rendered alongside `` and will appear above `` like a `Modal`. */ export default class PortalHost extends React.Component { static displayName = 'Portal.Host'; _nextKey = 0; _queue: Operation[] = []; _manager?: PortalManager; _addType: EventSubscription; _removeType: EventSubscription; componentDidMount() { const manager = this._manager; const queue = this._queue; this._addType = TopViewEventEmitter.addListener(addType, this._mount); this._removeType = TopViewEventEmitter.addListener( removeType, this._unmount ); while (queue.length && manager) { const action = queue.pop(); if (!action) { continue; } // tslint:disable-next-line:switch-default switch (action.type) { case 'mount': manager.mount(action.key, action.children); break; case 'update': manager.update(action.key, action.children); break; case 'unmount': manager.unmount(action.key); break; } } } componentWillUnmount() { this._addType.remove(); this._removeType.remove(); } _setManager = (manager?: any) => { this._manager = manager; }; _mount = (children: React.ReactNode, _key?: number) => { const key = _key || this._nextKey++; if (this._manager) { this._manager.mount(key, children); } else { this._queue.push({ type: 'mount', key, children }); } return key; }; _update = (key: number, children: React.ReactNode) => { if (this._manager) { this._manager.update(key, children); } else { const op: Operation = { type: 'mount', key, children }; const index = this._queue.findIndex( (o) => o.type === 'mount' || (o.type === 'update' && o.key === key) ); if (index > -1) { this._queue[index] = op; } else { this._queue.push(op); } } }; _unmount = (key: number) => { if (this._manager) { this._manager.unmount(key); } else { this._queue.push({ type: 'unmount', key }); } }; render() { return ( {/* Need collapsable=false here to clip the elevations, otherwise they appear above Portal components */} {this.props.children} ); } } const styles = StyleSheet.create({ container: { flex: 1, }, });