当前位置:   article > 正文

【iOS开发】(三)react Native核心组件十个20240418

【iOS开发】(三)react Native核心组件十个20240418

【前言】
(二)中 我们一起学习了属性和布局,现在,我们开始往layout内增加一些组件。

三 组件和api

简介 RN 中的核心组件

RN 中的核心组件,是对原生组件的封装

  • 原生组件: Android或ios内的组件
  • 核心组件:RN中最常用的,来在 react-native 的组件

在这里插入图片描述
在这里插入图片描述

开发 React Native 应用程序时,通常会使用以下核心组件:

View(视图):View 是 React Native 中最基本的组件之一,用于创建布局结构。它类似于 web 开发中的

元素,用于包裹和组织其他组件。
Text(文本):Text 组件用于显示文本内容,类似于 HTML 中的 或

元素。你可以在 Text 组件中显示静态文本或动态生成的文本。
TouchableOpacity(触摸透明度):TouchableOpacity 是一个用于处理用户触摸操作的组件。当用户触摸组件时,TouchableOpacity 会在视觉上降低其透明度,以提供触摸的视觉反馈。
Dimensions(尺寸):Dimensions 组件用于获取设备屏幕的尺寸信息,包括屏幕宽度和高度。这在创建响应式布局时非常有用。
StyleSheet(样式表):StyleSheet 是用于创建和管理样式的工具,类似于 CSS 样式表。你可以使用 StyleSheet 来定义组件的样式,并将其应用于对应的组件。
StatusBar(状态栏):StatusBar 组件用于控制应用程序的状态栏的样式和行为。你可以使用 StatusBar 来设置状态栏的背景颜色、文字颜色等属性。
Image(图片):Image 组件用于在应用程序中显示图片。你可以使用 Image 组件加载网络图片或本地图片,并根据需要对其进行调整和处理。
ImageBackground(背景图片):ImageBackground 组件允许你在其内部放置一个图片,并将其作为容器的背景。这在创建具有背景图片的复杂布局时非常有用。
在这里插入图片描述

1 alert
User
  createTwoButtonAlert = () => {
    Alert.alert('警告标题', '警告内容', [
      {text: '取消', onPress: () => console.log('Cancel'), style: 'cancel'},
      {text: '确定', onPress: () => console.log('Ok'), style: 'default'},
    ]);
  };
  createThreeButtonAlert = () => {
    Alert.alert('更新提示', '发现新版本是否现在更新?', [
      {text: '取消', onPress: () => console.log('Cancel'), style: 'cancel'},
      {text: '确定', onPress: () => console.log('Ok'), style: 'default'},
      {text: '稍后再试', onPress: () => console.log('稍后提醒我')},
    ]);
  };
  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15

在这里插入图片描述

import {View, Alert, Button, StyleSheet} from 'react-native';
import React, {Component} from 'react';

export default class index extends Component {
 // 创建一个包含两个按钮的警告框
createTwoButtonAlert = () => {
  // 调用 Alert.alert() 方法显示警告框
  Alert.alert(
    // 警告框标题
    '警告标题',
    // 警告框内容
    '警告内容',
    // 按钮配置数组
    [
      // 第一个按钮配置对象
      {
        // 按钮显示的文本
        text: '取消',
        // 按下按钮时触发的回调函数
        onPress: () => {
          // 在控制台输出 'Cancel',表示用户按下了取消按钮
          console.log('Cancel');
        },
        // 按钮样式(取消按钮)
        style: 'cancel'
      },
      // 第二个按钮配置对象
      {
        // 按钮显示的文本
        text: '确定',
        // 按下按钮时触发的回调函数
        onPress: () => {
          // 在控制台输出 'Ok',表示用户按下了确定按钮
          console.log('Ok');
        },
        // 按钮样式(默认按钮)
        style: 'default'
      },
    ]
  );
};

// 创建一个包含三个按钮的警告框
createThreeButtonAlert = () => {
  // 调用 Alert.alert() 方法显示警告框
  Alert.alert(
    // 警告框标题
    '更新提示',
    // 警告框内容
    '发现新版本是否现在更新?',
    // 按钮配置数组
    [
      // 第一个按钮配置对象
      {
        // 按钮显示的文本
        text: '取消',
        // 按下按钮时触发的回调函数
        onPress: () => {
          // 在控制台输出 'Cancel',表示用户按下了取消按钮
          console.log('Cancel');
        },
        // 按钮样式(取消按钮)
        style: 'cancel'
      },
      // 第二个按钮配置对象
      {
        // 按钮显示的文本
        text: '确定',
        // 按下按钮时触发的回调函数
        onPress: () => {
          // 在控制台输出 'Ok',表示用户按下了确定按钮
          console.log('Ok');
        },
        // 按钮样式(默认按钮)
        style: 'default'
      },
      // 第三个按钮配置对象
      {
        // 按钮显示的文本
        text: '稍后再试',
        // 按下按钮时触发的回调函数
        onPress: () => {
          // 在控制台输出 '稍后提醒我',表示用户按下了稍后再试按钮
          console.log('稍后提醒我');
        }
      },
    ]
  );
};

