在React Native中开发进度条组件,可以使用多种方式来实现,包括使用内置的`ProgressViewharmony`(仅限harmony)

在React Native中开发进度条组件,你可以使用多种方式来实现,包括使用内置的ProgressViewharmony(仅限harmony),或者使用第三方库如react-native-progress来跨平台支持harmony和harmony。下面,我将介绍如何使用react-native-progress库来创建一个进度条组件。

  1. 安装react-native-progress

首先,你需要通过npm或yarn来安装react-native-progress库。打开你的终端或命令提示符,然后运行以下命令:

bash 复制代码
npm install react-native-progress
或者
yarn add react-native-progress
  1. 使用react-native-progress创建进度条

安装完成后,你可以在你的React Native项目中引入并使用ProgressBar组件。下面是一个基本的示例:

javascript 复制代码
import React from 'react';
import { View, StyleSheet } from 'react-native';
import { ProgressBar } from 'react-native-progress';

const ProgressBarComponent = ({ progress }) => {
  return (
    <View style={styles.container}>
      <ProgressBar progress={progress} width={200} color="3498db" />
    </View>
  );
};

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

export default ProgressBarComponent;

在上面的代码中,ProgressBar组件接受一个progress属性,该属性是一个从0到1的数字,表示进度条的完成度。你可以根据需要调整widthcolor属性来改变进度条的宽度和颜色。

  1. 使用示例

你可以在任何组件中引入并使用ProgressBarComponent,如下所示:

javascript 复制代码
import React, { useState, useEffect } from 'react';
import ProgressBarComponent from './ProgressBarComponent'; // 确保路径正确

const App = () => {
  const [progress, setProgress] = useState(0);

  useEffect(() => {
    const timer = setInterval(() => {
      setProgress(prevProgress => {
        if (prevProgress >= 1) return 1; // 防止超过100%
        const newProgress = prevProgress + 0.01; // 增加进度,可以根据需要调整速度和增量
        return newProgress;
      });
    }, 100); // 每100毫秒更新一次进度,可以根据需要调整速度

    return () => clearInterval(timer); // 清理定时器以避免内存泄漏
  }, []);

  return (
    <View style={{ flex: 1, justifyContent: 'center', alignItems: 'center' }}>
      <ProgressBarComponent progress={progress} />
    </View>
  );
};

export default App;

在这个示例中,我们使用了React的useStateuseEffect钩子来创建一个动态更新的进度条。通过定时器,我们每隔一段时间更新进度值,从而模拟一个加载过程。你可以根据实际需求调整定时器的频率和进度增加的速率。


真实项目代码演示案例:

js 复制代码
// App.tsx
import React, { useState, useEffect } from 'react';
import { 
  View, 
  Text, 
  StyleSheet, 
  ScrollView, 
  SafeAreaView,
  Image,
  Dimensions,
  TouchableOpacity
} from 'react-native';

// Base64 Icons for progress components
const PROGRESS_ICONS = {
  download: '......',
  upload: '......',
  success: '......',
  error: '......',
  refresh: '......'
};

// 进度条组件
interface ProgressProps {
  percent: number;
  status?: 'normal' | 'active' | 'success' | 'exception';
  showInfo?: boolean;
  strokeWidth?: number;
  strokeColor?: string;
  trailColor?: string;
  format?: (percent: number) => string;
  type?: 'line' | 'circle';
  width?: number;
}

