图片

一个 react 的组件用以显示不同类型的图片,包括网络图片,静态资源,临时的本地图片,还有本地磁盘的图片,比如手机照片。

举例:

  1. renderImages: function() {
  2. return (
  3. <View>
  4. <Image
  5. style={styles.icon}
  6. source={require('image!myIcon')}
  7. />
  8. <Image
  9. style={styles.logo}
  10. source={{uri: 'http://facebook.github.io/react/img/logo_og.png'}}
  11. />
  12. </View>
  13. );
  14. },

支撑工具

onLayout 函数
在进行装载和布局改变的时候使用{nativeEvent: {layout: {x, y, width, height}}}调用。

resizeMode 枚举 (‘cover’, ‘contain’, ‘stretch’)
当帧与原始图像尺寸不匹配时用于确定如何调整图像的大小。

source {uri: string},编号
uri 是一个代表图片资源标识符的字符串,它可以是 http 地址、 本地文件路径或静态图像资源的名称 (它被包含在 require('image!name') 函数中) 。

style 样式
Flexbox……
Transforms
resizeMode Object.keys(ImageResizeMode)
backgroundColor 字符串
borderColor 字符串
borderWidth 数字
borderRadius 数字
overflow 枚举(‘visible’, ‘hidden’)
tintColor 字符串
opacity 数字

testID 字符串
一个在 UI 自动测试脚本中使用此元素的唯一标识符。

ios accessibilityLabel 字符串

在用户与图像交互时,该文本会由屏幕阅读器读取。

ios accessible 布尔值
当为 true 的时候,指示图像是可访问的元素。

ios capInsets {top: number, left: number, bottom: number, right: number}
当图像的大小被重新调整时,由 capInsets 指定的角落的大小将保持在一个固定的值,但中心内容和图像的边界将被拉伸。这用于创建可调整大小的圆形按钮、 阴影和其他可调整大小的资源。更多关于苹果的文档请点击此处。

ios defaultSource {uri: string}
在下载最终图像并且网络断开的时候用来显示的静态图像。

ios onError 函数
在加载错误的时候使用 {nativeEvent: {error}} 进行调用。

ios onLoadEndr 函数
当完全加载成功时进行调用。

ios onLoadEnd 函数
不管加载成功还是失败都会调用。

ios onLoadStart 函数
加载成功的时候调用。

ios onProgress 函数
在下载进程中使用 {nativeEvent: {loaded, total}} 进行调用。

描述

静态资源

在项目的进程中,添加并且移除和处理那些在应用程序不是经常使用的图片是很常见的情况。为了处理这种情况,我们需要找到一个方法来静态地定位那些被用在应用程序里的图片。因此,我们使用了一个标记器。唯一允许的指向 bundle 里的图片的方法就是在源文件中遍历地搜索 require('image!name-of-the-asset')

  1. // GOOD
  2. <Image source={require('image!my-icon')} />
  3. // BAD
  4. var icon = this.props.active ? 'my-icon-active' : 'my-icon-inactive';
  5. <Image source={require('image!' + icon)} />
  6. // GOOD
  7. var icon = this.props.active ? require('image!my-icon-active') : require('image!my-icon-inactive');
  8. <Image source={icon} />

当主要的代码执行到这里,你就可以做一些有趣的事情,比如自动将那些用于应用程序的 assets 打包。注意,这些代码不是强制实施的,但不代表将来也不会。

使用 Images.xcassets 将静态资源添加到你的 iOS 应用程序中

图像 - 图1

NOTE: 生成应用程序所需的新资源

无论在什么时候,您想把新的资源添加到 Images.xcassets 中,您都需要在使用它之前通过 Xcode 来重新构建您的应用程序 — — 仅在模拟器内重新加载它是不够的。

这一进程正在被改进,不久就会提供更好的工作流程。

将静态资源添加到您的 Android 应用程序中

将您的图像作为位图画板添加到 android 项目中(<yourapp>/android/app/src/main/res)。 为了给您的 assets 文件提供不同的分辨率,使用 配置限定符进行检查。 通常情况下,您将想要把您的 assets 文件放在下列目录 (如果它们不存在,那么在 res 下创建它们):

  • drawable-mdpi (1x)
  • drawable-hdpi (1.5x)
  • drawable-xhdpi (2x)
  • drawable-xxhdpi (3x)