  //注意:Button里面不能写style
  render() {
    return (
      <View styles={[styles.container]}>
        <Button
          title="Alert"
          onPress={() => {
            alert('我是一个按钮');
          }}
        />
        <Button
          title="Alert.alert"
          onPress={() => {
            Alert.alert('我是一个按钮');
          }}
          color={'red'}
        />
        <Button
          title="两个按钮"
          onPress={this.createTwoButtonAlert}
          color={'blue'}
        />
        <Button
          title="三个按钮"
          onPress={this.createThreeButtonAlert}
          color={'tomato'}
        />
      </View>
    );
  }
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    justifyContent: 'space-around',
    alignItems: 'center',
  },
});

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • 101
  • 102
  • 103
  • 104
  • 105
  • 106
  • 107
  • 108
  • 109
  • 110
  • 111
  • 112
  • 113
  • 114
  • 115
  • 116
  • 117
  • 118
  • 119
  • 120
  • 121
  • 122
  • 123
  • 124
  • 125
  • 126
  • 127
  • 128
  • 129
  • 130
2 Switch

(1)写一个开关,默认不隐藏状态栏(state 初始化为 不隐藏),并赋值给switch,这样 状态栏通过 this.state.hideStatusBar 联动。
点击swift就会触发 onValueChange={this.toggleStatusBar} 进行一次 state值取反,重新赋值给 StatusBar 并改变状态栏。

import React, { Component } from 'react'
import { Text, StyleSheet, View, StatusBar, Switch } from 'react-native'
// constructor():这是类的构造函数,用于初始化类的实例。
// 在 React 组件中,构造函数主要用于初始化组件的状态(state)和绑定事件处理方法。
// super():这是调用父类(即 React.Component)的构造函数,确保在初始化子类时先调用父类的构造函数,以便继承父类的属性和方法。
// this.state = { hideStatusBar: false }:这是在组件的状态对象中,定义了一个名为 hideStatusBar 的状态属性,并将其初始值设为 false。
// 在 React 组件中,状态对象用于存储组件内部的数据,并且当状态发生变化时,React 会自动重新渲染组件以反映这些变化。

export default class index extends Component {

    constructor() {
        // 调用父类的构造函数
        super();
        // 初始化组件的状态
        this.state = {
            // 属性:hideStatusBar,类型:boolean,初始值为 false
            hideStatusBar: false
        };
    }


    // 写成一个箭头函数的形式 在类组件中想要改状态的值的话  只能 在箭头函数中操作 
    //setState() 是 React 组件类中的一个方法,用于更新组件的状态。
    //setState() 方法会将新的状态合并到当前状态中,然后触发组件的重新渲染。这样做是为了保持状态的不变性,以避免直接修改状态可能引起的问题。
    toggleStatusBar = () => {
        this.setState({
            hideStatusBar: !this.state.hideStatusBar //直接取反

        })
    }
    render() {
        return (
            <View>
                <Text>index</Text>

                <StatusBar
                    // hidden={false}
                    hidden={this.state.hideStatusBar}
                    //不是直接写死,而是 通过hide StatusBar来控制 状态栏
                    backgroundColor={"red"}//ios不奏效
                    barStyle={"dark-content"}

                />

                <Switch
                    trackColor={{ false: '#999', true: '#666' }}
                    // thumbColor={'blue'} 按钮的前景 
                    thumbColor={this.state.hideStatusBar? "green":"red"}
                   
                    //下面添加一个具体的事件  先初始化一个值 再探测值是否发生变化
                    value={this.state.hideStatusBar}
                //value 属性被设置为 this.state.hideStatusBar,意味着 Switch 组件的状态取决于 hideStatusBar 的值。当 hideStatusBar 的值发生变化时,Switch 组件也会相应地更新其状态。

//这样做的目的是为了将 Switch 组件的状态与组件类中的状态保持同步,以确保用户操作能够正确地反映到界面上。
                    onValueChange={this.toggleStatusBar}
                //  onValueChange是switch内置方法, 探测value值是否发生变化   
                //this.toggleStatusBar 函数被传递给了 onValueChange,意味着当值发生变化时,将调用 toggleStatusBar 函数,结果将hideStatusBar 拨动。
                />

            </View>



        )
    }
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
3 ActivityIndicator
import {StyleSheet, View, ActivityIndicator, Platform} from 'react-native';
import React from 'react';

//加载图标
export default function index() {
  if (Platform.OS === 'android') {
    alert('当前平台是安卓');
  } else if (Platform.OS === 'ios') {
    alert('当前平台是IOS');
  }
  return (
    <View style={[styles.container]}>
      <ActivityIndicator color={'blue'} size={'large'} />
      <ActivityIndicator color={'green'} size={'small'} />
      {/*数字指定大小只在安卓应用有效*/}
      <ActivityIndicator color={'#00d0ff'} size={70} />
      <ActivityIndicator color={'red'} size={100} />
    </View>
  );
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    justifyContent: 'space-around',
    alignItems: 'center',
  },
});

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
4 image
import {Text, StyleSheet, View, Image, Dimensions, ScrollView} from 'react-native';
import React, {Component} from 'react';

