2015-01-30 01:10:49 +00:00
|
|
|
/**
|
2015-03-23 22:07:33 +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.
|
2015-01-30 01:10:49 +00:00
|
|
|
*
|
|
|
|
* @providesModule StyleSheet
|
2015-03-25 02:34:12 +00:00
|
|
|
* @flow
|
2015-01-30 01:10:49 +00:00
|
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
|
2016-01-15 13:14:27 +00:00
|
|
|
var PixelRatio = require('PixelRatio');
|
2016-11-04 12:40:26 +00:00
|
|
|
var ReactNativePropRegistry = require('ReactNativePropRegistry');
|
2015-03-17 20:42:44 +00:00
|
|
|
var StyleSheetValidation = require('StyleSheetValidation');
|
2016-01-15 13:14:27 +00:00
|
|
|
|
|
|
|
var flatten = require('flattenStyle');
|
|
|
|
|
2016-08-15 17:37:29 +00:00
|
|
|
export type Styles = {[key: string]: Object};
|
|
|
|
export type StyleSheet<S: Styles> = {[key: $Keys<S>]: number};
|
|
|
|
|
2016-01-15 13:14:27 +00:00
|
|
|
var hairlineWidth = PixelRatio.roundToNearestPixel(0.4);
|
|
|
|
if (hairlineWidth === 0) {
|
|
|
|
hairlineWidth = 1 / PixelRatio.get();
|
|
|
|
}
|
2015-01-30 01:10:49 +00:00
|
|
|
|
2016-06-06 23:08:20 +00:00
|
|
|
const absoluteFillObject = {
|
|
|
|
position: 'absolute',
|
|
|
|
left: 0,
|
|
|
|
right: 0,
|
|
|
|
top: 0,
|
|
|
|
bottom: 0,
|
|
|
|
};
|
|
|
|
const absoluteFill = ReactNativePropRegistry.register(absoluteFillObject); // This also freezes it
|
|
|
|
|
2015-01-30 01:10:49 +00:00
|
|
|
/**
|
|
|
|
* A StyleSheet is an abstraction similar to CSS StyleSheets
|
|
|
|
*
|
|
|
|
* Create a new StyleSheet:
|
|
|
|
*
|
2015-03-17 20:42:44 +00:00
|
|
|
* ```
|
2015-01-30 01:10:49 +00:00
|
|
|
* var styles = StyleSheet.create({
|
|
|
|
* container: {
|
|
|
|
* borderRadius: 4,
|
|
|
|
* borderWidth: 0.5,
|
|
|
|
* borderColor: '#d6d7da',
|
|
|
|
* },
|
|
|
|
* title: {
|
|
|
|
* fontSize: 19,
|
|
|
|
* fontWeight: 'bold',
|
|
|
|
* },
|
|
|
|
* activeTitle: {
|
|
|
|
* color: 'red',
|
|
|
|
* },
|
2015-03-17 20:42:44 +00:00
|
|
|
* });
|
|
|
|
* ```
|
2015-01-30 01:10:49 +00:00
|
|
|
*
|
|
|
|
* Use a StyleSheet:
|
|
|
|
*
|
2015-03-17 20:42:44 +00:00
|
|
|
* ```
|
2015-01-30 01:10:49 +00:00
|
|
|
* <View style={styles.container}>
|
|
|
|
* <Text style={[styles.title, this.props.isActive && styles.activeTitle]} />
|
|
|
|
* </View>
|
2015-03-17 20:42:44 +00:00
|
|
|
* ```
|
2015-01-30 01:10:49 +00:00
|
|
|
*
|
|
|
|
* Code quality:
|
2015-03-17 20:42:44 +00:00
|
|
|
*
|
2015-01-30 01:10:49 +00:00
|
|
|
* - By moving styles away from the render function, you're making the code
|
2015-04-10 08:33:10 +00:00
|
|
|
* easier to understand.
|
2015-01-30 01:10:49 +00:00
|
|
|
* - Naming the styles is a good way to add meaning to the low level components
|
|
|
|
* in the render function.
|
|
|
|
*
|
|
|
|
* Performance:
|
2015-03-17 20:42:44 +00:00
|
|
|
*
|
2015-01-30 01:10:49 +00:00
|
|
|
* - Making a stylesheet from a style object makes it possible to refer to it
|
|
|
|
* by ID instead of creating a new style object every time.
|
|
|
|
* - It also allows to send the style only once through the bridge. All
|
|
|
|
* subsequent uses are going to refer an id (not implemented yet).
|
|
|
|
*/
|
2016-01-15 13:14:27 +00:00
|
|
|
module.exports = {
|
|
|
|
/**
|
|
|
|
* This is defined as the width of a thin line on the platform. It can be
|
|
|
|
* used as the thickness of a border or division between two elements.
|
|
|
|
* Example:
|
|
|
|
* ```
|
|
|
|
* {
|
|
|
|
* borderBottomColor: '#bbb',
|
|
|
|
* borderBottomWidth: StyleSheet.hairlineWidth
|
|
|
|
* }
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* This constant will always be a round number of pixels (so a line defined
|
|
|
|
* by it look crisp) and will try to match the standard width of a thin line
|
|
|
|
* on the underlying platform. However, you should not rely on it being a
|
|
|
|
* constant size, because on different platforms and screen densities its
|
|
|
|
* value may be calculated differently.
|
2016-11-04 12:40:26 +00:00
|
|
|
*
|
2016-08-25 14:52:50 +00:00
|
|
|
* A line with hairline width may not be visible if your simulator is downscaled.
|
2016-01-15 13:14:27 +00:00
|
|
|
*/
|
|
|
|
hairlineWidth,
|
|
|
|
|
2016-06-06 23:08:20 +00:00
|
|
|
/**
|
|
|
|
* A very common pattern is to create overlays with position absolute and zero positioning,
|
|
|
|
* so `absoluteFill` can be used for convenience and to reduce duplication of these repeated
|
|
|
|
* styles.
|
|
|
|
*/
|
|
|
|
absoluteFill,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sometimes you may want `absoluteFill` but with a couple tweaks - `absoluteFillObject` can be
|
|
|
|
* used to create a customized entry in a `StyleSheet`, e.g.:
|
|
|
|
*
|
|
|
|
* const styles = StyleSheet.create({
|
|
|
|
* wrapper: {
|
|
|
|
* ...StyleSheet.absoluteFillObject,
|
|
|
|
* top: 10,
|
|
|
|
* backgroundColor: 'transparent',
|
|
|
|
* },
|
|
|
|
* });
|
|
|
|
*/
|
|
|
|
absoluteFillObject,
|
|
|
|
|
2016-03-25 22:12:16 +00:00
|
|
|
/**
|
|
|
|
* Flattens an array of style objects, into one aggregated style object.
|
|
|
|
* Alternatively, this method can be used to lookup IDs, returned by
|
|
|
|
* StyleSheet.register.
|
|
|
|
*
|
|
|
|
* > **NOTE**: Exercise caution as abusing this can tax you in terms of
|
|
|
|
* > optimizations.
|
|
|
|
* >
|
|
|
|
* > IDs enable optimizations through the bridge and memory in general. Refering
|
|
|
|
* > to style objects directly will deprive you of these optimizations.
|
|
|
|
*
|
|
|
|
* Example:
|
|
|
|
* ```
|
|
|
|
* var styles = StyleSheet.create({
|
|
|
|
* listItem: {
|
|
|
|
* flex: 1,
|
|
|
|
* fontSize: 16,
|
|
|
|
* color: 'white'
|
|
|
|
* },
|
|
|
|
* selectedListItem: {
|
|
|
|
* color: 'green'
|
|
|
|
* }
|
|
|
|
* });
|
|
|
|
*
|
2016-03-30 17:02:58 +00:00
|
|
|
* StyleSheet.flatten([styles.listItem, styles.selectedListItem])
|
2016-03-25 22:12:16 +00:00
|
|
|
* // returns { flex: 1, fontSize: 16, color: 'green' }
|
|
|
|
* ```
|
|
|
|
* Alternative use:
|
|
|
|
* ```
|
|
|
|
* StyleSheet.flatten(styles.listItem);
|
|
|
|
* // return { flex: 1, fontSize: 16, color: 'white' }
|
|
|
|
* // Simply styles.listItem would return its ID (number)
|
|
|
|
* ```
|
|
|
|
* This method internally uses `StyleSheetRegistry.getStyleByID(style)`
|
|
|
|
* to resolve style objects represented by IDs. Thus, an array of style
|
|
|
|
* objects (instances of StyleSheet.create), are individually resolved to,
|
|
|
|
* their respective objects, merged as one and then returned. This also explains
|
|
|
|
* the alternative use.
|
|
|
|
*/
|
2016-01-15 13:14:27 +00:00
|
|
|
flatten,
|
2015-11-05 11:31:11 +00:00
|
|
|
|
2016-01-15 13:14:27 +00:00
|
|
|
/**
|
|
|
|
* Creates a StyleSheet style reference from the given object.
|
|
|
|
*/
|
2016-08-15 17:37:29 +00:00
|
|
|
create<S: Styles>(obj: S): StyleSheet<S> {
|
|
|
|
const result: StyleSheet<S> = {};
|
2015-01-30 01:10:49 +00:00
|
|
|
for (var key in obj) {
|
2015-03-17 20:42:44 +00:00
|
|
|
StyleSheetValidation.validateStyle(key, obj);
|
2016-03-24 22:12:36 +00:00
|
|
|
result[key] = ReactNativePropRegistry.register(obj[key]);
|
2015-01-30 01:10:49 +00:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2016-01-15 13:14:27 +00:00
|
|
|
};
|