Skip to content

Latest commit

 

History

History
883 lines (695 loc) · 34.4 KB

File metadata and controls

883 lines (695 loc) · 34.4 KB

三、实现复杂用户界面——第一部分

在本章中,我们将实现复杂的用户界面。我们将了解更多关于使用 flexbox 创建在不同屏幕尺寸上工作的组件的信息,以及如何检测方向变化等。

本章将介绍以下配方:

  • 创建具有主题支持的可重用按钮
  • 使用 flexbox 为平板电脑构建复杂布局
  • 包括自定义字体
  • 使用字体图标

创建具有主题支持的可重用按钮

在开发软件时,可重用性非常重要。我们应该避免重复同一件事一遍又一遍,相反,我们应该创建可以重复使用尽可能多的小组件。

在这个配方中,我们将创建一个Button组件,我们还将定义几个属性来更改其外观。在学习这个方法的同时,我们将学习如何动态地将不同的样式应用于组件。

准备

我们需要创建一个空的应用。让我们把它命名为reusable-button

怎么做。。。

  1. 在新应用的根目录中,我们需要为可重用的按钮相关代码创建一个新的Button文件夹。让我们在新的Button文件夹中创建index.jsstyles.js
  2. 我们将从导入新组件的依赖项开始。在Button/index.js文件中,我们将创建一个Button组件。这意味着我们需要进口TextTouchableOpacity组件。您会注意到,我们还导入了尚不存在的样式。我们将在本配方后面的不同文件中定义这些样式。在Button/index.js文件中,我们应该有以下导入:
import React, { Component } from 'react';

import {
  Text,
  TouchableOpacity,
} from 'react-native';

import {
  Base,
  Default,
  Danger,
  Info,
  Success
} from './styles';
  1. 现在我们已经导入了依赖项,让我们定义这个组件的类。我们需要一些性质和两种方法。我们还需要导出此组件,以便在其他地方使用:
export default class Button extends Component {
  getTheme() {
    // Defined in a later step
  }

  render() {
    // Defined in a later step
  }
}
  1. 我们需要根据给定的properties选择要应用于我们组件的样式。为此,我们将定义getTheme方法。此方法将检查properties中是否有true项,并返回相应的样式,如果没有true项,则返回Default样式:
  getTheme() {
    const { danger, info, success } = this.properties;

    if (info) {
      return Info;
    }

    if (success) {
      return Success;
    }

    if (danger) {
      return Danger;
    }

    return Default;
  }
  1. 要求所有组件都有一个render方法。这里,我们需要返回这个组件的 JSX 元素。在本例中,我们将获得给定properties的样式,并将其应用于TouchableOpacity组件。 我们还为按钮定义了一个标签。在这个标签中,我们将呈现children属性。如果收到回调函数,则当用户按下此组件时将执行该函数:
  render() {
    const theme = this.getTheme();
    const {
      children,
      onPress,
      style,
      rounded,
    } = this.properties;

    return (
      <TouchableOpacity
        activeOpacity={0.8}
        style={[
          Base.main,
          theme.main,
          rounded ? Base.rounded : null ,
          style,
        ]}
        onPress={onPress}
      >
        <Text style={[Base.label, theme.label]}>{children}</Text>
      </TouchableOpacity>
    );
  }
  1. 我们的Button组件快完成了。我们仍然需要定义样式,但首先让我们转到项目根目录中的App.js文件。我们需要导入依赖项,包括我们创建的Button组件。 当用户点击按钮时,我们将显示一条警告消息,因此我们还需要导入Alert组件:
import React from 'react';
import {
  Alert,
  StyleSheet,
  View
} from 'react-native';
import Button from './Button';
  1. 一旦我们有了所有的依赖项,让我们定义一个呈现几个按钮的无状态组件。第一个按钮将使用默认样式,第二个按钮将使用成功样式,这将为按钮的背景添加一个漂亮的绿色。最后一个按钮按下时将显示警报。为此,我们需要定义将使用Alert组件的回调函数,只需设置标题和消息:
export default class App extends React.Component {
  handleButtonPress() {
    Alert.alert('Alert', 'You clicked this button!');
  }

  render() {
    return(
      <View style={styles.container}>
        <Button style={styles.button}>
          My first button
        </Button>
        <Button success style={styles.button}>
          Success button
        </Button>
        <Button info style={styles.button}>
          Info button
        </Button>
        <Button danger rounded style={styles.button}
        onPress={this.handleButtonPress}>
          Rounded button
        </Button>
      </View>
    );
  }
}
  1. 我们将为主布局如何对齐和对齐每个按钮添加一些样式,以及一些边距:
const styles = StyleSheet.create({
  container: {
      flex: 1,
      alignItems: 'center',
      justifyContent: 'center',
    },
  button: {
    margin: 10,
  },
});
  1. 如果我们现在尝试运行该应用,将会出现一些错误。这是因为我们尚未声明按钮的样式。让我们现在就开始吧。在Button/styles.js文件中,我们需要定义基本样式。这些样式将应用于按钮的每个实例。在这里,我们将定义半径、填充、字体颜色以及此组件所需的所有常用样式:
import { StyleSheet } from 'react-native';

const Base = StyleSheet.create({
  main: {
    padding: 10,
    borderRadius: 3,
  },
  label: {
    color: '#fff',
  },
  rounded: {
    borderRadius: 20,
  },
});
  1. 一旦我们有了按钮的通用样式,我们需要定义DangerInfoSuccessDefault主题的样式。为此,我们将为每个主题定义不同的对象。在每个主题中,我们将使用相同的对象,但该主题具有特定的样式。 为了简单起见,我们只打算更改backgroundColor,但我们可以选择使用任意多的样式属性:
const Danger = StyleSheet.create({
  main: {
    backgroundColor: '#e74c3c',
  },
});

const Info = StyleSheet.create({
  main: {
    backgroundColor: '#3498db',
  },
});

const Success = StyleSheet.create({
  main: {
    backgroundColor: '#1abc9c',
  },
});

const Default = StyleSheet.create({
  main: {
    backgroundColor: 'rgba(0 ,0 ,0, 0)',
  },
  label: {
    color: '#333',
  },
});
  1. 最后,让我们导出样式。此步骤是必要的,以便Button组件可以导入每个主题的所有样式:
export {
  Base,
  Danger,
  Info,
  Success,
  Default,
};
  1. 如果我们打开应用,我们应该能够看到我们完成的布局:

它是如何工作的。。。

在本例中,我们使用了TouchableOpacity组件。该组件允许我们定义一个漂亮的动画,当用户按下按钮时,该动画会更改不透明度。

按下按钮时,我们可以使用activeOpacity属性设置不透明度值。数值可以是介于01之间的任意数字,其中0是完全透明的。

如果按下圆形按钮,我们将看到一条本机警报消息,如以下屏幕截图所示:

使用 flexbox 为平板电脑构建复杂布局

Flexbox 在创建响应性布局时非常方便。React Native 使用 flexbox 作为布局系统,如果您已经熟悉这些概念,那么开始创建任何类型的布局都非常容易。

如前一章所述,flexbox 在 React Native 中的工作方式与在 CSS 中的工作方式存在一些差异。有关 React-Native 和 CSS flexbox 之间差异的更多信息,请参阅第 2 章使用 flexbox 创建布局配方创建简单的 React-Native 应用中的*如何工作…*部分。

在这个配方中,我们将创建一个布局来显示博客文章列表。每篇文章都是一张小卡片,上面有一张图片、一段摘录和一个阅读更多内容的按钮。我们将使用 flexbox 根据屏幕大小在主容器上安排帖子。这将允许我们通过在横向和纵向上正确对齐卡片来处理屏幕旋转。

准备

我们需要一个新的应用来制作这个食谱。让我们把它命名为tablet-flexbox