export default class index extends Component {
  render() {
    return (
      <ScrollView>  
      <View style={[styles.container]}>
     
        <Image
        style={[styles.itemImage]}
        source={require('./images/wdmpp.jpg')}
        
        />
        <Image
          style={styles.tinyLogo}
          source={{
            uri: 'https://reactnative.dev/img/tiny_logo.png',
          }}
        />
        <Image
          style={styles.logo}
          source={{
            uri: '',
          }}
        />
      </View>
      </ScrollView>
    );
  }
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    justifyContent: 'center',
    alignItems: 'center',
  },
  itemImage: {
    height: Dimensions.get("window").height,
    width: Dimensions.get('window').width,
    marginVertical: 20,
  },
  tinyLogo: {
    width: 100,
    height: 100,
    marginVertical: 20,
  },
  logo: {
    width: 66,
    height: 58,
  },
});

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
5 textinput

在React Native中,TextInput组件提供了一系列属性来控制文本输入框的行为和样式。以下是一些常用的TextInput属性及其介绍:

  1. value: 设置或获取用户往文本输入框中输入的值。
  2. onChangeText: 当文本输入框的值发生变化时,调用的回调函数。
  3. placeholder: 设置文本输入框的占位符文本,当输入框为空时显示“请输入用户名”。
  4. placeholderTextColor: 设置占位符文本的颜色。
  5. editable: 设置文本输入框是否可编辑。
  6. multiline: 设置文本输入框是否支持多行输入。
  7. numberOfLines: 当multilinetrue时,设置文本输入框的行数。
  8. secureTextEntry: 设置文本输入框是否用于输入密码,当为true时输入内容会被隐藏。
  9. autoCapitalize: 设置文本输入框的自动大写风格,可选值包括nonesentenceswordscharacters
  10. autoCorrect: 设置文本输入框是否启用自动纠正功能。
  11. autoFocus: 设置文本输入框是否自动获取焦点。
  12. keyboardType: 设置文本输入框对应的键盘类型,如defaultnumericemail-address等。
  13. returnKeyType: 设置键盘上返回键的类型,如donegosearch等。
  14. onFocus: 文本输入框获取焦点时调用的回调函数。
  15. onBlur: 文本输入框失去焦点时调用的回调函数。
  16. onSubmitEditing: 当用户提交编辑时调用的回调函数。
  17. style: 设置文本输入框的样式。
  18. maxLength: 设置文本输入框可输入的最大字符数。
  19. keyboardAppearance: 设置键盘的外观,如defaultdarklight
  20. clearButtonMode: 设置清除按钮的显示模式,如neverwhile-editingunless-editingalways

闲聊回调函数
回调函数就像是一份待办事项清单上的任务。当某个任务完成时,你会去执行它,而不是在开始时就执行。在编程中,这个“任务”就是函数,而“完成”则意味着某个事件发生了或某个操作完成了。

在React Native中,当用户输入文本或点击按钮时,你想要执行的操作可能会有所不同,这就是回调函数的用武之地。你可以告诉React Native:“当用户输入文本时,请执行这个函数”,或者:“当用户点击按钮时,请执行另一个函数”。这些函数就是回调函数,它们会在特定的事件发生时被调用,以执行你预先定义的操作。

/* eslint-disable no-alert */
import { StyleSheet, View, TextInput, Dimensions, Button } from 'react-native';
import React, { Component } from 'react';

export default class index extends Component {

  constructor() {

    // まず、値を取得するために状態を使用できます.なので、コンストラクタ を書きます
    //コンストラクタ中に 状態を初期化します
    //状態は set state というメソッドを使用してのみ 変更できます .その値は
    super()
    this.state = {
      username: ''

    }

  }

  doLogin = () => {
    // フォーム中のデータを取得(しゅとく)するには、3つ(みっつ)の方法(ほうほう)があります。

// onChangeValueハンドラーでテキスト入力の変更を監視(かんし)し、set.state.valueを使って状態中にのusernameを更新します。
// そして、状態のusername値をdoLoginのusername変数に代入(だいにゅう)し、ポップアップ(popーUP)で表示します。
    alert(this.state.username)

  }
  render() {
    return (
      <View style={[styles.container]}>
        <TextInput
          //  入力ファームを作成
          style={[styles.input]}
          placeholder='请输入狗狗的名字'
          value={this.state.username}

          onChangeText={
            // val パラメーター

            (val) => {
              this.setState({
                username: val
              })

            }
          }

        // このコールバック関数は、テキストが変更されるたびに 後続(こうぞく)のイベントがトリガーされることを意味します
        // 这个回调函数意味着每当文本发生变化时,后续事件就会被触发。


        />
        <View>
          {/* // viewタグを作成し、その中にbuttonボタンを配置して、ファームの送信機能を実装します。 */}
          {/* // 在react中 写一个view标签 ,view标签中写一个button按钮  ,实现提交表单功能 */}
          <Button title='登陆' onPress={this.doLogin} />
          {/* 点击按钮的时候,触发onpress这个方法   进行了表单提交 
      ボタンをクリックした時に、 on Pressメソッドをトリガーして ファームの送信を行います */}
        </View>

      </View>

    );
  }
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    justifyContent: 'center',
  },
  input: {
    width: Dimensions.get('window').width - 20,
    margin: 10,
    borderWidth: 1,
    borderColor: 'red',
    paddingHorizontal: 5,
  },
  btn: {
    margin: 10,
    // borderWidth: 1,

  },
});

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
6 TouchableHighlight
/* eslint-disable no-alert */
import {
  Text,
  StyleSheet,
  View,
  TouchableHighlight,
  TouchableOpacity,
  TouchableWithoutFeedback,
} from 'react-native';
import React, {Component} from 'react';

