mirror of
https://github.com/status-im/react-native.git
synced 2025-01-16 20:44:10 +00:00
613ca14612
Summary:Since the React 0.14 split of modules, the findNodeHandle feature is part of the renderer and not the generic React API. This just greps for React.findNodeHandle and replace them with ReactNative.findNodeHandle. I fixed up the imports manually. I also found two callers each of ReactNative.createClass and React.render with the exception of downstream and examples will fix them separately. I'll need to find more things like `var { PropTypes } = ReactNative;` separately. I think this is a good start though. Reviewed By: vjeux Differential Revision: D3149356 fb-gh-sync-id: 50ed60bc67270b16f561d4c641f2f19e85724d3b fbshipit-source-id: 50ed60bc67270b16f561d4c641f2f19e85724d3b
185 lines
5.1 KiB
JavaScript
185 lines
5.1 KiB
JavaScript
/**
|
|
* Copyright (c) 2013-present, Facebook, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This source code is licensed under the BSD-style license found in the
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
*
|
|
* @providesModule Portal
|
|
* @flow
|
|
*/
|
|
'use strict';
|
|
|
|
var Platform = require('Platform');
|
|
var React = require('React');
|
|
var ReactNative = require('ReactNative');
|
|
var StyleSheet = require('StyleSheet');
|
|
var UIManager = require('UIManager');
|
|
var View = require('View');
|
|
|
|
var _portalRef: any;
|
|
|
|
// Unique identifiers for modals.
|
|
var lastUsedTag = 0;
|
|
|
|
/*
|
|
* Note: Only intended for Android at the moment. Just use Modal in your iOS
|
|
* code.
|
|
*
|
|
* A container that renders all the modals on top of everything else in the application.
|
|
*
|
|
* Portal makes it possible for application code to pass modal views all the way up to
|
|
* the root element created in `renderApplication`.
|
|
*
|
|
* Never use `<Portal>` in your code. There is only one Portal instance rendered
|
|
* by the top-level `renderApplication`.
|
|
*/
|
|
var Portal = React.createClass({
|
|
statics: {
|
|
/**
|
|
* Use this to create a new unique tag for your component that renders
|
|
* modals. A good place to allocate a tag is in `componentWillMount`
|
|
* of your component.
|
|
* See `showModal` and `closeModal`.
|
|
*/
|
|
allocateTag: function(): string {
|
|
return '__modal_' + (++lastUsedTag);
|
|
},
|
|
|
|
/**
|
|
* Render a new modal.
|
|
* @param tag A unique tag identifying the React component to render.
|
|
* This tag can be later used in `closeModal`.
|
|
* @param component A React component to be rendered.
|
|
*/
|
|
showModal: function(tag: string, component: any) {
|
|
if (!_portalRef) {
|
|
console.error('Calling showModal but no Portal has been rendered.');
|
|
return;
|
|
}
|
|
_portalRef._showModal(tag, component);
|
|
},
|
|
|
|
/**
|
|
* Remove a modal from the collection of modals to be rendered.
|
|
* @param tag A unique tag identifying the React component to remove.
|
|
* Must exactly match the tag previously passed to `showModal`.
|
|
*/
|
|
closeModal: function(tag: string) {
|
|
if (!_portalRef) {
|
|
console.error('Calling closeModal but no Portal has been rendered.');
|
|
return;
|
|
}
|
|
_portalRef._closeModal(tag);
|
|
},
|
|
|
|
/**
|
|
* Get an array of all the open modals, as identified by their tag string.
|
|
*/
|
|
getOpenModals: function(): Array<string> {
|
|
if (!_portalRef) {
|
|
console.error('Calling getOpenModals but no Portal has been rendered.');
|
|
return [];
|
|
}
|
|
return _portalRef._getOpenModals();
|
|
},
|
|
|
|
notifyAccessibilityService: function() {
|
|
if (!_portalRef) {
|
|
console.error('Calling closeModal but no Portal has been rendered.');
|
|
return;
|
|
}
|
|
_portalRef._notifyAccessibilityService();
|
|
},
|
|
},
|
|
|
|
getInitialState: function() {
|
|
return {modals: {}};
|
|
},
|
|
|
|
_showModal: function(tag: string, component: any) {
|
|
// We are about to open first modal, so Portal will appear.
|
|
// Let's disable accessibility for background view on Android.
|
|
if (this._getOpenModals().length === 0) {
|
|
this.props.onModalVisibilityChanged(true);
|
|
}
|
|
// This way state is chained through multiple calls to
|
|
// _showModal, _closeModal correctly.
|
|
this.setState((state) => {
|
|
var modals = state.modals;
|
|
modals[tag] = component;
|
|
return {modals};
|
|
});
|
|
},
|
|
|
|
_closeModal: function(tag: string) {
|
|
if (!this.state.modals.hasOwnProperty(tag)) {
|
|
return;
|
|
}
|
|
// We are about to close last modal, so Portal will disappear.
|
|
// Let's enable accessibility for application view on Android.
|
|
if (this._getOpenModals().length === 1) {
|
|
this.props.onModalVisibilityChanged(false);
|
|
}
|
|
// This way state is chained through multiple calls to
|
|
// _showModal, _closeModal correctly.
|
|
this.setState((state) => {
|
|
var modals = state.modals;
|
|
delete modals[tag];
|
|
return {modals};
|
|
});
|
|
},
|
|
|
|
_getOpenModals: function(): Array<string> {
|
|
return Object.keys(this.state.modals);
|
|
},
|
|
|
|
_notifyAccessibilityService: function() {
|
|
if (Platform.OS === 'android') {
|
|
// We need to send accessibility event in a new batch, as otherwise
|
|
// TextViews have no text set at the moment of populating event.
|
|
setTimeout(() => {
|
|
if (this._getOpenModals().length > 0) {
|
|
UIManager.sendAccessibilityEvent(
|
|
ReactNative.findNodeHandle(this),
|
|
UIManager.AccessibilityEventTypes.typeWindowStateChanged);
|
|
}
|
|
}, 0);
|
|
}
|
|
},
|
|
|
|
render: function() {
|
|
_portalRef = this;
|
|
if (!this.state.modals) {
|
|
return null;
|
|
}
|
|
var modals = [];
|
|
for (var tag in this.state.modals) {
|
|
modals.push(this.state.modals[tag]);
|
|
}
|
|
if (modals.length === 0) {
|
|
return null;
|
|
}
|
|
return (
|
|
<View
|
|
style={styles.modalsContainer}
|
|
importantForAccessibility="yes">
|
|
{modals}
|
|
</View>
|
|
);
|
|
}
|
|
});
|
|
|
|
var styles = StyleSheet.create({
|
|
modalsContainer: {
|
|
position: 'absolute',
|
|
left: 0,
|
|
top: 0,
|
|
right: 0,
|
|
bottom: 0,
|
|
},
|
|
});
|
|
|
|
module.exports = Portal;
|