当我们用 Expo 创建一个新的应用时,会在项目的基础上创建一个app.json,提供一些基本配置。在这个配方中,我们正在构建一个应用,我们希望确保它在平板电脑上看起来不错,尤其是在横向模式下。当我们打开app.json时,我们会看到orientation属性设置为'portrait'。此属性确定应用中应允许的方向。orientation属性接受'portrait'(将应用锁定为纵向模式)、'landscape'(将应用锁定为横向模式)和'default'(允许应用根据设备方向调整屏幕方向)。对于我们的应用,我们将把orientation设置为'landscape',这样我们就可以支持横向和纵向布局。

我们还将使用一些图像,这些图像需要远程托管,以便正确模拟加载远程数据和使用Image组件显示图像。我已将这些图像上传到www.imgur.com图像托管服务,并在配方用于其耗材数据的data.json文件中引用了这些远程图像。如果出于任何原因,这些远程图像无法正确加载,它们也会包含在/assets文件夹下的此配方存储库中。请随时将它们上载到任何服务器或托管服务,并相应地更新data.json中的图像 URL。可在 GitHub 的上找到存储库 https://github.com/warlyware/react-native-cookbook/tree/master/chapter-3/tablet-flexbox

怎么做。。。

  1. 首先,我们需要在项目的根目录中创建一个Post文件夹。我们还需要在新的Post文件夹中创建index.jsstyles.js文件。我们将使用这个Post组件来显示我们应用的每个帖子。最后,我们需要在项目的根目录中添加一个data.json文件,我们将使用该文件定义一个帖子列表。
  2. 现在我们可以继续构建App.js组件。首先,我们需要导入这个类的依赖项。我们将使用ListView组件来呈现帖子列表。我们还需要TextView组件用于内容容器。我们将创建一个自定义的Post组件来呈现列表中的每个帖子,我们还需要导入data.json文件:
import React, { Component } from 'react';
import { ListView, StyleSheet, Text, View } from 'react-native';

import Post from './Post';
import data from './data.json';
  1. 让我们为App组件创建类。在这里,我们将使用.json文件中的数据为列表创建dataSource。我们将在下一步向我们的data.json文件中添加一些实际数据。在render方法中,我们将定义一个简单的顶部工具栏和List组件。我们将对每条记录使用Post组件,并从state中获取dataSource

如果您对ListView组件有任何疑问,请查看第 2 章中的配方,创建一个简单的 React 原生应用,我们在其中创建了一个订单列表:

const dataSource = new ListView.DataSource({
  rowHasChanged: (r1, r2) => r1 !== r2,
});

export default class App extends Component {
  state = {
    dataSource: dataSouce.cloneWithRows(data.posts),
  };

  render() {
    return (
      <View style={styles.container}>
        <View style={styles.toolbar}>
          <Text style={styles.title}>Latest posts</Text>
        </View>
        <ListView
          dataSource={this.state.dataSource}
          renderRow={post => <Post {...post} />}
          style={styles.list}
          contentContainerStyle={styles.content}
        />
      </View>
    );
  }
}
  1. 仍然缺少两个文件:包含数据的.json文件和Post组件。在这一步中,我们将创建用于每篇文章的数据。简单地说,下面的代码片段中只有一条数据记录,但是我在这个配方中使用的POST对象的其余部分可以在这个配方的代码库的data.json文件中找到,位于https://github.com/warlyware/react-native-cookbook/blob/master/chapter-3/tablet-flexbox/data.json :
{
  "posts": [
    {
      "title": "The Best Article Ever Written",
      "img": "https://i.imgur.com/mf9daCT.jpg",
      "content": "Lorem ipsum dolor sit amet...",
      "author": "Bob Labla"
    },
    // Add more records here.
  ]
}
  1. 现在我们有了一些数据,我们已经准备好处理Post组件。在这个组件中,我们需要显示图像、标题和按钮。因为这个组件不需要知道状态,所以我们将使用无状态组件。下面的代码使用了我们在第 2 章中学习的所有组件,创建了一个简单的 React 原生应用。如果有不清楚的地方,请重新阅读该章。 此组件将数据作为参数接收,然后我们将其用于显示组件中的内容。Image组件将使用data.json文件中每个对象上定义的img属性来显示远程图像:
import React from 'react';
import {
  Image,
  Text,
  TouchableOpacity,
  View
} from 'react-native';

import styles from './styles';

const Post = ({ content, img, title }) => (
  <View style={styles.main}>
    <Image
      source={{ uri: img }}
      style={styles.image}
    />
    <View style={styles.content}>
      <Text style={styles.title}>{title}</Text>
      <Text>{content}</Text>
    </View>
    <TouchableOpacity style={styles.button} activeOpacity={0.8}>
      <Text style={styles.buttonText}>Read more</Text>
    </TouchableOpacity>
  </View>
);

export default Post;
  1. 一旦我们定义了组件,我们还需要为每个帖子定义样式。让我们创建一个空的StyleSheet导出,以便依赖styles.jsPost组件能够正常工作:
import { StyleSheet } from 'react-native';

const styles = StyleSheet.create({
  // Defined in later steps
});

export default styles;
  1. 如果我们尝试运行该应用,我们应该能够在屏幕上看到来自.json文件的数据。但它不会很漂亮,因为我们还没有应用任何样式。
  2. 我们在屏幕上有我们需要的一切。现在我们已经准备好开始布局工作。首先,让我们为Post容器添加样式。我们将设置widthheightborderRadius和其他一些。让我们将它们添加到/Post/styles.js文件中:
const styles = StyleSheet.create({
  main: {
    backgroundColor: '#fff',
    borderRadius: 3,
    height: 340,
    margin: 5,
    width: 240,
  }
});
  1. 到现在为止,我们应该看到小盒子垂直对齐。这是一个进步,但我们需要为图像添加更多的样式,以便我们可以在屏幕上看到它。让我们将一个image属性添加到上一步的相同styles常量中。resizeMode属性允许我们设置如何调整图像大小。在这种情况下,通过选择cover,图像将保持原稿的纵横比:
  image: {
    backgroundColor: '#ccc',
    height: 120,
    resizeMode: 'cover',
  }
  1. 对于帖子的content,我们希望占据卡片上所有可用的高度,因此我们需要使其灵活并添加一些填充。我们还将在内容中添加overflow: hidden以避免View元素溢出。对于title,我们只需更改fontSize并在底部添加一个margin
  content: {
    padding: 10,
    overflow: 'hidden',
    flex: 1,
  },
  title: {
    fontSize: 18,
    marginBottom: 5,
  },
  1. 最后,对于按钮,我们将backgroundColor设置为绿色,文本设置为白色。我们还需要增加一些paddingmargin的间距:
  button: {
    backgroundColor: '#1abc9c',
    borderRadius: 3,
    padding: 10,
    margin: 10,
  },
  buttonText: {
    color: '#fff',
    textAlign: 'center',
  }
  1. 如果我们刷新模拟器,我们应该可以在小卡片上看到我们的帖子。目前,卡片是垂直排列的,但我们希望水平渲染所有卡片。我们将通过以下步骤解决此问题:

Primary styles have been added for all post elements

  1. 目前,我们只能在一列中看到列表中的前三项,而不是在屏幕上的一行中。让我们回到App.js文件,开始添加我们的样式。我们将flex: 1添加到container中,以便我们的布局始终填充屏幕。我们还希望在顶部显示工具栏。为此,我们只需要定义一些paddingcolor如下:
const styles = StyleSheet.create({
  container: {
    flex: 1,
  },
  toolbar: {
    backgroundColor: '#34495e',
    padding: 10,
    paddingTop: 20,
  },
  title: {
    color: '#fff',
    fontSize: 20,
    textAlign: 'center',
  }
});
  1. 让我们也为list添加一些基本样式。只是一个漂亮的背景色和一些填充。我们还将添加flex属性,这将确保列表占据屏幕上所有可用的高度。这里只有两个组件:工具栏和列表。工具栏大约需要 50 像素。如果我们使列表灵活,它将占用所有剩余的可用空间,这正是我们在旋转设备或以不同屏幕分辨率运行应用时所需要的:
  list: {
    backgroundColor: '#f0f3f4',
    flex: 1,
    paddingTop: 5,
    paddingBottom: 5,
  }
  1. 如果我们再次在模拟器中检查应用,我们应该能够看到工具栏和列表按预期布局:

Styles have been applied to each post to give them a card like appearance

  1. 我们几乎完成了这个应用。我们所要做的就是水平排列卡片。使用 flexbox 可以通过三个简单的步骤实现这一点:
        content: { 
          flexDirection: 'row', 
          flexWrap: 'wrap', 
          justifyContent: 'space-around', 
        }, 

第一步是通过ListView组件中的contentContainerStyle属性应用这些content样式。在内部,ListView组件将这些样式应用于内容容器,内容容器封装所有子视图。 然后我们将flexDirection设置为row。这将水平对齐列表上的卡片;然而,这带来了一个新问题:我们只能看到一行帖子。为了解决这个问题,我们需要包装这些项目。为此,我们将flexWrap属性设置为wrap,这将自动将视图中不适合的项目移动到下一行。最后,我们使用了{ To8t}属性,并将其设置为 AutoT99T,这将将我们的 Ty10 T10 置于我们的应用的中间。

  1. 我们现在有了一款响应迅速的应用,在平板电脑上以横向模式运行时效果良好:

Side-by-side comparison of iPad and Android tablet screenshots in landscape mode

在纵向模式下看起来也一样好:

Side-by-side comparison of iPad and Android tablet screenshots in portrait mode

还有更多。。。

Expo 还提供了一个ScreenOrientation助手,用于更改应用的方向配置。此帮助器还允许进行更精细的方向设置(如ALL_BUT_UPSIDE_DOWNLANDSCAPE_RIGHT)。如果您的应用需要动态、精细地控制屏幕方向,请参阅ScreenOrientation世博会文档以获取信息:https://docs.expo.io/versions/v24.0.0/sdk/screen-orientation.html

另见

静态图像资源和<Image>组件的官方文档可在找到 https://facebook.github.io/react-native/docs/images.html

包括自定义字体

在某种程度上,我们可能希望使用自定义字体族显示文本。到目前为止,我们一直使用默认字体,但我们可以使用任何我们喜欢的其他字体。

在世博会之前,添加自定义字体的过程更加困难,需要使用本机代码,并且需要在 iOS 和 Android 中以不同的方式实现。幸运的是,通过使用 Expo 的 font helper 库,这已变得精简和简化

在此配方中,我们将导入一些字体,然后使用每个导入的字体系列显示文本。我们还将使用不同的字体样式,如粗体斜体

准备