export default class index extends Component {
  constructor() {
    super();
    this.state = {
      highlightColor: 'black',
    };
  }
  TouchableHighlightPress = () => {
    //alert('触碰高亮显示');
    this.setState({highlightColor: 'white'});
  };
  TouchableHighlightPressOut = () => {
    //alert('触碰高亮显示PressOut');
    this.setState({highlightColor: 'black'});
  };
  render() {
    return (
      <View style={[styles.container]}>
        <React.Fragment>
          <TouchableHighlight
            underlayColor="blue"
            onFocus={this.TouchableHighlightPress}
            onPress={this.TouchableHighlightPress}
            onLongPress={this.TouchableHighlightPress}
            onHideUnderlay={this.TouchableHighlightPressOut}>
            <View style={[styles.Item]}>
              <Text style={{color: this.state.highlightColor}}>
                触碰高亮显示
              </Text>
            </View>
          </TouchableHighlight>
          <TouchableOpacity onPress={() => alert('触碰透明度变化')}>
            <View style={[styles.Item]}>
              <Text>触碰透明度变化</Text>
            </View>
          </TouchableOpacity>
          <TouchableWithoutFeedback onPress={() => alert('触碰无响应')}>
            <View style={[styles.Item]}>
              <Text>触触碰无响应</Text>
            </View>
          </TouchableWithoutFeedback>
        </React.Fragment>
      </View>
    );
  }
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    justifyContent: 'center',
    alignItems: 'center',
  },
  Item: {
    marginBottom: 20,
    padding: 10,
    borderWidth: 1,
    borderColor: 'red',
  },
  textColor: {
    color: 'black',
  },
});

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
7 ScrollView


/* eslint-disable react-native/no-inline-styles */
import { StyleSheet, Text, View, ScrollView, Platform } from 'react-native';
import React, { Component } from 'react';

export default class index extends Component {
  render() {
    // showsVerticalScrollIndicator 隐藏垂直方向的滚动条
    //showsHorizontalScrollIndicator 隐藏水平方向的滚动条
    // SafeAreaView 组件在有刘海屏的手机上能够避开刘海屏显示
    return (
      <View>
        <ScrollView style={{ backgroundColor: '#dfb' }} horizontal={true}>
          <Text style={[styles.nav]}>新闻</Text>
          <Text style={[styles.nav]}>娱乐</Text>
          <Text style={[styles.nav]}>体育</Text>
          <Text style={[styles.nav]}>财经</Text>
          <Text style={[styles.nav]}>军事</Text>
          <Text style={[styles.nav]}>时尚</Text>
          <Text style={[styles.nav]}>科技</Text>
        </ScrollView>
        <ScrollView
          style={[styles.scrollView]}
          // react-native/no-inline-styles
          contentContainerStyle={{ margin: 30 }}
          showsHorizontalScrollIndicator={true}
          showsVerticalScrollIndicator={true}>
          <Text style={[styles.text]}>
            《三字经》 

            人之初,性本善。性相近,习相远。
            苟不教,性乃迁。教之道,贵以专。

            昔孟母,择邻处。子不学,断机杼。
            窦燕山,有义方。教五子,名俱扬。

            养不教,父之过。教不严,师之惰。
            子不学,非所宜。幼不学,老何为。

            玉不琢,不成器。人不学,不知义。
            为人子,方少时。亲师友,习礼仪。

            香九龄,能温席。孝于亲,所当执。
            融四岁,能让梨。弟于长,宜先知。

            首孝弟,次见闻。知某数,识某文。
            一而十,十而百。百而千,千而万。




          </Text>
          {/*解决ScrollView在安卓下滚动不到底的问题*/}
          {/*// eslint-disable-next-line react-native/no-inline-styles*/}
          {/*   以上のように、このコードの目的は、異なる設備でscroll viewの高さを制定し、iOS設備でscroll viewを非表示にすることです */}
          <View style={{ height: Platform.OS === 'ios' ? 0 : 200 }} />
        </ScrollView>
      </View>  
    );
  }
}