如果您的 asset 文件丢失了一种分辨率,那么 Android 将采取下一个最好的分辨率并且为您调整它的大小。

NOTE: 生成应用程序所需的新资源

无论在什么时候您把新的资源添加到您的画板中您都需要在使用它之前通过运行 react-native run-android 重新构建您的应用程序 - 仅重新加载 JS 是不够的。

这一进程正在被改进,不久就会提供更好的工作流程。

网络资源

在您进行编译的时候,许多您的应用程序中需要展示的图片都不能使用,或者你会想要通过加载一些动态图片来保持二进制大小在较低的状态。不像静态资源那样,您将需要手动指定图像的尺寸

  1. // GOOD
  2. <Image source={{uri: 'https://facebook.github.io/react/img/logo_og.png'}}
  3. style={{width: 400, height: 400}} />
  4. // BAD
  5. <Image source={{uri: 'https://facebook.github.io/react/img/logo_og.png'}} />

本地文件系统资源

请在 CameraRoll 中查看使用 Images.xcassets 之外的本地资源示例 。

最好的相片册图片

iOS 的相片册可以让你将同一张图片保存为不同的尺寸,对于选择那张接近你想要尺寸的图片来说,这很重要。你不会想用一张 3264x2448 分辨率的图片作为资源来显示一个 200x200 的缩略图。如果确实有符合你要求的尺寸, React Native 会自动选择它,否则它会使用第一张比特定尺寸大 50% 的图片来避免重新定义尺寸时带来的模糊失真。这些工作 React Native 自动帮你完成了,所以你不必再自己编写乏味和容易出错的代码。

为什么不自动调整所有事物?

在浏览器中 如果你不给一个图像规定大小,那么浏览器会呈现一个 0x0 的元素、 随后下载图像,然后再以正确的尺寸呈现图像。这种行为最大的问题是,您的 UI 会在正在加载的图像四周跳动,这会造成一个非常糟糕的用户体验。

React Native 中故意不实施该行为。它的目的是让开发人员可以提前知道远程图像的尺寸 (或图形比例),我们认为这样做的话可以实现更好的用户体验。通过使用 require('image!x') 语法从应用程序包中加载的静态图片可以自动的调整大小,因为它们的尺寸在安装时立即可用。

例如,上述使用 ‘require(‘image!logo’)’ 屏幕截图的结果:

  1. {"__packager_asset":true,"isStatic":true,"path":"/Users/react/HelloWorld/iOS/Images.xcassets/react.imageset/logo.png","uri":"logo","width":591,"height":573}

Source 是一个对象类型

在 React Native 中,一个有趣的决定是 src 特性将会被命名为 source,并且不作为一个字符串而是一个 uri 特性的对象类型。

  1. <Image source={{uri: 'something.jpg'}} />

站在底层来看,这样做的原因是它允许将元数据依附到这个对象中。举个例子,你正在使用 require('image!icon'),我们将添加 isStatic 作为一个 flag 来标识本地文件(不要依赖这例子,将来这可能会改变!)。这在将来同时也会成为可能,比如我们可能会支持子画面,并用它来取代输出 {uri: ...},我们可以输出 {uri: ..., crop: {left: 10, top: 50, width: 20, height: 40}} 同时支持在所有已经存在的网站中透明地显示子画面。

在用户角度上,这会让你用有用的特性比如图片的几何尺寸来注释对象类型,从而计算出将要显示出来的尺寸。尽情地使用这种数据类型来储存你的图片吧。

背景图片叠加

一个对于 web 开发者们很常见的需求是 background-image。这种情况下,创建一个简单的 <Image> 组件然后将它作为子 layer 添加到你想要添加的 layer 上面。

  1. return (
  2. <Image source={...}>
  3. <Text>Inside</Text>
  4. </Image>
  5. );

非主线程加载