const Progress: React.FC<ProgressProps> = ({
  percent = 0,
  status = 'normal',
  showInfo = true,
  strokeWidth = 10,
  strokeColor,
  trailColor = '#e2e8f0',
  format,
  type = 'line',
  width = 120
}) => {
  const getStatusColor = () => {
    switch (status) {
      case 'success':
        return '#10b981';
      case 'exception':
        return '#ef4444';
      case 'active':
        return '#3b82f6';
      default:
        return strokeColor || '#3b82f6';
    }
  };

  const getColor = getStatusColor();
  
  const displayPercent = Math.min(100, Math.max(0, percent));
  const formattedText = format ? format(displayPercent) : `${Math.round(displayPercent)}%`;

  if (type === 'circle') {
    const radius = (width - strokeWidth) / 2;
    const circumference = 2 * Math.PI * radius;
    const strokeDashoffset = circumference - (displayPercent / 100) * circumference;
    
    return (
      <View style={styles.circleProgressContainer}>
        <View style={styles.circleWrapper}>
          <Svg width={width} height={width}>
            <Circle
              cx={width / 2}
              cy={width / 2}
              r={radius}
              stroke={trailColor}
              strokeWidth={strokeWidth}
              fill="none"
            />
            <Circle
              cx={width / 2}
              cy={width / 2}
              r={radius}
              stroke={getColor}
              strokeWidth={strokeWidth}
              fill="none"
              strokeDasharray={`${circumference} ${circumference}`}
              strokeDashoffset={strokeDashoffset}
              strokeLinecap="round"
              transform={`rotate(-90 ${width / 2} ${width / 2})`}
            />
          </Svg>
          {showInfo && (
            <View style={styles.circleInfo}>
              <Text style={[styles.circleText, { color: getColor }]}>{formattedText}</Text>
            </View>
          )}
        </View>
      </View>
    );
  }

  return (
    <View style={styles.progressContainer}>
      <View style={styles.progressBar}>
        <View 
          style={[
            styles.progressTrail,
            { 
              height: strokeWidth,
              backgroundColor: trailColor,
              borderRadius: strokeWidth / 2
            }
          ]}
        >
          <Animated.View 
            style={[
              styles.progressFill,
              {
                width: `${displayPercent}%`,
                height: strokeWidth,
                backgroundColor: getColor,
                borderRadius: strokeWidth / 2
              }
            ]}
          />
        </View>
      </View>
      {showInfo && (
        <View style={styles.progressInfo}>
          <Text style={[styles.progressText, { color: getColor }]}>{formattedText}</Text>
        </View>
      )}
    </View>
  );
};

// 由于React Native不支持SVG,我们需要简化圆形进度条的实现
const CircleProgress: React.FC<{
  percent: number;
  size?: number;
  strokeWidth?: number;
  color?: string;
  bgColor?: string;
  showInfo?: boolean;
}> = ({ 
  percent, 
  size = 120, 
  strokeWidth = 10, 
  color = '#3b82f6', 
  bgColor = '#e2e8f0',
  showInfo = true
}) => {
  const displayPercent = Math.min(100, Math.max(0, percent));
  const radius = (size - strokeWidth) / 2;
  const circumference = 2 * Math.PI * radius;
  const strokeDashoffset = circumference - (displayPercent / 100) * circumference;
  
  return (
    <View style={[styles.circleProgress, { width: size, height: size }]}>
      <View style={[styles.circleBg, { 
        width: size, 
        height: size, 
        borderRadius: size / 2,
        borderWidth: strokeWidth,
        borderColor: bgColor
      }]} />
      <View style={[
        styles.circleFill, 
        { 
          width: size, 
          height: size, 
          borderRadius: size / 2,
          borderWidth: strokeWidth,
          borderColor: color,
          transform: [{ rotate: '-90deg' }],
          borderLeftColor: 'transparent',
          borderBottomColor: 'transparent',
          borderRightColor: 'transparent'
        }
      ]} />
      {showInfo && (
        <View style={[styles.circleInfo, { width: size, height: size }]}>
          <Text style={[styles.circleText, { color }]}>{Math.round(displayPercent)}%</Text>
        </View>
      )}
    </View>
  );
};