为了处理这个例子,我们需要一些字体。你可以使用任何你想要的字体。我建议去谷歌字体(https://fonts.google.com/ 下载您的收藏夹。对于这个配方,我们将使用Josefin Sans 和 Raleway 字体。

*下载字体后,让我们创建一个空应用并将其命名为custom-fonts。当我们使用 Expo 创建一个空白应用时,它会在项目根目录中创建一个assets文件夹,用于放置您的所有资产(图像、字体等),因此我们将按照标准将字体添加到此文件夹。让我们创建/img/fonts文件夹并添加从谷歌字体下载的自定义字体文件。

从 Google fonts 下载字体时,您将获得一个包含每个字体系列变体的.ttf文件的.zip文件。我们将使用常规的粗体斜体变体,因此将每个系列中每个变体对应的.ttf文件复制到我们的/img/fonts文件夹。

**# 怎么做。。。

  1. 有了我们的字体文件,第一步是打开App.js并添加我们需要的导入:
import React from 'react';
import { Text, View, StyleSheet } from 'react-native';
import { Font } from 'expo';
  1. 接下来,我们将添加一个简单的组件,用于显示一些要使用自定义字体设置样式的文本。我们将从一个Text元素开始,以显示 Roboto 字体的常规变体:
export default class App extends React.Component {
  render() {
    return (
      <View style={styles.container}>
        <Text style={styles.josefinSans}>
          Hello, Josefin Sans!
        </Text>
      </View>
    );
  }
}
  1. 让我们为刚刚创建的组件添加一些初学者样式。现在,我们只增加josefinSans类样式的字体大小:
const styles = StyleSheet.create({
  container: {
    flex: 1,
    backgroundColor: '#fff',
    alignItems: 'center',
    justifyContent: 'center',
  },
  josefinSans: {
    fontSize: 40,
  }
});
  1. 如果我们现在在模拟器中打开应用,我们将看到 Hello,Josefin Sans!使用默认字体显示在屏幕中部的文本:

  1. 让我们加载我们的JosefinSans-Regular.ttf字体文件,这样我们就可以用它来设置文本的样式。我们将使用 React Native 提供的componentDidMount生命周期挂钩告诉我们的应用何时开始加载字体:
export default class App extends React.Component {

 componentDidMount() {
    Font.loadAsync({
      'josefin-sans-regular': require('./img/fonts/JosefinSans-Regular.ttf'),
    });
  }

  render() {
    return (
      <View style={styles.container}>
        <Text style={styles.josefinSans}>
          Hello, Josefin Sans!
        </Text>
      </View>
    );
  }
}
  1. 接下来,我们将把加载的字体添加到应用于Text元素的样式中:
const styles = StyleSheet.create({
  // Other styles from step 3
  josefinSans: {
    fontSize: 40,
 fontFamily: 'josefin-sans-regular'
  }
});
  1. 我们现在有了风格,对吗?嗯,不完全是。如果我们回顾一下我们的模拟器,我们会发现我们得到了一个错误:
console.error: "fontFamily 'josefin-sans-regular' is not a system font and has not been loaded through Expo.Font.loadAsync"
  1. 但我们只是通过Expo.Font.loadAsync加载字体!有什么好处?结果是我们手上有一个比赛条件。我们为Text元素定义的josefinSans样式是在加载 Josefin Sans 字体之前应用的。要处理此问题,需要使用组件的state来跟踪字体的加载状态:
export default class App extends React.Component {
 state = {
    fontLoaded: false
  };
  1. 既然我们的组件有一个state,那么一旦加载字体,我们就可以将状态的fontLoaded属性更新为true。使用 ES6 功能async/await使这一点简洁明了。让我们在componentDidMount代码块中执行此操作:
  async componentDidMount() {
    await Font.loadAsync({
      'josefin-sans-regular': require('./img/fonts/JosefinSans-
      Regular.ttf'),
    });
  }
  1. 由于我们现在正在等待Font.loadAsync()呼叫,一旦呼叫完成,我们可以将fontLoaded的状态设置为true
  async componentDidMount() {
    await Font.loadAsync({
      'josefin-sans-regular': require('./img/fonts/JosefinSans-
      Regular.ttf'),
    });

 this.setState({ fontLoaded: true });
  }
  1. 剩下要做的就是更新我们的render方法,当fontLoaded状态属性为true时,只呈现依赖于自定义字体的Text元素:
      <View style={styles.container}>
 {
          this.state.fontLoaded ? (
            <Text style={styles.josefinSans}>
              Hello, Josefin Sans!
            </Text>
          ) : null
        }
      </View>
  1. 现在,当我们在模拟器中查看我们的应用时,我们应该看到我们的自定义字体正在应用:

  1. 让我们加载其余字体,以便我们也可以在应用中使用它们:
    await Font.loadAsync({
      'josefin-sans-regular': require('./img/fonts/JosefinSans-
      Regular.ttf'),
 'josefin-sans-bold': require('./img/fonts/JosefinSans-
      Bold.ttf'),
      'josefin-sans-italic': require('./img/fonts/JosefinSans-
      Italic.ttf'),
      'raleway-regular': require('./img/fonts/Raleway-
      Regular.ttf'),
      'raleway-bold': require('./img/fonts/Raleway-Bold.ttf'),
      'raleway-italic': require('./img/fonts/Raleway-
      Italic.ttf'),
    });
  1. 我们还需要Text元素来显示每个新字体系列/变体中的文本。注意,我们还需要将所有的Text元素包装到另一个View元素中,因为 JSX 表达式要求只有一个父节点。我们现在还向style属性传递了一系列要应用的样式,以便整合我们将在下一步应用的fontSizepadding样式:
  render() {
    return (
      <View style={styles.container}>
 {
          this.state.fontLoaded ? (
            <View style={styles.container}>
              <Text style={[styles.josefinSans, 
              styles.textFormatting]}>
                Hello, Josefin Sans!
              </Text>
              <Text style={[styles.josefinSansBold,
              styles.textFormatting]}>
                Hello, Josefin Sans!
              </Text>
              <Text style={[styles.josefinSansItalic, 
              styles.textFormatting]}>
                Hello, Josefin Sans!
              </Text>
              <Text style={[styles.raleway, styles.textFormatting]}>
                Hello, Raleway!
              </Text>
              <Text style={[styles.ralewayBold, 
              styles.textFormatting]}>
                Hello, Raleway!
              </Text>
              <Text style={[styles.ralewayItalic, 
              styles.textFormatting]}>
                Hello, Raleway!
              </Text>
            </View>
          ) : null
        }
      </View>
    );
  }
  1. 只需将新样式添加到StyleSheet即可应用我们的自定义字体:
const styles = StyleSheet.create({
  container: {
    flex: 1,
    backgroundColor: '#fff',
    alignItems: 'center',
    justifyContent: 'center',
  },
  josefinSans: {
    fontFamily: 'josefin-sans-regular',
  },
  josefinSansBold: {
    fontFamily: 'josefin-sans-bold',
  },
  josefinSansItalic: {
    fontFamily: 'josefin-sans-italic',
  },
  raleway: {
    fontFamily: 'raleway-regular',
  },
  ralewayBold: {
    fontFamily: 'josefin-sans-bold'
  },
  ralewayItalic: {
    fontFamily: 'josefin-sans-italic',
  },
  textFormatting: {
    fontSize: 40,
    paddingBottom: 20
  }
});
  1. 现在,在我们的应用中,我们将看到六个不同的文本元素,每个元素都有自己的自定义字体:

它是如何工作的。。。

步骤 5步骤 6中,我们使用componentDidMountReact 生命周期钩子来告诉我们的应用何时完成加载。虽然使用componentWillMount似乎很诱人,但这也会引发错误,因为componentWillMount不能保证等待Font.loadAsync完成。通过使用componentDidMount,我们还可以确保不会阻止应用的初始呈现。

步骤 9中,我们使用了 ES6 功能async/await。如果您是一名 web 开发人员,您可能会熟悉这种模式,但如果您想了解更多信息,我在另见本配方末尾的部分中包含了一篇来自ponyfoo.com的精彩文章,这很好地解释了async/await的工作原理。

步骤 11中,我们使用了一个三元语句来呈现自定义字体样式的Text元素(如果已加载),或者通过返回null来呈现未加载的内容

Fonts loaded through Expo don’t currently support the fontWeight or fontStyle properties—you will need to load those variations of the font and specify them by name, as we have done here with bold and italic.

另见

一篇关于async/await的好文章可以在找到 https://ponyfoo.com/articles/understanding-javascript-async-await

使用字体图标

图标几乎是任何应用不可或缺的一部分,尤其是在导航和按钮中。与上一章介绍的 Expo 字体助手类似,Expo 还有一个图标助手,它使添加图标字体比使用 Villa React Native 简单得多。在这个配方中,我们将看到如何将图标助手模块与流行的FontAwesomeIonicons图标字体库一起使用。

准备

我们需要为这个配方做一个新的项目。让我们把这个项目命名为font-icons

怎么做。。。

  1. 我们将首先打开App.js并导入构建应用所需的依赖项:
import React from 'react';
import { StyleSheet, Text, View } from 'react-native';
import { FontAwesome, Ionicons } from '@expo/vector-icons';
  1. 接下来,我们可以添加应用的外壳,在这里我们将显示图标:
export default class App extends React.Component {
  render() {
    return (
      <View style={styles.container}>
      </View>
    );
  }
}
  1. View元素内部,我们再添加两个View元素,用于保存每个图标集中的图标:
export default class App extends React.Component {
  render() {
    return (
      <View style={styles.container}>
 <View style={styles.iconRow}>

        </View>
        <View style={styles.iconRow}>

        </View>
      </View>
    );
  }
}
  1. 现在,让我们为每个声明的元素添加样式。正如我们在前面的食谱中所看到的,container样式用flex: 1填充屏幕,并将alignItemsjustifyContent设置为center使项目居中。iconRow属性将flexDirection设置为row,这样我们的图标将排成一行:
const styles = StyleSheet.create({
  container: {
    flex: 1,
    backgroundColor: '#fff',
    alignItems: 'center',
    justifyContent: 'center',
  },
  iconRow: {
    flexDirection: 'row',
  },
});
  1. 现在我们的应用的基本结构已经就绪,让我们添加图标。在第一行图标中,我们将使用四个FontAwesome组件来显示FontAwesome字体库中的四个图标。name属性决定应该使用哪个图标,size属性设置图标的像素大小,color设置图标应该是什么颜色:
<View style={styles.iconRow}>
 <FontAwesome style={styles.iconPadding} name="glass" size={48} color="green" />
  <FontAwesome style={styles.iconPadding} name="beer" size={48} color="red" />
  <FontAwesome style={styles.iconPadding} name="music" size={48} color="blue" />
  <FontAwesome style={styles.iconPadding} name="taxi" size={48} color="#1CB5AD" />
</View>

Just as in CSS, the color property can be a color keyword defined in the CSS specification (you can check out the full list in the MDN docs at https://developer.mozilla.org/en-US/docs/Web/CSS/color_value), or a hex code for a given color.

  1. 在下一个View元素中,我们将添加Ionicons字体库中的图标。如您所见,Ionicons元素与上一步中使用的FontAwesome元素具有相同的属性:
<View style={styles.iconRow}>
 <Ionicons style={styles.iconPadding} name="md-pizza" size={48} color="orange" />
  <Ionicons style={styles.iconPadding} name="md-tennisball" size={48} color="maroon" />
  <Ionicons style={styles.iconPadding} name="ios-thunderstorm" size={48} color="purple" />
  <Ionicons style={styles.iconPadding} name="ios-happy" size={48} color="#DF7977" />
</View>
  1. 本食谱的最后一步是添加剩余的样式iconPadding,它只是添加一些填充,以均匀地隔开每个图标:
const styles = StyleSheet.create({
  container: {
    flex: 1,
    backgroundColor: '#fff',
    alignItems: 'center',
    justifyContent: 'center',
  },
  iconRow: {
    flexDirection: 'row',
  },
 iconPadding: {
    padding: 8,
  }
});
  1. 就这些!当我们查看我们的应用时,将有两行图标,每行分别显示来自FontAwesomeIonicons的图标:

它是如何工作的。。。

世博会附带的vector-icons套餐提供 11 套完整的图标。您所要做的就是导入相关组件(例如,字体可怕图标的FontAwesome组件),并为其提供与要使用的图标集中的图标对应的名称。您可以在vector-icons目录中找到可用于vector-icons助手库的所有图标的完整、可搜索列表,该目录位于https://expo.github.io/vector-icons/ 。只需将元素的name属性设置为目录中列出的图标名称,添加sizecolor属性,就完成了!

正如vector-icons的 GitHub 自述文件所述,该库是为使用 Expo 中react-native-vector-icons包提供的图标而创建的兼容层。您可以在找到此套餐 https://github.com/oblador/react-native-vector-icons 。如果您正在构建一个没有 Expo 的 React 本机应用,您可以使用react-native-vector-icons库获得相同的功能。

另见

vector-icons库中所有可用图标的目录可在中找到 https://expo.github.io/vector-icons/ 。***