const styles = StyleSheet.create({
  text: {
    fontSize: 30,
  },
  scrollView: {
    backgroundColor: '#ddd',
    marginHorizontal: 30,
  },
  nav: {
    margin: 10,
    height: 50,
    fontSize: 30,
  },
});

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
8 SectionList
  1. sections:必需属性,用于指定列表的数据源,通常是一个包含多个对象的数组,每个对象代表一个分组,包含 data(该分组的数据) 和可选的 title(该分组的标题)。

  2. renderItem:必需属性,用于指定渲染列表项的函数,函数接收一个包含 itemindex 属性的对象作为参数,用于渲染列表中的每一项。

  3. renderSectionHeader:可选属性,用于指定渲染分组头部的函数,函数接收一个包含 section 属性的对象作为参数,用于渲染每个分组的头部。

  4. ItemSeparatorComponent:可选属性,用于指定列表项之间的分隔组件,通常用于添加分割线或其他样式。

  5. ListEmptyComponent:可选属性,用于指定当列表数据为空时渲染的组件,通常用于显示空列表时的占位内容。

  6. refreshing:可选属性,用于指定列表是否处于刷新状态,通常与 onRefresh 属性一起使用。

  7. onRefresh:可选属性,用于指定下拉刷新时触发的函数,通常用于请求最新的数据并更新列表内容。

  8. onEndReachedThreshold:可选属性,用于指定列表滚动到底部时触发加载更多数据的阈值,通常设置为一个小于 1 的值,表示列表滚动到底部还剩余多少比例时触发加载更多。

  9. onEndReached:可选属性,用于指定列表滚动到底部时触发的函数,通常用于请求更多的数据并追加到列表末尾。

  10. ListHeaderComponent:可选属性,用于指定列表头部的组件,通常用于显示额外的信息或操作。

  11. ListFooterComponent:可选属性,用于指定列表尾部的组件,通常用于显示加载更多的提示信息或加载动画。

  12. keyExtractor:可选属性,用于指定用于唯一标识列表项的属性,默认为 key

这段代码是一个使用React Native的SectionList组件实现的三国英雄列表展示,包含了下拉刷新和上拉加载更多的功能。

/* eslint-disable react/no-unstable-nested-components */
/* eslint-disable no-alert */
/* eslint-disable react-native/no-inline-styles */
import {Text, StyleSheet, View, SafeAreaView, SectionList} from 'react-native';
import React, {Component} from 'react';

// 数据源
const DATA = [
  {
    title: '魏国',
    data: ['曹操', '司马懿', '张辽'],
  },
  {
    title: '蜀国',
    data: ['刘备', '关羽', '张飞'],
  },
  {
    title: '吴国',
    data: ['孙权', '周瑜', '黄盖'],
  },
];

// 列表项组件
const Item = ({title}) => (
  <View style={styles.item}>
    <Text style={styles.title}>{title}</Text>
  </View>
);

class Index extends Component {
  constructor() {
    super();
    this.state = {
      isRefresh: false, // 用于标记是否正在刷新
    };
  }

  // 模拟加载数据的函数
  loadData = () => {
    // 开启加载动画
    this.setState({
      isRefresh: true,
    });

    // 模拟请求数据
    setTimeout(() => {
      this.setState({
        isRefresh: false,
      });
    }, 2000);
    alert('下拉刷新');
  };

  render() {
    return (
      <SafeAreaView>
        <SectionList
          sections={DATA} // 数据源
          keyExtractor={(item, index) => item + index} // 为每个列表项生成唯一的key
          renderItem={({item}) => <Item title={item} />} // 渲染列表项
          renderSectionHeader={({section: {title}}) => (
            <Text style={styles.header}>{title}</Text> // 渲染分组头部
          )}
          ItemSeparatorComponent={() => {
            // 声明项目之间的分隔符
            return (
              <View style={{borderBottomWidth: 1, borderBottomColor: 'red'}} />
            );
          }}
          ListEmptyComponent={() => {
            // 当列表数据为空时展示的组件
            return <Text style={{fontSize: 30}}>空空如也</Text>;
          }}
          refreshing={this.state.isRefresh} // 控制下拉刷新状态
          onRefresh={() => {
            // 下拉刷新触发的函数
            this.loadData();
          }}
          onEndReachedThreshold={0.1} // 设置列表底部触发加载更多的阈值
          onEndReached={() => {
            // 上拉加载更多触发的函数
            alert('到底了');
          }}
          ListHeaderComponent={() => {
            // 列表头部组件
            return <Text style={{fontSize: 40}}>三国英雄榜</Text>;
          }}
          ListFooterComponent={() => {
            // 列表尾部组件
            return <Text style={{fontSize: 30}}>没有更多了</Text>;
          }}
        />
      </SafeAreaView>
    );
  }
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    marginTop: 50,
    marginHorizontal: 16,
  },
  item: {
    backgroundColor: '#f9c2ff',
    padding: 20,
    marginVertical: 8,
  },
  header: {
    fontSize: 32,
    backgroundColor: '#fff',
  },
  title: {
    fontSize: 24,
  },
});

