在React Native中开发进度条组件,你可以使用多种方式来实现,包括使用内置的ProgressViewharmony(仅限harmony),或者使用第三方库如react-native-progress来跨平台支持harmony和harmony。下面,我将介绍如何使用react-native-progress库来创建一个进度条组件。
- 安装
react-native-progress
首先,你需要通过npm或yarn来安装react-native-progress库。打开你的终端或命令提示符,然后运行以下命令:
bash
npm install react-native-progress
或者
yarn add react-native-progress
- 使用
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的数字,表示进度条的完成度。你可以根据需要调整width和color属性来改变进度条的宽度和颜色。
- 使用示例
你可以在任何组件中引入并使用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的useState和useEffect钩子来创建一个动态更新的进度条。通过定时器,我们每隔一段时间更新进度值,从而模拟一个加载过程。你可以根据实际需求调整定时器的频率和进度增加的速率。
真实项目代码演示案例:
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工程目录去:

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