// 主应用组件
const App = () => {
  const [progressValues, setProgressValues] = useState([
    { id: 1, percent: 30, status: 'normal', label: '文件下载' },
    { id: 2, percent: 65, status: 'active', label: '数据上传' },
    { id: 3, percent: 100, status: 'success', label: '任务完成' },
    { id: 4, percent: 45, status: 'exception', label: '处理错误' }
  ]);
  
  const [dynamicProgress, setDynamicProgress] = useState(0);
  const [isRunning, setIsRunning] = useState(false);

  useEffect(() => {
    let interval: NodeJS.Timeout;
    
    if (isRunning) {
      interval = setInterval(() => {
        setDynamicProgress(prev => {
          if (prev >= 100) {
            setIsRunning(false);
            return 100;
          }
          return prev + 1;
        });
      }, 100);
    }
    
    return () => clearInterval(interval);
  }, [isRunning]);

  const startProgress = () => {
    if (!isRunning) {
      setIsRunning(true);
      setDynamicProgress(0);
    }
  };

  const resetProgress = () => {
    setIsRunning(false);
    setDynamicProgress(0);
  };

  return (
    <SafeAreaView style={styles.container}>
      <View style={styles.header}>
        <Text style={styles.headerTitle}>进度条组件演示</Text>
        <Text style={styles.headerSubtitle}>多种样式和状态展示</Text>
      </View>
      
      <ScrollView contentContainerStyle={styles.contentContainer}>
        <View style={styles.section}>
          <Text style={styles.sectionTitle}>线性进度条</Text>
          {progressValues.map(item => (
            <View key={item.id} style={styles.progressItem}>
              <View style={styles.progressLabel}>
                <Image 
                  source={{ uri: PROGRESS_ICONS.download }} 
                  style={[styles.progressIcon, { 
                    tintColor: item.status === 'success' ? '#10b981' : 
                              item.status === 'exception' ? '#ef4444' : '#3b82f6' 
                  }]} 
                />
                <Text style={styles.progressLabelText}>{item.label}</Text>
              </View>
              <Progress 
                percent={item.percent}
                status={item.status as any}
                strokeWidth={12}
                showInfo={true}
              />
            </View>
          ))}
        </View>
        
        <View style={styles.section}>
          <Text style={styles.sectionTitle}>环形进度条</Text>
          <View style={styles.circleProgressSection}>
            <View style={styles.circleProgressItem}>
              <CircleProgress 
                percent={75}
                size={100}
                strokeWidth={12}
                color="#3b82f6"
                showInfo={true}
              />
              <Text style={styles.circleProgressLabel}>下载进度</Text>
            </View>
            
            <View style={styles.circleProgressItem}>
              <CircleProgress 
                percent={45}
                size={100}
                strokeWidth={12}
                color="#10b981"
                showInfo={true}
              />
              <Text style={styles.circleProgressLabel}>上传进度</Text>
            </View>
            
            <View style={styles.circleProgressItem}>
              <CircleProgress 
                percent={100}
                size={100}
                strokeWidth={12}
                color="#ef4444"
                showInfo={true}
              />
              <Text style={styles.circleProgressLabel}>错误状态</Text>
            </View>
          </View>
        </View>
        
        <View style={styles.section}>
          <Text style={styles.sectionTitle}>动态进度演示</Text>
          <View style={styles.dynamicProgressSection}>
            <Progress 
              percent={dynamicProgress}
              status={dynamicProgress === 100 ? 'success' : 'active'}
              strokeWidth={15}
              showInfo={true}
            />
            <View style={styles.dynamicButtons}>
              <TouchableOpacity 
                style={[styles.dynamicButton, styles.startButton]} 
                onPress={startProgress}
                disabled={isRunning}
              >
                <Text style={styles.dynamicButtonText}>
                  {isRunning ? '运行中...' : '开始'}
                </Text>
              </TouchableOpacity>
              <TouchableOpacity 
                style={[styles.dynamicButton, styles.resetButton]} 
                onPress={resetProgress}
              >
                <Text style={styles.dynamicButtonText}>重置</Text>
              </TouchableOpacity>
            </View>
          </View>
        </View>
        
        <View style={styles.featuresSection}>
          <Text style={styles.featuresTitle}>功能特性</Text>
          <View style={styles.featureList}>
            <View style={styles.featureItem}>
              <Text style={styles.featureBullet}>•</Text>
              <Text style={styles.featureText}>支持线性和环形两种进度展示</Text>
            </View>
            <View style={styles.featureItem}>
              <Text style={styles.featureBullet}>•</Text>
              <Text style={styles.featureText}>四种状态样式(正常、活动、成功、异常)</Text>
            </View>
            <View style={styles.featureItem}>
              <Text style={styles.featureBullet}>•</Text>
              <Text style={styles.featureText}>可自定义颜色、粗细和尺寸</Text>
            </View>
            <View style={styles.featureItem}>
              <Text style={styles.featureBullet}>•</Text>
              <Text style={styles.featureText}>支持百分比文本显示</Text>
            </View>
          </View>
        </View>
        
        <View style={styles.usageSection}>
          <Text style={styles.usageTitle}>使用说明</Text>
          <Text style={styles.usageText}>
            进度条组件可用于展示任务进度、文件传输状态等场景。
            支持多种样式和状态,可根据业务需求灵活配置。
          </Text>
        </View>
      </ScrollView>
      
      <View style={styles.footer}>
        <Text style={styles.footerText}>© 2023 进度条组件. All rights reserved.</Text>
      </View>
    </SafeAreaView>
  );
};