export default Index;


  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • 101
  • 102
  • 103
  • 104
  • 105
  • 106
  • 107
  • 108
  • 109
  • 110
  • 111
  • 112
  • 113
  • 114
  • 115
  • 116
  • 117
  • 118
  • 119
  • 120
9 FlatList

FlatList 组件是 React Native 中用于呈现列表数据的高性能组件,它适用于大型数据集。以下是 FlatList 组件的一些常用属性:

  1. data:要渲染的列表数据数组。
  2. renderItem:用于渲染列表中每个项目的函数。
  3. keyExtractor:用于从数据中提取唯一标识符的函数。
  4. ListHeaderComponent:列表的头部组件,会在列表头部渲染。
  5. ListFooterComponent:列表的尾部组件,会在列表尾部渲染。
  6. ItemSeparatorComponent:用于渲染在列表项之间的分隔符的组件。
  7. horizontal:指定列表是否水平滚动。
  8. numColumns:指定在多列布局中每行的列数。
  9. initialNumToRender:指定初始渲染的项目数量。
  10. onEndReached:滚动到列表底部时调用的函数。
  11. onEndReachedThreshold:触发 onEndReached 事件的阈值,即距离列表底部多远时触发。
  12. refreshing:指定列表是否处于刷新状态。
  13. onRefresh:下拉刷新时调用的函数。
  14. refreshControl:自定义刷新指示器的组件。
/* eslint-disable react-native/no-inline-styles */
/* eslint-disable react/no-unstable-nested-components */
/* eslint-disable no-alert */
import {
  Text,
  StyleSheet,
  View,
  SafeAreaView,
  FlatList,
  TouchableOpacity,
} from 'react-native';
import React, { Component } from 'react';

export default class Index extends Component {
  constructor() {
    super();
    this.state = {
      isLoading: false,
      selectedId: null,
      list: [
        {
          id: '1',
          title: '麻辣贝贝火锅'
        },
        {
          id: '2',
          title: '孜然烤贝贝串'
        },
        {
          id: '3',
          title: '红烧贝贝面'
        },
        {
          id: '4',
          title: '酱爆贝贝丝'
        },
        {
          id: '5',
          title: '水煮贝贝'
        },
        {
          id: '6',
          title: '蒜蓉炒贝贝'
        },
        {
          id: '7',
          title: '黑椒贝贝粒'
        },
        {
          id: '8',
          title: '酸辣贝贝汤'
        },
        {
          id: '9',
          title: '梅菜扣贝贝'
        },
        {
          id: '10',
          title: '椒盐贝贝块'
        },
        {
          id: '11',
          title: '川味贝贝炒年糕'
        },
        {
          id: '12',
          title: '香菇贝贝炖饭'
        },
        {
          id: '13',
          title: '柠檬贝贝片'
        },
        {
          id: '14',
          title: '肉夹馍(夹贝贝)'
        },
        {
          id: '15',
          title: '葱爆贝贝'
        },
        {
          id: '16',
          title: '鱼香茄子煲贝贝'
        },
        {
          id: '17',
          title: '干锅贝贝'
        },
        {
          id: '18',
          title: '糖醋贝贝'
        },
        {
          id: '19',
          title: '红烧贝贝丸子'
        },
        {
          id: '20',
          title: '酸菜贝贝面'
        }
      ]
      ,
    };
  }
  renderItem = ({ index, item }) => {
    console.log(item);
    const backgroundColor =
      item.id === this.state.selectedId ? '#dfb' : '#f9c2ff';
    return (
      <TouchableOpacity
        style={[styles.item, { backgroundColor }]}
        onPress={() => {
          this.setState({ selectedId: item.id });
        }}>
        <Text style={[styles.title]}>{item.title}</Text>
      </TouchableOpacity>
    );
  };
  loadData = () => {
    this.setState({
      isLoading: true,
    });
    setTimeout(() => {
      //模拟请求数据
      alert('刷新请求数据');
      this.setState({ isLoading: false });
    }, 2000);
  };
  render() {
    return (

      <View>
        <Text></Text>
        <Text></Text>

        <Text style = { { fontSize:40}}  >可爱玉林狗</Text>
        <SafeAreaView>

          <FlatList
            data={this.state.list}
            renderItem={this.renderItem}
            keyExtractor={item => item.id}
            horizontal={false} //是否水平布局模式
            initialScrollIndex={0} //初始化时滚动的索引位置
            initialNumToRender={20} // 指定初始渲染数据的数量,一般数量要填满一屏幕
            numColumns={2} // 设置展示多少列 数据项必须等高,无法支持瀑布流
            inverted={false} //反转列表
            extraData={this.state.selectedId}
            ItemSeparatorComponent={() => {
              //声明项目之间的分割符
              return <View style={[styles.itemSeparator]} />;
            }}
            ListEmptyComponent={() => {
              //列表数据为空时展示组件
              return <Text style={{ fontSize: 30 }}>空空如也</Text>;
            }}
            //下拉刷新
            refreshing={this.state.isLoading}
            onRefresh={this.loadData}
            // 上拉加载
            // onEndReachedThreshold设置列表触底还剩多少刷新0.1表示列表还剩10%的时候刷新
            onEndReachedThreshold={0.1}
            onEndReached={() => {
              //此处为上拉加载的具体逻辑代码
              // alert('到底了');
            }}
            ListHeaderComponent={() => {
              //声明列表的头部组件
              return <Text style={[styles.header]}>列表头部</Text>;
            }}
            ListFooterComponent={() => {
              //声明列表的头部组件
              return <Text style={[styles.footer]}>没有更多了</Text>;
            }}
          />
        </SafeAreaView>
      </View>
    );
  }
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    marginTop: 50,
    marginHorizontal: 16,
  },
  item: {
    backgroundColor: '#f9c2ff',
    height: 100,
    width: '50%',
    //fontSize: 55,
    textAlign: 'cenleftter',
    verticalAlign: 'middle',
    justifyContent: 'center',
    alignItems: 'left',
    marginVertical: 8,
    marginHorizontal: 8,
  },
  header: {
    fontSize: 10,
    backgroundColor: '#fff',
  },
  title: {
    fontSize: 24,
  },
  itemSeparator: {
    borderBottomWidth: 1,
    borderBottomColor: 'red',
  },
  footer: { fontSize: 30, textAlign: 'center' },
});

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • 101
  • 102
  • 103
  • 104
  • 105
  • 106
  • 107
  • 108
  • 109
  • 110
  • 111
  • 112
  • 113
  • 114
  • 115
  • 116
  • 117
  • 118
  • 119
  • 120
  • 121
  • 122
  • 123
  • 124
  • 125
  • 126
  • 127
  • 128
  • 129
  • 130
  • 131
  • 132
  • 133
  • 134
  • 135
  • 136
  • 137
  • 138
  • 139
  • 140
  • 141
  • 142
  • 143
  • 144
  • 145
  • 146
  • 147
  • 148
  • 149
  • 150
  • 151
  • 152
  • 153
  • 154
  • 155
  • 156
  • 157
  • 158
  • 159
  • 160
  • 161
  • 162
  • 163
  • 164
  • 165
  • 166
  • 167
  • 168
  • 169
  • 170
  • 171
  • 172
  • 173
  • 174
  • 175
  • 176
  • 177
  • 178
  • 179
  • 180
  • 181
  • 182
  • 183
  • 184
  • 185
  • 186
  • 187
  • 188
  • 189
  • 190
  • 191
  • 192
  • 193
  • 194
  • 195
  • 196
  • 197
  • 198
  • 199
  • 200
  • 201
  • 202
  • 203
  • 204
  • 205
  • 206
  • 207
  • 208
  • 209
  • 210
  • 211
  • 212
  • 213