图片的解析会花费很多的时间。这是导致网页的帧数下降的其中一个重要的原因,因为解析工作会被执行在主线程中。在 React Native 中,图片的解析会在不同的线程中执行。在实际操作中,你已经处理好这种情况,当图片还没有下载完成,因此需要将 placeholder 显示出来,这不用你写任何代码。

举例

  1. 'use strict';
  2. var React = require('react-native');
  3. var {
  4. Image,
  5. StyleSheet,
  6. Text,
  7. View,
  8. ActivityIndicatorIOS
  9. } = React;
  10. var ImageCapInsetsExample = require('./ImageCapInsetsExample');
  11. var NetworkImageExample = React.createClass({
  12. watchID: (null: ?number),
  13. getInitialState: function() {
  14. return {
  15. error: false,
  16. loading: false,
  17. progress: 0
  18. };
  19. },
  20. render: function() {
  21. var loader = this.state.loading ?
  22. <View style={styles.progress}>
  23. <Text>{this.state.progress}%</Text>
  24. <ActivityIndicatorIOS style={{marginLeft:5}}/>
  25. </View> : null;
  26. return this.state.error ?
  27. <Text>{this.state.error}</Text> :
  28. <Image
  29. source={this.props.source}
  30. style={[styles.base, {overflow: 'visible'}]}
  31. onLoadStart={(e) => this.setState({loading: true})}
  32. onError={(e) => this.setState({error: e.nativeEvent.error, loading: false})}
  33. onProgress={(e) => this.setState({progress: Math.round(100 * e.nativeEvent.loaded / e.nativeEvent.total)})}
  34. onLoad={() => this.setState({loading: false, error: false})}>
  35. {loader}
  36. </Image>;
  37. }
  38. });
  39. exports.displayName = (undefined: ?string);
  40. exports.framework = 'React';
  41. exports.title = '<Image>';
  42. exports.description = 'Base component for displaying different types of images.';
  43. exports.examples = [
  44. {
  45. title: 'Plain Network Image',
  46. description: 'If the `source` prop `uri` property is prefixed with ' +
  47. '"http", then it will be downloaded from the network.',
  48. render: function() {
  49. return (
  50. <Image
  51. source={{uri: 'http://facebook.github.io/react/img/logo_og.png'}}
  52. style={styles.base}
  53. />
  54. );
  55. },
  56. },
  57. {
  58. title: 'Plain Static Image',
  59. description: 'Static assets should be required by prefixing with `image!` ' +
  60. 'and are located in the app bundle.',
  61. render: function() {
  62. return (
  63. <View style={styles.horizontal}>
  64. <Image source={require('image!uie_thumb_normal')} style={styles.icon} />
  65. <Image source={require('image!uie_thumb_selected')} style={styles.icon} />
  66. <Image source={require('image!uie_comment_normal')} style={styles.icon} />
  67. <Image source={require('image!uie_comment_highlighted')} style={styles.icon} />
  68. </View>
  69. );
  70. },
  71. },
  72. {
  73. title: 'Error Handler',
  74. render: function() {
  75. return (
  76. <NetworkImageExample source={{uri: 'http://TYPO_ERROR_facebook.github.io/react/img/logo_og.png'}} />
  77. );
  78. },
  79. platform: 'ios',
  80. },
  81. {
  82. title: 'Image Download Progress',
  83. render: function() {
  84. return (
  85. <NetworkImageExample source={{uri: 'http://facebook.github.io/origami/public/images/blog-hero.jpg?r=1'}}/>
  86. );
  87. },
  88. platform: 'ios',
  89. },
  90. {
  91. title: 'Border Color',
  92. render: function() {
  93. return (
  94. <View style={styles.horizontal}>
  95. <Image
  96. source={smallImage}
  97. style={[
  98. styles.base,
  99. styles.background,
  100. {borderWidth: 3, borderColor: '#f099f0'}
  101. ]}
  102. />
  103. </View>
  104. );
  105. },
  106. platform: 'ios',
  107. },
  108. {
  109. title: 'Border Width',
  110. render: function() {
  111. return (
  112. <View style={styles.horizontal}>
  113. <Image
  114. source={smallImage}
  115. style={[
  116. styles.base,
  117. styles.background,
  118. {borderWidth: 5, borderColor: '#f099f0'}
  119. ]}
  120. />
  121. </View>
  122. );
  123. },
  124. platform: 'ios',
  125. },
  126. {
  127. title: 'Border Radius',
  128. render: function() {
  129. return (
  130. <View style={styles.horizontal}>
  131. <Image
  132. style={[styles.base, {borderRadius: 5}]}
  133. source={fullImage}
  134. />
  135. <Image
  136. style={[styles.base, styles.leftMargin, {borderRadius: 19}]}
  137. source={fullImage}
  138. />
  139. </View>
  140. );
  141. },
  142. },
  143. {
  144. title: 'Background Color',
  145. render: function() {
  146. return (
  147. <View style={styles.horizontal}>
  148. <Image source={smallImage} style={styles.base} />
  149. <Image
  150. style={[
  151. styles.base,
  152. styles.leftMargin,
  153. {backgroundColor: 'rgba(0, 0, 100, 0.25)'}
  154. ]}
  155. source={smallImage}
  156. />
  157. <Image
  158. style={[styles.base, styles.leftMargin, {backgroundColor: 'red'}]}
  159. source={smallImage}
  160. />
  161. <Image
  162. style={[styles.base, styles.leftMargin, {backgroundColor: 'black'}]}
  163. source={smallImage}
  164. />
  165. </View>
  166. );
  167. },
  168. },
  169. {
  170. title: 'Opacity',
  171. render: function() {
  172. return (
  173. <View style={styles.horizontal}>
  174. <Image
  175. style={[styles.base, {opacity: 1}]}
  176. source={fullImage}
  177. />
  178. <Image
  179. style={[styles.base, styles.leftMargin, {opacity: 0.8}]}
  180. source={fullImage}
  181. />
  182. <Image
  183. style={[styles.base, styles.leftMargin, {opacity: 0.6}]}
  184. source={fullImage}
  185. />
  186. <Image
  187. style={[styles.base, styles.leftMargin, {opacity: 0.4}]}
  188. source={fullImage}
  189. />
  190. <Image
  191. style={[styles.base, styles.leftMargin, {opacity: 0.2}]}
  192. source={fullImage}
  193. />
  194. <Image
  195. style={[styles.base, styles.leftMargin, {opacity: 0}]}
  196. source={fullImage}
  197. />
  198. </View>
  199. );
  200. },
  201. },
  202. {
  203. title: 'Nesting',
  204. render: function() {
  205. return (
  206. <Image
  207. style={{width: 60, height: 60, backgroundColor: 'transparent'}}
  208. source={fullImage}>
  209. <Text style={styles.nestedText}>
  210. React
  211. </Text>
  212. </Image>
  213. );
  214. },
  215. },
  216. {
  217. title: 'Tint Color',
  218. description: 'The `tintColor` style prop changes all the non-alpha ' +
  219. 'pixels to the tint color.',
  220. render: function() {
  221. return (
  222. <View>
  223. <View style={styles.horizontal}>
  224. <Image
  225. source={require('image!uie_thumb_normal')}
  226. style={[styles.icon, {borderRadius: 5, tintColor: '#5ac8fa' }]}
  227. />
  228. <Image
  229. source={require('image!uie_thumb_normal')}
  230. style={[styles.icon, styles.leftMargin, {borderRadius: 5, tintColor: '#4cd964' }]}
  231. />
  232. <Image
  233. source={require('image!uie_thumb_normal')}
  234. style={[styles.icon, styles.leftMargin, {borderRadius: 5, tintColor: '#ff2d55' }]}
  235. />
  236. <Image
  237. source={require('image!uie_thumb_normal')}
  238. style={[styles.icon, styles.leftMargin, {borderRadius: 5, tintColor: '#8e8e93' }]}
  239. />
  240. </View>
  241. <Text style={styles.sectionText}>
  242. It also works with downloaded images:
  243. </Text>
  244. <View style={styles.horizontal}>
  245. <Image
  246. source={smallImage}
  247. style={[styles.base, {borderRadius: 5, tintColor: '#5ac8fa' }]}
  248. />
  249. <Image
  250. source={smallImage}
  251. style={[styles.base, styles.leftMargin, {borderRadius: 5, tintColor: '#4cd964' }]}
  252. />
  253. <Image
  254. source={smallImage}
  255. style={[styles.base, styles.leftMargin, {borderRadius: 5, tintColor: '#ff2d55' }]}
  256. />
  257. <Image
  258. source={smallImage}
  259. style={[styles.base, styles.leftMargin, {borderRadius: 5, tintColor: '#8e8e93' }]}
  260. />
  261. </View>
  262. </View>
  263. );
  264. },
  265. },
  266. {
  267. title: 'Resize Mode',
  268. description: 'The `resizeMode` style prop controls how the image is ' +
  269. 'rendered within the frame.',
  270. render: function() {
  271. return (
  272. <View style={styles.horizontal}>
  273. <View>
  274. <Text style={[styles.resizeModeText]}>
  275. Contain
  276. </Text>
  277. <Image
  278. style={styles.resizeMode}
  279. resizeMode={Image.resizeMode.contain}
  280. source={fullImage}
  281. />
  282. </View>
  283. <View style={styles.leftMargin}>
  284. <Text style={[styles.resizeModeText]}>
  285. Cover
  286. </Text>
  287. <Image
  288. style={styles.resizeMode}
  289. resizeMode={Image.resizeMode.cover}
  290. source={fullImage}
  291. />
  292. </View>
  293. <View style={styles.leftMargin}>
  294. <Text style={[styles.resizeModeText]}>
  295. Stretch
  296. </Text>
  297. <Image
  298. style={styles.resizeMode}
  299. resizeMode={Image.resizeMode.stretch}
  300. source={fullImage}
  301. />
  302. </View>
  303. </View>
  304. );
  305. },
  306. },
  307. {
  308. title: 'Animated GIF',
  309. render: function() {
  310. return (
  311. <Image
  312. style={styles.gif}
  313. source={{uri: 'http://38.media.tumblr.com/9e9bd08c6e2d10561dd1fb4197df4c4e/tumblr_mfqekpMktw1rn90umo1_500.gif'}}
  314. />
  315. );
  316. },
  317. platform: 'ios',
  318. },
  319. {
  320. title: 'Cap Insets',
  321. description:
  322. 'When the image is resized, the corners of the size specified ' +
  323. 'by capInsets will stay a fixed size, but the center content and ' +
  324. 'borders of the image will be stretched. This is useful for creating ' +
  325. 'resizable rounded buttons, shadows, and other resizable assets.',
  326. render: function() {
  327. return <ImageCapInsetsExample />;
  328. },
  329. platform: 'ios',
  330. },
  331. ];
  332. var fullImage = {uri: 'http://facebook.github.io/react/img/logo_og.png'};
  333. var smallImage = {uri: 'http://facebook.github.io/react/img/logo_small_2x.png'};
  334. var styles = StyleSheet.create({
  335. base: {
  336. width: 38,
  337. height: 38,
  338. },
  339. progress: {
  340. flex: 1,
  341. alignItems: 'center',
  342. flexDirection: 'row',
  343. width: 100
  344. },
  345. leftMargin: {
  346. marginLeft: 10,
  347. },
  348. background: {
  349. backgroundColor: '#222222'
  350. },
  351. sectionText: {
  352. marginVertical: 6,
  353. },
  354. nestedText: {
  355. marginLeft: 12,
  356. marginTop: 20,
  357. backgroundColor: 'transparent',
  358. color: 'white'
  359. },
  360. resizeMode: {
  361. width: 90,
  362. height: 60,
  363. borderWidth: 0.5,
  364. borderColor: 'black'
  365. },
  366. resizeModeText: {
  367. fontSize: 11,
  368. marginBottom: 3,
  369. },
  370. icon: {
  371. width: 15,
  372. height: 15,
  373. },
  374. horizontal: {
  375. flexDirection: 'row',
  376. },
  377. gif: {
  378. flex: 1,
  379. height: 200,
  380. },
  381. });