const { width, height } = Dimensions.get('window');

const styles = StyleSheet.create({
  container: {
    flex: 1,
    backgroundColor: '#f8fafc',
  },
  header: {
    backgroundColor: '#ffffff',
    paddingTop: 20,
    paddingBottom: 25,
    paddingHorizontal: 20,
    borderBottomWidth: 1,
    borderBottomColor: '#e2e8f0',
    shadowColor: '#000',
    shadowOffset: { width: 0, height: 2 },
    shadowOpacity: 0.05,
    shadowRadius: 4,
    elevation: 2,
  },
  headerTitle: {
    fontSize: 26,
    fontWeight: '700',
    color: '#0f172a',
    textAlign: 'center',
    marginBottom: 5,
  },
  headerSubtitle: {
    fontSize: 15,
    color: '#64748b',
    textAlign: 'center',
  },
  contentContainer: {
    padding: 20,
  },
  section: {
    marginBottom: 30,
  },
  sectionTitle: {
    fontSize: 22,
    fontWeight: '700',
    color: '#0f172a',
    marginBottom: 20,
    paddingLeft: 10,
    borderLeftWidth: 4,
    borderLeftColor: '#3b82f6',
  },
  progressItem: {
    marginBottom: 25,
  },
  progressLabel: {
    flexDirection: 'row',
    alignItems: 'center',
    marginBottom: 10,
  },
  progressIcon: {
    width: 20,
    height: 20,
    marginRight: 10,
  },
  progressLabelText: {
    fontSize: 16,
    fontWeight: '600',
    color: '#334155',
  },
  progressContainer: {
    flexDirection: 'row',
    alignItems: 'center',
  },
  progressBar: {
    flex: 1,
  },
  progressTrail: {
    backgroundColor: '#e2e8f0',
    borderRadius: 5,
    overflow: 'hidden',
  },
  progressFill: {
    backgroundColor: '#3b82f6',
    height: '100%',
  },
  progressInfo: {
    marginLeft: 15,
    minWidth: 40,
  },
  progressText: {
    fontSize: 16,
    fontWeight: '600',
  },
  circleProgressSection: {
    flexDirection: 'row',
    justifyContent: 'space-around',
    alignItems: 'center',
  },
  circleProgressItem: {
    alignItems: 'center',
  },
  circleProgress: {
    position: 'relative',
    justifyContent: 'center',
    alignItems: 'center',
  },
  circleBg: {
    position: 'absolute',
  },
  circleFill: {
    position: 'absolute',
  },
  circleInfo: {
    position: 'absolute',
    justifyContent: 'center',
    alignItems: 'center',
  },
  circleText: {
    fontSize: 18,
    fontWeight: '700',
  },
  circleProgressLabel: {
    marginTop: 10,
    fontSize: 14,
    color: '#64748b',
  },
  dynamicProgressSection: {
    backgroundColor: '#ffffff',
    borderRadius: 16,
    padding: 20,
    borderWidth: 1,
    borderColor: '#e2e8f0',
  },
  dynamicButtons: {
    flexDirection: 'row',
    justifyContent: 'center',
    marginTop: 20,
  },
  dynamicButton: {
    paddingHorizontal: 25,
    paddingVertical: 12,
    borderRadius: 8,
    marginHorizontal: 10,
  },
  startButton: {
    backgroundColor: '#3b82f6',
  },
  resetButton: {
    backgroundColor: '#f1f5f9',
  },
  dynamicButtonText: {
    fontSize: 16,
    fontWeight: '600',
  },
  featuresSection: {
    backgroundColor: '#ffffff',
    borderRadius: 16,
    padding: 20,
    marginBottom: 30,
    borderWidth: 1,
    borderColor: '#e2e8f0',
  },
  featuresTitle: {
    fontSize: 20,
    fontWeight: '700',
    color: '#0f172a',
    marginBottom: 15,
    textAlign: 'center',
  },
  featureList: {
    paddingLeft: 10,
  },
  featureItem: {
    flexDirection: 'row',
    alignItems: 'center',
    marginBottom: 12,
  },
  featureBullet: {
    fontSize: 18,
    color: '#3b82f6',
    marginRight: 10,
  },
  featureText: {
    fontSize: 16,
    color: '#334155',
    flex: 1,
  },
  usageSection: {
    backgroundColor: '#ffffff',
    borderRadius: 16,
    padding: 20,
    borderWidth: 1,
    borderColor: '#e2e8f0',
  },
  usageTitle: {
    fontSize: 20,
    fontWeight: '700',
    color: '#0f172a',
    marginBottom: 15,
    textAlign: 'center',
  },
  usageText: {
    fontSize: 16,
    color: '#334155',
    lineHeight: 24,
    textAlign: 'center',
  },
  footer: {
    paddingVertical: 15,
    alignItems: 'center',
    borderTopWidth: 1,
    borderTopColor: '#e2e8f0',
    backgroundColor: '#ffffff',
  },
  footerText: {
    fontSize: 14,
    color: '#64748b',
    fontWeight: '500',
  },
});