10 Animated

在 React Native 中,Animated 是一个内置的动画库,它提供了一组组件和函数,用于创建各种动画效果。

特点:
  • 高性能Animated 使用原生动画驱动来执行动画,因此具有良好的性能。

  • 易于使用:提供了简单易用的 API,使得创建和控制动画变得简单。

  • 支持多种动画效果:可以实现平移、缩放、旋转、透明度变化等各种动画效果。

  • 适用性广泛:适用于各种组件和场景,包括 View、Text、Image 等。

通过 Animated,开发者可以轻松地为 React Native 应用添加动画效果,提升用户体验,增强应用的吸引力和交互性。

React Native 中的动画组件

在 React Native 中,有几个直接可用的动画组件,它们包括:

  1. Animated.View:用于创建一个可以进行动画操作的视图组件。可以通过 Animated.View 实现平移、缩放、旋转、透明度等动画效果。

  2. Animated.Text:类似于普通的文本组件,但可以通过 Animated API 来实现文本的动画效果,如透明度变化、颜色变化等。

  3. Animated.ScrollView:基于 ScrollView 组件,但可以使用 Animated API 来创建滚动动画效果,包括滚动到指定位置、滚动速度控制等。

  4. Animated.Image:类似于普通的图片组件,但可以通过 Animated API 来实现图片的动画效果,如透明度变化、位置移动等。

  5. Animated.FlatList:与普通的 FlatList 类似,但可以使用 Animated API 来创建列表项的动画效果,如渐变、平移等。

  6. Animated.SectionList:类似于 FlatList,但专门用于显示分组数据。可以通过 Animated API 实现分组列表项的动画效果。

  7. Animated.TouchableOpacity:与 TouchableOpacity 类似,但可以使用 Animated API 实现按钮的动画效果,如缩放、颜色变化等。

  8. Animated.TextInput:与 TextInput 类似,但可以使用 Animated API 实现输入框的动画效果,如聚焦时的放大效果、边框颜色变化等。

  9. 在这里插入图片描述


/* eslint-disable no-alert */
import {
  Animated,
  Text,
  StyleSheet,
  View,
  SafeAreaView,
  Button,
  Image,
} from 'react-native';
import React, { Component } from 'react';

