2015-02-19 20:10:52 -08:00
|
|
|
/**
|
2015-03-23 13:35:08 -07: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.
|
2015-02-19 20:10:52 -08:00
|
|
|
*
|
|
|
|
* @providesModule View
|
2015-04-03 11:03:41 -07:00
|
|
|
* @flow
|
2015-02-19 20:10:52 -08:00
|
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
var NativeMethodsMixin = require('NativeMethodsMixin');
|
|
|
|
var PropTypes = require('ReactPropTypes');
|
2015-04-21 21:07:17 -07:00
|
|
|
var RCTUIManager = require('NativeModules').UIManager;
|
2015-03-31 16:12:55 -07:00
|
|
|
var React = require('React');
|
2015-05-08 09:45:43 -07:00
|
|
|
var ReactNativeStyleAttributes = require('ReactNativeStyleAttributes');
|
|
|
|
var ReactNativeViewAttributes = require('ReactNativeViewAttributes');
|
2015-02-19 20:10:52 -08:00
|
|
|
var StyleSheetPropType = require('StyleSheetPropType');
|
|
|
|
var ViewStylePropTypes = require('ViewStylePropTypes');
|
|
|
|
|
2015-05-08 09:45:43 -07:00
|
|
|
var createReactNativeComponentClass = require('createReactNativeComponentClass');
|
2015-03-09 09:28:51 -07:00
|
|
|
|
|
|
|
var stylePropType = StyleSheetPropType(ViewStylePropTypes);
|
|
|
|
|
2015-05-18 07:32:21 -07:00
|
|
|
var AccessibilityTraits = [
|
|
|
|
'none',
|
|
|
|
'button',
|
|
|
|
'link',
|
|
|
|
'header',
|
|
|
|
'search',
|
|
|
|
'image',
|
|
|
|
'selected',
|
|
|
|
'plays',
|
|
|
|
'key',
|
|
|
|
'text',
|
|
|
|
'summary',
|
|
|
|
'disabled',
|
|
|
|
'frequentUpdates',
|
|
|
|
'startsMedia',
|
|
|
|
'adjustable',
|
|
|
|
'allowsDirectInteraction',
|
|
|
|
'pageTurn',
|
|
|
|
];
|
|
|
|
|
2015-07-30 01:28:14 -07:00
|
|
|
|
2015-07-14 01:59:28 -07:00
|
|
|
// <<<<< WARNING >>>>>
|
|
|
|
// If adding any properties to View that could change the way layout-only status
|
|
|
|
// works on iOS, make sure to update ReactNativeViewAttributes.js and
|
|
|
|
// RCTShadowView.m (in the -[RCTShadowView isLayoutOnly] method).
|
|
|
|
// <<<<< WARNING >>>>>
|
|
|
|
|
2015-02-19 20:10:52 -08:00
|
|
|
/**
|
2015-03-09 09:28:51 -07:00
|
|
|
* The most fundamental component for building UI, `View` is a
|
2015-02-19 20:10:52 -08:00
|
|
|
* container that supports layout with flexbox, style, some touch handling, and
|
|
|
|
* accessibility controls, and is designed to be nested inside other views and
|
|
|
|
* to have 0 to many children of any type. `View` maps directly to the native
|
2015-04-27 13:55:01 -07:00
|
|
|
* view equivalent on whatever platform React is running on, whether that is a
|
2015-02-19 20:10:52 -08:00
|
|
|
* `UIView`, `<div>`, `android.view`, etc. This example creates a `View` that
|
|
|
|
* wraps two colored boxes and custom component in a row with padding.
|
|
|
|
*
|
2015-03-09 09:28:51 -07:00
|
|
|
* ```
|
|
|
|
* <View style={{flexDirection: 'row', height: 100, padding: 20}}>
|
|
|
|
* <View style={{backgroundColor: 'blue', flex: 0.3}} />
|
|
|
|
* <View style={{backgroundColor: 'red', flex: 0.5}} />
|
|
|
|
* <MyCustomComponent {...customProps} />
|
|
|
|
* </View>
|
|
|
|
* ```
|
2015-02-19 20:10:52 -08:00
|
|
|
*
|
|
|
|
* `View`s are designed to be used with `StyleSheet`s for clarity and
|
2015-03-25 16:00:21 -07:00
|
|
|
* performance, although inline styles are also supported.
|
2015-02-19 20:10:52 -08:00
|
|
|
*/
|
2015-03-31 16:12:55 -07:00
|
|
|
var View = React.createClass({
|
|
|
|
mixins: [NativeMethodsMixin],
|
2015-02-19 20:10:52 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* `NativeMethodsMixin` will look for this when invoking `setNativeProps`. We
|
|
|
|
* make `this` look like an actual native component class.
|
|
|
|
*/
|
|
|
|
viewConfig: {
|
|
|
|
uiViewClassName: 'RCTView',
|
2015-05-08 09:45:43 -07:00
|
|
|
validAttributes: ReactNativeViewAttributes.RCTView
|
2015-02-19 20:10:52 -08:00
|
|
|
},
|
|
|
|
|
|
|
|
propTypes: {
|
|
|
|
/**
|
2015-03-25 16:00:21 -07:00
|
|
|
* When true, indicates that the view is an accessibility element. By default,
|
|
|
|
* all the touchable elements are accessible.
|
2015-02-19 20:10:52 -08:00
|
|
|
*/
|
|
|
|
accessible: PropTypes.bool,
|
|
|
|
|
2015-03-25 16:00:21 -07:00
|
|
|
/**
|
|
|
|
* Overrides the text that's read by the screen reader when the user interacts
|
|
|
|
* with the element. By default, the label is constructed by traversing all the
|
|
|
|
* children and accumulating all the Text nodes separated by space.
|
|
|
|
*/
|
|
|
|
accessibilityLabel: PropTypes.string,
|
|
|
|
|
2015-07-29 09:12:16 -07:00
|
|
|
/**
|
|
|
|
* Indicates to accessibility services to treat UI component like a
|
|
|
|
* native one. Works for Android only.
|
|
|
|
*/
|
|
|
|
accessibilityComponentType: PropTypes.oneOf([
|
|
|
|
'none',
|
|
|
|
'button',
|
|
|
|
]),
|
|
|
|
|
2015-07-30 01:28:14 -07:00
|
|
|
/**
|
|
|
|
* Indicates to accessibility services whether the user should be notified
|
|
|
|
* when this view changes. Works for Android API >= 19 only.
|
|
|
|
* See http://developer.android.com/reference/android/view/View.html#attr_android:accessibilityLiveRegion
|
|
|
|
* for references.
|
|
|
|
*/
|
|
|
|
accessibilityLiveRegion: PropTypes.oneOf([
|
|
|
|
'none',
|
|
|
|
'polite',
|
|
|
|
'assertive',
|
|
|
|
]),
|
|
|
|
|
2015-05-18 07:32:21 -07:00
|
|
|
/**
|
|
|
|
* Provides additional traits to screen reader. By default no traits are
|
|
|
|
* provided unless specified otherwise in element
|
|
|
|
*/
|
|
|
|
accessibilityTraits: PropTypes.oneOfType([
|
|
|
|
PropTypes.oneOf(AccessibilityTraits),
|
|
|
|
PropTypes.arrayOf(PropTypes.oneOf(AccessibilityTraits)),
|
|
|
|
]),
|
2015-05-19 06:21:52 -07:00
|
|
|
|
2015-05-20 08:33:16 -07:00
|
|
|
/**
|
|
|
|
* When `accessible` is true, the system will try to invoke this function
|
|
|
|
* when the user performs accessibility tap gesture.
|
|
|
|
*/
|
|
|
|
onAcccessibilityTap: PropTypes.func,
|
|
|
|
|
2015-05-19 06:21:52 -07:00
|
|
|
/**
|
|
|
|
* When `accessible` is true, the system will invoke this function when the
|
|
|
|
* user performs the magic tap gesture.
|
|
|
|
*/
|
|
|
|
onMagicTap: PropTypes.func,
|
|
|
|
|
2015-02-19 20:10:52 -08:00
|
|
|
/**
|
2015-08-07 10:32:56 -01:00
|
|
|
* Used to locate this view in end-to-end tests. NB: disables the 'layout-only
|
|
|
|
* view removal' optimization for this view!
|
2015-02-19 20:10:52 -08:00
|
|
|
*/
|
|
|
|
testID: PropTypes.string,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* For most touch interactions, you'll simply want to wrap your component in
|
2015-03-25 16:00:21 -07:00
|
|
|
* `TouchableHighlight` or `TouchableOpacity`. Check out `Touchable.js`,
|
|
|
|
* `ScrollResponder.js` and `ResponderEventPlugin.js` for more discussion.
|
2015-02-19 20:10:52 -08:00
|
|
|
*/
|
2015-03-25 16:00:21 -07:00
|
|
|
onMoveShouldSetResponder: PropTypes.func,
|
2015-02-19 20:10:52 -08:00
|
|
|
onResponderGrant: PropTypes.func,
|
|
|
|
onResponderMove: PropTypes.func,
|
2015-03-25 16:00:21 -07:00
|
|
|
onResponderReject: PropTypes.func,
|
2015-02-19 20:10:52 -08:00
|
|
|
onResponderRelease: PropTypes.func,
|
|
|
|
onResponderTerminate: PropTypes.func,
|
|
|
|
onResponderTerminationRequest: PropTypes.func,
|
|
|
|
onStartShouldSetResponder: PropTypes.func,
|
|
|
|
onStartShouldSetResponderCapture: PropTypes.func,
|
|
|
|
|
[ReactNative] Introduce onLayout events
Summary:
Simply add an `onLayout` callback to a native view component, and the callback
will be invoked with the current layout information when the view is mounted and
whenever the layout changes.
The only limitation is that scroll position and other stuff the layout system
isn't aware of is not taken into account. This is because onLayout events
wouldn't be triggered for these changes and if they are desired they should be
tracked separately (e.g. with `onScroll`) and combined.
Also fixes some bugs with LayoutAnimation callbacks.
@public
Test Plan:
- Run new LayoutEventsExample in UIExplorer and see it work correctly.
- New integration test passes internally (IntegrationTest project seems busted).
- New jest test case passes.
{F22318433}
```
2015-05-06 15:45:05.848 [info][tid:com.facebook.React.JavaScript] "Running application "UIExplorerApp" with appParams: {"rootTag":1,"initialProps":{}}. __DEV__ === true, development-level warning are ON, performance optimizations are OFF"
2015-05-06 15:45:05.881 [info][tid:com.facebook.React.JavaScript] "received text layout event
", {"target":27,"layout":{"y":123,"x":12.5,"width":140.5,"height":18}}
2015-05-06 15:45:05.882 [info][tid:com.facebook.React.JavaScript] "received image layout event
", {"target":23,"layout":{"y":12.5,"x":122,"width":50,"height":50}}
2015-05-06 15:45:05.883 [info][tid:com.facebook.React.JavaScript] "received view layout event
", {"target":22,"layout":{"y":70.5,"x":20,"width":294,"height":204}}
2015-05-06 15:45:05.897 [info][tid:com.facebook.React.JavaScript] "received text layout event
", {"target":27,"layout":{"y":206.5,"x":12.5,"width":140.5,"height":18}}
2015-05-06 15:45:05.897 [info][tid:com.facebook.React.JavaScript] "received view layout event
", {"target":22,"layout":{"y":70.5,"x":20,"width":294,"height":287.5}}
2015-05-06 15:45:09.847 [info][tid:com.facebook.React.JavaScript] "layout animation done."
2015-05-06 15:45:09.847 [info][tid:com.facebook.React.JavaScript] "received image layout event
", {"target":23,"layout":{"y":12.5,"x":82,"width":50,"height":50}}
2015-05-06 15:45:09.848 [info][tid:com.facebook.React.JavaScript] "received view layout event
", {"target":22,"layout":{"y":110.5,"x":60,"width":214,"height":287.5}}
2015-05-06 15:45:09.862 [info][tid:com.facebook.React.JavaScript] "received text layout event
", {"target":27,"layout":{"y":206.5,"x":12.5,"width":120,"height":68}}
2015-05-06 15:45:09.863 [info][tid:com.facebook.React.JavaScript] "received image layout event
", {"target":23,"layout":{"y":12.5,"x":55,"width":50,"height":50}}
2015-05-06 15:45:09.863 [info][tid:com.facebook.React.JavaScript] "received view layout event
", {"target":22,"layout":{"y":128,"x":60,"width":160,"height":337.5}}
```
2015-05-07 12:11:02 -07:00
|
|
|
/**
|
2015-05-15 18:05:49 -07:00
|
|
|
* Invoked on mount and layout changes with
|
|
|
|
*
|
|
|
|
* {nativeEvent: { layout: {x, y, width, height}}}.
|
[ReactNative] Introduce onLayout events
Summary:
Simply add an `onLayout` callback to a native view component, and the callback
will be invoked with the current layout information when the view is mounted and
whenever the layout changes.
The only limitation is that scroll position and other stuff the layout system
isn't aware of is not taken into account. This is because onLayout events
wouldn't be triggered for these changes and if they are desired they should be
tracked separately (e.g. with `onScroll`) and combined.
Also fixes some bugs with LayoutAnimation callbacks.
@public
Test Plan:
- Run new LayoutEventsExample in UIExplorer and see it work correctly.
- New integration test passes internally (IntegrationTest project seems busted).
- New jest test case passes.
{F22318433}
```
2015-05-06 15:45:05.848 [info][tid:com.facebook.React.JavaScript] "Running application "UIExplorerApp" with appParams: {"rootTag":1,"initialProps":{}}. __DEV__ === true, development-level warning are ON, performance optimizations are OFF"
2015-05-06 15:45:05.881 [info][tid:com.facebook.React.JavaScript] "received text layout event
", {"target":27,"layout":{"y":123,"x":12.5,"width":140.5,"height":18}}
2015-05-06 15:45:05.882 [info][tid:com.facebook.React.JavaScript] "received image layout event
", {"target":23,"layout":{"y":12.5,"x":122,"width":50,"height":50}}
2015-05-06 15:45:05.883 [info][tid:com.facebook.React.JavaScript] "received view layout event
", {"target":22,"layout":{"y":70.5,"x":20,"width":294,"height":204}}
2015-05-06 15:45:05.897 [info][tid:com.facebook.React.JavaScript] "received text layout event
", {"target":27,"layout":{"y":206.5,"x":12.5,"width":140.5,"height":18}}
2015-05-06 15:45:05.897 [info][tid:com.facebook.React.JavaScript] "received view layout event
", {"target":22,"layout":{"y":70.5,"x":20,"width":294,"height":287.5}}
2015-05-06 15:45:09.847 [info][tid:com.facebook.React.JavaScript] "layout animation done."
2015-05-06 15:45:09.847 [info][tid:com.facebook.React.JavaScript] "received image layout event
", {"target":23,"layout":{"y":12.5,"x":82,"width":50,"height":50}}
2015-05-06 15:45:09.848 [info][tid:com.facebook.React.JavaScript] "received view layout event
", {"target":22,"layout":{"y":110.5,"x":60,"width":214,"height":287.5}}
2015-05-06 15:45:09.862 [info][tid:com.facebook.React.JavaScript] "received text layout event
", {"target":27,"layout":{"y":206.5,"x":12.5,"width":120,"height":68}}
2015-05-06 15:45:09.863 [info][tid:com.facebook.React.JavaScript] "received image layout event
", {"target":23,"layout":{"y":12.5,"x":55,"width":50,"height":50}}
2015-05-06 15:45:09.863 [info][tid:com.facebook.React.JavaScript] "received view layout event
", {"target":22,"layout":{"y":128,"x":60,"width":160,"height":337.5}}
```
2015-05-07 12:11:02 -07:00
|
|
|
*/
|
|
|
|
onLayout: PropTypes.func,
|
|
|
|
|
2015-02-19 20:10:52 -08:00
|
|
|
/**
|
|
|
|
* In the absence of `auto` property, `none` is much like `CSS`'s `none`
|
2015-03-04 14:04:52 -08:00
|
|
|
* value. `box-none` is as if you had applied the `CSS` class:
|
2015-02-19 20:10:52 -08:00
|
|
|
*
|
2015-03-25 16:00:21 -07:00
|
|
|
* ```
|
|
|
|
* .box-none {
|
|
|
|
* pointer-events: none;
|
|
|
|
* }
|
|
|
|
* .box-none * {
|
|
|
|
* pointer-events: all;
|
|
|
|
* }
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* `box-only` is the equivalent of
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* .box-only {
|
|
|
|
* pointer-events: all;
|
|
|
|
* }
|
|
|
|
* .box-only * {
|
|
|
|
* pointer-events: none;
|
|
|
|
* }
|
|
|
|
* ```
|
2015-02-19 20:10:52 -08:00
|
|
|
*
|
|
|
|
* But since `pointerEvents` does not affect layout/appearance, and we are
|
|
|
|
* already deviating from the spec by adding additional modes, we opt to not
|
|
|
|
* include `pointerEvents` on `style`. On some platforms, we would need to
|
|
|
|
* implement it as a `className` anyways. Using `style` or not is an
|
|
|
|
* implementation detail of the platform.
|
|
|
|
*/
|
|
|
|
pointerEvents: PropTypes.oneOf([
|
2015-03-04 14:04:52 -08:00
|
|
|
'box-none',
|
|
|
|
'none',
|
|
|
|
'box-only',
|
|
|
|
'auto',
|
2015-02-19 20:10:52 -08:00
|
|
|
]),
|
|
|
|
style: stylePropType,
|
|
|
|
|
|
|
|
/**
|
2015-03-17 03:08:46 -07:00
|
|
|
* This is a special performance property exposed by RCTView and is useful
|
2015-02-19 20:10:52 -08:00
|
|
|
* for scrolling content when there are many subviews, most of which are
|
|
|
|
* offscreen. For this property to be effective, it must be applied to a
|
|
|
|
* view that contains many subviews that extend outside its bound. The
|
|
|
|
* subviews must also have overflow: hidden, as should the containing view
|
|
|
|
* (or one of its superviews).
|
|
|
|
*/
|
|
|
|
removeClippedSubviews: PropTypes.bool,
|
2015-04-20 02:47:20 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether this view should render itself (and all of its children) into a
|
|
|
|
* single hardware texture on the GPU.
|
|
|
|
*
|
|
|
|
* On Android, this is useful for animations and interactions that only
|
|
|
|
* modify opacity, rotation, translation, and/or scale: in those cases, the
|
|
|
|
* view doesn't have to be redrawn and display lists don't need to be
|
|
|
|
* re-executed. The texture can just be re-used and re-composited with
|
|
|
|
* different parameters. The downside is that this can use up limited video
|
|
|
|
* memory, so this prop should be set back to false at the end of the
|
|
|
|
* interaction/animation.
|
2015-08-04 22:37:38 -01:00
|
|
|
* @platform android
|
2015-04-20 02:47:20 -07:00
|
|
|
*/
|
|
|
|
renderToHardwareTextureAndroid: PropTypes.bool,
|
2015-08-04 22:37:38 -01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether this view should be rendered as a bitmap before compositing.
|
|
|
|
*
|
|
|
|
* On iOS, this is useful for animations and interactions that do not
|
|
|
|
* modify this component's dimensions nor its children; for example, when
|
|
|
|
* translating the position of a static view, rasterization allows the
|
|
|
|
* renderer to reuse a cached bitmap of a static view and quickly composite
|
|
|
|
* it during each frame.
|
|
|
|
*
|
|
|
|
* Rasterization incurs an off-screen drawing pass and the bitmap consumes
|
|
|
|
* memory. Test and measure when using this property.
|
|
|
|
* @platform ios
|
|
|
|
*/
|
|
|
|
shouldRasterizeIOS: PropTypes.bool,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @platform android
|
|
|
|
*/
|
2015-07-31 22:03:05 -07:00
|
|
|
collapsable: PropTypes.bool,
|
2015-02-19 20:10:52 -08:00
|
|
|
},
|
|
|
|
|
|
|
|
render: function() {
|
2015-03-31 16:12:55 -07:00
|
|
|
return <RCTView {...this.props} />;
|
2015-02-19 20:10:52 -08:00
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2015-05-08 09:45:43 -07:00
|
|
|
var RCTView = createReactNativeComponentClass({
|
|
|
|
validAttributes: ReactNativeViewAttributes.RCTView,
|
2015-02-19 20:10:52 -08:00
|
|
|
uiViewClassName: 'RCTView',
|
|
|
|
});
|
2015-03-17 03:08:46 -07:00
|
|
|
RCTView.propTypes = View.propTypes;
|
2015-04-21 21:07:17 -07:00
|
|
|
if (__DEV__) {
|
|
|
|
var viewConfig = RCTUIManager.viewConfigs && RCTUIManager.viewConfigs.RCTView || {};
|
|
|
|
for (var prop in viewConfig.nativeProps) {
|
|
|
|
var viewAny: any = View; // Appease flow
|
2015-05-08 09:45:43 -07:00
|
|
|
if (!viewAny.propTypes[prop] && !ReactNativeStyleAttributes[prop]) {
|
2015-04-21 21:07:17 -07:00
|
|
|
throw new Error(
|
|
|
|
'View is missing propType for native prop `' + prop + '`'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-02-19 20:10:52 -08:00
|
|
|
|
2015-03-17 03:08:46 -07:00
|
|
|
var ViewToExport = RCTView;
|
2015-02-19 20:10:52 -08:00
|
|
|
if (__DEV__) {
|
|
|
|
ViewToExport = View;
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = ViewToExport;
|