export default App;

这段React Native进度条组件代码实现了一个双模式的可视化进度指示器,支持线性进度条和环形进度条两种显示方式。组件通过状态配置系统来区分不同的进度状态,包括正常、活动、成功和异常四种类型,每种状态对应不同的颜色编码。成功状态使用绿色,异常状态使用红色,活动状态使用蓝色,正常状态则使用默认蓝色或自定义颜色。

代码的核心原理是通过数学计算和样式组合来实现进度可视化。对于线性进度条,通过设置内部填充View的宽度百分比来显示进度;对于环形进度条,则通过计算圆环周长和进度比例来确定strokeDashoffset值,从而实现环形填充效果。通过transform旋转-90度,让进度从顶部开始顺时针填充,符合用户对进度条的传统认知。

在鸿蒙系统适配方面,这段代码面临着深刻的架构差异。React Native的进度条依赖于基础组件的样式组合,特别是使用SVG库来实现环形进度效果。而鸿蒙的ArkUI框架提供了Progress组件作为系统级的进度指示器实现。鸿蒙的Progress组件采用声明式配置方式,开发者只需设置进度值和类型参数,系统会自动处理进度显示和动画效果。

鸿蒙的Progress组件支持线性、环形和刻度三种类型,内置了丰富的样式选项和动画效果。线性进度条可以通过设置borderRadius实现圆角效果,环形进度条则自动处理stroke的动画过渡。特别是当进度状态发生变化时,鸿蒙的组件能够提供更平滑的状态转换。

性能优化层面,React Native的进度条由于包含复杂的样式计算和SVG渲染,在频繁更新时可能遇到性能瓶颈。而鸿蒙的Progress组件在Native层实现,能够提供更高效的渲染性能。


打包

接下来通过打包命令npn run harmony将reactNative的代码打包成为bundle,这样可以进行在开源鸿蒙OpenHarmony中进行使用。

打包之后再将打包后的鸿蒙OpenHarmony文件拷贝到鸿蒙的DevEco-Studio工程目录去:

最后运行效果图如下显示:

欢迎大家加入开源鸿蒙跨平台开发者社区,一起共建开源鸿蒙跨平台生态。

相关推荐
2501_9462447815 小时前
Flutter & OpenHarmony OA系统下拉刷新组件开发指南
开发语言·javascript·flutter
没事多睡觉66615 小时前
零基础React + TypeScript 教程
前端·react.js·typescript
_Kayo_15 小时前
React useState setState之后获取到的数据一直是初始值
前端·javascript·react.js
黎明初时15 小时前
react基础框架搭建3-配置 Redux:react+router+redux+axios+Tailwind+webpack
前端·react.js·webpack
yang9yun15 小时前
PostMan加载三方JS
javascript·测试工具·postman
Object~15 小时前
2.变量声明
开发语言·前端·javascript
释怀不想释怀15 小时前
vue前端crud(页面布局,新增,vue中反向代理)
前端·javascript·vue.js
zpjing~.~15 小时前
iframe和父页面消息通信
开发语言·前端·javascript
释怀不想释怀17 小时前
Ajax,vue生命周期(自动加载页面发出请求)Axios
前端·javascript·ajax
cz追天之路1 天前
华为机考--- 字符串最后一个单词的长度
javascript·css·华为·less