2015-09-14 14:35:58 +00:00
|
|
|
/**
|
|
|
|
* Copyright (c) 2015-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 ToolbarAndroid
|
|
|
|
*/
|
|
|
|
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
var Image = require('Image');
|
2016-07-05 13:34:00 +00:00
|
|
|
var NativeMethodsMixin = require('react/lib/NativeMethodsMixin');
|
2015-09-14 14:35:58 +00:00
|
|
|
var React = require('React');
|
|
|
|
var ReactNativeViewAttributes = require('ReactNativeViewAttributes');
|
2016-07-05 13:34:00 +00:00
|
|
|
var ReactPropTypes = require('react/lib/ReactPropTypes');
|
2015-11-27 13:39:00 +00:00
|
|
|
var UIManager = require('UIManager');
|
2015-11-18 13:32:46 +00:00
|
|
|
var View = require('View');
|
2015-12-23 03:29:01 +00:00
|
|
|
var ColorPropType = require('ColorPropType');
|
2015-09-14 14:35:58 +00:00
|
|
|
|
2015-09-17 15:36:08 +00:00
|
|
|
var requireNativeComponent = require('requireNativeComponent');
|
2015-09-30 12:31:30 +00:00
|
|
|
var resolveAssetSource = require('resolveAssetSource');
|
2015-09-14 14:35:58 +00:00
|
|
|
|
2015-10-09 16:31:51 +00:00
|
|
|
var optionalImageSource = ReactPropTypes.oneOfType([
|
|
|
|
Image.propTypes.source,
|
|
|
|
// Image.propTypes.source is required but we want it to be optional, so we OR
|
|
|
|
// it with a nullable propType.
|
|
|
|
ReactPropTypes.oneOf([])
|
|
|
|
]);
|
|
|
|
|
2015-09-14 14:35:58 +00:00
|
|
|
/**
|
|
|
|
* React component that wraps the Android-only [`Toolbar` widget][0]. A Toolbar can display a logo,
|
|
|
|
* navigation icon (e.g. hamburger menu), a title & subtitle and a list of actions. The title and
|
|
|
|
* subtitle are expanded so the logo and navigation icons are displayed on the left, title and
|
|
|
|
* subtitle in the middle and the actions on the right.
|
|
|
|
*
|
|
|
|
* If the toolbar has an only child, it will be displayed between the title and actions.
|
|
|
|
*
|
2015-09-30 12:31:30 +00:00
|
|
|
* Although the Toolbar supports remote images for the logo, navigation and action icons, this
|
|
|
|
* should only be used in DEV mode where `require('./some_icon.png')` translates into a packager
|
|
|
|
* URL. In release mode you should always use a drawable resource for these icons. Using
|
|
|
|
* `require('./some_icon.png')` will do this automatically for you, so as long as you don't
|
|
|
|
* explicitly use e.g. `{uri: 'http://...'}`, you will be good.
|
|
|
|
*
|
2015-09-14 14:35:58 +00:00
|
|
|
* Example:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* render: function() {
|
|
|
|
* return (
|
|
|
|
* <ToolbarAndroid
|
2015-12-23 00:24:07 +00:00
|
|
|
* logo={require('./app_logo.png')}
|
2015-09-14 14:35:58 +00:00
|
|
|
* title="AwesomeApp"
|
2015-12-23 00:24:07 +00:00
|
|
|
* actions={[{title: 'Settings', icon: require('./icon_settings.png'), show: 'always'}]}
|
2015-09-14 14:35:58 +00:00
|
|
|
* onActionSelected={this.onActionSelected} />
|
|
|
|
* )
|
|
|
|
* },
|
|
|
|
* onActionSelected: function(position) {
|
|
|
|
* if (position === 0) { // index of 'Settings'
|
|
|
|
* showSettings();
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* [0]: https://developer.android.com/reference/android/support/v7/widget/Toolbar.html
|
|
|
|
*/
|
|
|
|
var ToolbarAndroid = React.createClass({
|
|
|
|
mixins: [NativeMethodsMixin],
|
|
|
|
|
|
|
|
propTypes: {
|
2015-11-18 13:32:46 +00:00
|
|
|
...View.propTypes,
|
2015-09-14 14:35:58 +00:00
|
|
|
/**
|
|
|
|
* Sets possible actions on the toolbar as part of the action menu. These are displayed as icons
|
|
|
|
* or text on the right side of the widget. If they don't fit they are placed in an 'overflow'
|
|
|
|
* menu.
|
|
|
|
*
|
|
|
|
* This property takes an array of objects, where each object has the following keys:
|
|
|
|
*
|
|
|
|
* * `title`: **required**, the title of this action
|
2015-12-23 00:24:07 +00:00
|
|
|
* * `icon`: the icon for this action, e.g. `require('./some_icon.png')`
|
2015-09-14 14:35:58 +00:00
|
|
|
* * `show`: when to show this action as an icon or hide it in the overflow menu: `always`,
|
|
|
|
* `ifRoom` or `never`
|
|
|
|
* * `showWithText`: boolean, whether to show text alongside the icon or not
|
|
|
|
*/
|
|
|
|
actions: ReactPropTypes.arrayOf(ReactPropTypes.shape({
|
|
|
|
title: ReactPropTypes.string.isRequired,
|
2015-10-09 16:31:51 +00:00
|
|
|
icon: optionalImageSource,
|
2015-09-14 14:35:58 +00:00
|
|
|
show: ReactPropTypes.oneOf(['always', 'ifRoom', 'never']),
|
|
|
|
showWithText: ReactPropTypes.bool
|
|
|
|
})),
|
|
|
|
/**
|
|
|
|
* Sets the toolbar logo.
|
|
|
|
*/
|
2015-10-09 16:31:51 +00:00
|
|
|
logo: optionalImageSource,
|
2015-09-14 14:35:58 +00:00
|
|
|
/**
|
|
|
|
* Sets the navigation icon.
|
|
|
|
*/
|
2015-10-09 16:31:51 +00:00
|
|
|
navIcon: optionalImageSource,
|
2015-09-14 14:35:58 +00:00
|
|
|
/**
|
2015-12-15 17:08:39 +00:00
|
|
|
* Callback that is called when an action is selected. The only argument that is passed to the
|
2015-09-14 14:35:58 +00:00
|
|
|
* callback is the position of the action in the actions array.
|
|
|
|
*/
|
|
|
|
onActionSelected: ReactPropTypes.func,
|
|
|
|
/**
|
|
|
|
* Callback called when the icon is selected.
|
|
|
|
*/
|
|
|
|
onIconClicked: ReactPropTypes.func,
|
2015-10-22 11:40:21 +00:00
|
|
|
/**
|
|
|
|
* Sets the overflow icon.
|
|
|
|
*/
|
|
|
|
overflowIcon: optionalImageSource,
|
2015-09-14 14:35:58 +00:00
|
|
|
/**
|
|
|
|
* Sets the toolbar subtitle.
|
|
|
|
*/
|
|
|
|
subtitle: ReactPropTypes.string,
|
|
|
|
/**
|
|
|
|
* Sets the toolbar subtitle color.
|
|
|
|
*/
|
2015-12-23 03:29:01 +00:00
|
|
|
subtitleColor: ColorPropType,
|
2015-09-14 14:35:58 +00:00
|
|
|
/**
|
|
|
|
* Sets the toolbar title.
|
|
|
|
*/
|
|
|
|
title: ReactPropTypes.string,
|
|
|
|
/**
|
|
|
|
* Sets the toolbar title color.
|
|
|
|
*/
|
2015-12-23 03:29:01 +00:00
|
|
|
titleColor: ColorPropType,
|
2015-12-23 00:45:14 +00:00
|
|
|
/**
|
|
|
|
* Sets the content inset for the toolbar starting edge.
|
|
|
|
*
|
|
|
|
* The content inset affects the valid area for Toolbar content other than
|
|
|
|
* the navigation button and menu. Insets define the minimum margin for
|
|
|
|
* these components and can be used to effectively align Toolbar content
|
|
|
|
* along well-known gridlines.
|
|
|
|
*/
|
|
|
|
contentInsetStart: ReactPropTypes.number,
|
|
|
|
/**
|
|
|
|
* Sets the content inset for the toolbar ending edge.
|
|
|
|
*
|
|
|
|
* The content inset affects the valid area for Toolbar content other than
|
|
|
|
* the navigation button and menu. Insets define the minimum margin for
|
|
|
|
* these components and can be used to effectively align Toolbar content
|
|
|
|
* along well-known gridlines.
|
|
|
|
*/
|
|
|
|
contentInsetEnd: ReactPropTypes.number,
|
2015-12-03 12:38:07 +00:00
|
|
|
/**
|
|
|
|
* Used to set the toolbar direction to RTL.
|
|
|
|
* In addition to this property you need to add
|
|
|
|
*
|
|
|
|
* android:supportsRtl="true"
|
|
|
|
*
|
|
|
|
* to your application AndroidManifest.xml and then call
|
|
|
|
* `setLayoutDirection(LayoutDirection.RTL)` in your MainActivity
|
|
|
|
* `onCreate` method.
|
|
|
|
*/
|
|
|
|
rtl: ReactPropTypes.bool,
|
2015-09-14 14:35:58 +00:00
|
|
|
/**
|
|
|
|
* Used to locate this view in end-to-end tests.
|
|
|
|
*/
|
|
|
|
testID: ReactPropTypes.string,
|
|
|
|
},
|
|
|
|
|
|
|
|
render: function() {
|
|
|
|
var nativeProps = {
|
|
|
|
...this.props,
|
|
|
|
};
|
|
|
|
if (this.props.logo) {
|
2015-09-30 12:31:30 +00:00
|
|
|
nativeProps.logo = resolveAssetSource(this.props.logo);
|
2015-09-14 14:35:58 +00:00
|
|
|
}
|
|
|
|
if (this.props.navIcon) {
|
2015-09-30 12:31:30 +00:00
|
|
|
nativeProps.navIcon = resolveAssetSource(this.props.navIcon);
|
2015-09-14 14:35:58 +00:00
|
|
|
}
|
2015-10-22 11:40:21 +00:00
|
|
|
if (this.props.overflowIcon) {
|
|
|
|
nativeProps.overflowIcon = resolveAssetSource(this.props.overflowIcon);
|
|
|
|
}
|
2015-09-14 14:35:58 +00:00
|
|
|
if (this.props.actions) {
|
2015-11-18 13:32:46 +00:00
|
|
|
var nativeActions = [];
|
2015-09-14 14:35:58 +00:00
|
|
|
for (var i = 0; i < this.props.actions.length; i++) {
|
|
|
|
var action = {
|
|
|
|
...this.props.actions[i],
|
|
|
|
};
|
|
|
|
if (action.icon) {
|
2015-09-30 12:31:30 +00:00
|
|
|
action.icon = resolveAssetSource(action.icon);
|
|
|
|
}
|
|
|
|
if (action.show) {
|
2015-11-27 13:39:00 +00:00
|
|
|
action.show = UIManager.ToolbarAndroid.Constants.ShowAsAction[action.show];
|
2015-09-14 14:35:58 +00:00
|
|
|
}
|
2015-11-18 13:32:46 +00:00
|
|
|
nativeActions.push(action);
|
2015-09-14 14:35:58 +00:00
|
|
|
}
|
2015-11-18 13:32:46 +00:00
|
|
|
nativeProps.nativeActions = nativeActions;
|
2015-09-14 14:35:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return <NativeToolbar onSelect={this._onSelect} {...nativeProps} />;
|
|
|
|
},
|
|
|
|
|
|
|
|
_onSelect: function(event) {
|
|
|
|
var position = event.nativeEvent.position;
|
|
|
|
if (position === -1) {
|
|
|
|
this.props.onIconClicked && this.props.onIconClicked();
|
|
|
|
} else {
|
|
|
|
this.props.onActionSelected && this.props.onActionSelected(position);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
var toolbarAttributes = {
|
|
|
|
...ReactNativeViewAttributes.UIView,
|
|
|
|
actions: true,
|
|
|
|
logo: true,
|
|
|
|
navIcon: true,
|
2015-10-22 11:40:21 +00:00
|
|
|
overflowIcon: true,
|
2015-12-03 12:38:07 +00:00
|
|
|
rtl: true,
|
2015-09-14 14:35:58 +00:00
|
|
|
subtitle: true,
|
|
|
|
subtitleColor: true,
|
|
|
|
title: true,
|
|
|
|
titleColor: true,
|
|
|
|
};
|
|
|
|
|
2015-11-18 13:32:46 +00:00
|
|
|
var NativeToolbar = requireNativeComponent('ToolbarAndroid', ToolbarAndroid, {
|
|
|
|
nativeOnly: {
|
|
|
|
nativeActions: true,
|
|
|
|
}
|
|
|
|
});
|
2015-09-14 14:35:58 +00:00
|
|
|
|
|
|
|
module.exports = ToolbarAndroid;
|