export default class Index extends Component {
  // fadeAnim 将透明度设置为0
  state = {
    fadeAnim: new Animated.Value(0),
    moveAnim: new Animated.Value(0),
  };
  fadeIn = () => {
    Animated.timing(this.state.fadeAnim, {
      toValue: 1, //目标值
      useNativeDriver: true, //启用原生方式,渲染动画(执行效率会更高)
      duration: 1, // 动画执行时间
    }).start(() => {
      // 动画执行结束后的回调函数
      alert('请吃贝贝');
    });
  };

  fadeOut = () => {
    // Will change fadeAnim value to 0 in 3 seconds
    Animated.timing(this.state.fadeAnim, {
      toValue: 0,
      duration: 2,
      useNativeDriver: true,
    }).start(() => {
      // 动画执行结束后的回调函数
      alert('我藏起来了');
    });
  };
  componentDidMount() {
    //组件加载后触发动画
    this.scanMove();
  }
  scanMove = () => {
    // 将moveAnim的初始值设置为0
    this.state.moveAnim.setValue(0);
    Animated.timing(this.state.moveAnim, {
      toValue: 200,
      duration: 1000,
      useNativeDriver: true,
    }).start(() => {
      this.scanMove();
    });
  };
  render() {
    return (
      <SafeAreaView style={styles.container}>
        <Animated.View
          style={[
            styles.fadingContainer,
            {
              // Bind opacity to animated value
              opacity: this.state.fadeAnim,
            },
          ]}>
          <Text style={styles.fadingText}> 做好了</Text>
          
          <Image
            source={require('../src_13_Animated/eatbb.jpg')}
            style={[styles.backgroundImage, { width: 100, height: 100 }]} // 修改宽高为100x100
          />
          

        </Animated.View>
        <View style={styles.buttonRow}>
          <Button title="做成菜" onPress={this.fadeIn} />
          <Button title="送去玉林" onPress={this.fadeOut} />
        </View>
        <View style={[styles.scanContainer]}>

          <Animated.View

            style={[
              styles.border,
              {
                transform: [
                  {
                    translateY: this.state.moveAnim,
                  },
                ],
              },
            ]}
          />
          <Image
            source={require('../src_13_Animated/bb.jpg')} // 背景图片路径
            style={styles.backgroundImage}
          />
        </View>
      </SafeAreaView>
    );
  }
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    alignItems: 'center',
    justifyContent: 'center',
  },
  fadingContainer: {
    padding: 20,
    backgroundColor: 'powderblue',
  },
  fadingText: {
    fontSize: 15,
    backgroundColor:'yellow'
  },
  buttonRow: {
    flexBasis: 100,
    justifyContent: 'space-evenly',
    marginVertical: 16,
  },
  // scanContainer: {
  //   height: 200,
  //   width: 200,
  //   borderWidth: 1,
  //   borderColor: 'green',
  // },
  border: {
    borderWidth: 1,
    borderColor: 'red',
  },

  // ----------
  scanContainer: {
    position: 'relative', // 确保容器视图相对定位
    height: 200,
    width: 200,
    borderWidth: 1,
    borderColor: 'green',
  },
  // backgroundImage: {
  //   position: 'absolute', // 确保背景图片绝对定位
  //   top: 0,
  //   left: 0,
  //   width: '70%', // 图片宽度与容器宽度相同
  //   height: '100%', // 图片高度与容器高度相同
  //   resizeMode: 'cover', // 图片裁剪方式
  //   zIndex: -1, // 将背景图片放置在视图的底层
  // },
  backgroundImage: {
    position: 'absolute',
    top: 0,
    left: 0,
    width: '100%',
    height: '100%',
    resizeMode: 'contain', // 修改为contain
    zIndex: -1,
  },
  
});


  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 59
  • 60
  • 61
  • 62
  • 63
  • 64
  • 65
  • 66
  • 67
  • 68
  • 69
  • 70
  • 71
  • 72
  • 73
  • 74
  • 75
  • 76
  • 77
  • 78
  • 79
  • 80
  • 81
  • 82
  • 83
  • 84
  • 85
  • 86
  • 87
  • 88
  • 89
  • 90
  • 91
  • 92
  • 93
  • 94
  • 95
  • 96
  • 97
  • 98
  • 99
  • 100
  • 101
  • 102
  • 103
  • 104
  • 105
  • 106
  • 107
  • 108
  • 109
  • 110
  • 111
  • 112
  • 113
  • 114
  • 115
  • 116
  • 117
  • 118
  • 119
  • 120
  • 121
  • 122
  • 123
  • 124
  • 125
  • 126
  • 127
  • 128
  • 129
  • 130
  • 131
  • 132
  • 133
  • 134
  • 135
  • 136
  • 137
  • 138
  • 139
  • 140
  • 141
  • 142
  • 143
  • 144
  • 145
  • 146
  • 147
  • 148
  • 149
  • 150
  • 151
  • 152
  • 153
  • 154
  • 155
  • 156
  • 157
  • 158
  • 159
  • 160
  • 161
  • 162
  • 163
  • 164
  • 165

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

在这里插入图片描述

下一节,我们会接着介绍其他常用组件

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/我家小花儿/article/detail/474626
推荐阅读
相关标签
  

闽ICP备14008679号