【精通篇】打造React Native鸿蒙跨平台开发高级复合组件库开发系列:Circle 环形进度条(圆环形的进度条组件)

在 React Native 中开发鸿蒙(HarmonyOS)的 Circle 环形进度条,你可以使用现有的第三方库或者自己实现一个自定义组件。鸿蒙操作系统是基于 Java/Kotlin 开发的,而 React Native 主要使用 JavaScript,但你可以通过一些桥梁技术(如使用原生模块)来实现两者的交互。

方法一:使用第三方库

目前 React Native 社区有一些库可以帮助你实现环形进度条,例如 react-native-progress。虽然这个库主要用于 Harmony 和 Android,但你可以尝试在你的项目中加入它,并通过原生模块适配鸿蒙操作系统。

  1. 安装 react-native-progress:

    bash 复制代码
    npm install react-native-progress
  2. 创建原生模块 (如果你需要在鸿蒙上实现特定的定制化):

    • 创建一个新的原生模块,例如 RNCircleProgressBar.javaRNCircleProgressBar.kt
    • 在该模块中,你可以使用鸿蒙的 UI 组件库来实现环形进度条,比如使用 ShapeElement 来绘制圆形。
  3. 在 React Native 中调用原生模块:

    javascript 复制代码
    import { NativeModules } from 'react-native';
    const { RNCircleProgressBar } = NativeModules;
    
    function App() {
      return (
        <View>
          <RNCircleProgressBar progress={0.5} /> {/* 假设这是你的进度条组件 */}
        </View>
      );
    }

方法二:自定义组件

如果你想完全自定义环形进度条,你可以创建一个自定义的 React Native 组件,并使用原生代码绘制圆形进度条。

  1. 创建自定义组件:

    javascript 复制代码
    import React from 'react';
    import { View, requireNativeComponent } from 'react-native';
    
    const CircleProgress = requireNativeComponent('CircleProgress'); // 注意这里用的是原生组件名
    
    export default function CircleProgressComponent({ progress }) {
      return <CircleProgress progress={progress} />;
    }
  2. 在鸿蒙原生模块中实现:

    • 创建一个鸿蒙原生模块,例如 CircleProgressModule.javaCircleProgressModule.kt
    • 使用鸿蒙的 Canvas 或其他图形绘制方法来实现环形进度条。
    • 确保你的模块正确导出了可以被 JavaScript 调用的方法。
  3. 注册原生模块:

    在鸿蒙的原生代码中注册你的模块,确保 React Native 可以加载并使用它。

方法三:使用 WebView 加载 HTML5 环形进度条

如果上述方法都不适合,你可以考虑在 WebView 中加载一个 HTML5 的环形进度条。这种方法相对简单,但性能和交互性可能不如原生实现。

  1. 在 React Native 中使用 WebView:

    javascript 复制代码
    import React from 'react';
    import { View, WebView } from 'react-native';
    
    export default function CircleProgressWebView() {
      return (
        <View style={{ flex: 1 }}>
          <WebView 
            source={{ uri: 'https://example.com/circle-progress.html' }} 
            style={{ flex: 1 }} 
          />
        </View>
      );
    }

    确保你的 HTML 文件中有正确的 CSS 和 JavaScript 来绘制环形进度条。

总结

选择哪种方法取决于你的具体需求和开发时间。如果需要高性能和深度定制,建议使用方法一或方法二。如果只是简单的展示,方法三可能更快捷。无论哪种方法,都需要确保你的 React Native 应用能够正确调用和展示鸿蒙的原生组件。


真实实际代码演示案例:

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

// Base64 Icons for circular progress indicators
const TECH_ICONS = {
  performance: '......',
  battery: '......',
  network: '......',
  temperature: '......',
  security: '......'
};

// 环形进度条组件(使用纯React Native实现)
interface TechCircularProgressProps {
  size?: number;
  strokeWidth?: number;
  progress: number;
  color: string;
  backgroundColor?: string;
  icon?: string;
  title: string;
  subtitle: string;
  valueSuffix?: string;
}

const TechCircularProgress: React.FC<TechCircularProgressProps> = ({
  size = 130,
  strokeWidth = 12,
  progress,
  color,
  backgroundColor = "#1a1a2e",
  icon,
  title,
  subtitle,
  valueSuffix = "%"
}) => {
  const radius = (size - strokeWidth) / 2;
  const circumference = radius * 2 * Math.PI;
  const strokeDashoffset = circumference - (progress / 100) * circumference;

  // 创建圆形路径数据
  const createCirclePath = (cx: number, cy: number, r: number) => {
    return `M ${cx - r}, ${cy}
            a ${r},${r} 0 1,0 ${r * 2},0
            a ${r},${r} 0 1,0 ${-r * 2},0`;
  };

  const backgroundPath = createCirclePath(size / 2, size / 2, radius);
  const progressPath = createCirclePath(size / 2, size / 2, radius);

  return (
    <View style={styles.progressContainer}>
      <View style={styles.progressCircleWrapper}>
        <View style={[styles.circleBackground, { width: size, height: size }]}>
          {/* Background circle */}
          <View style={[
            styles.circleBase,
            {
              width: size,
              height: size,
              borderRadius: size / 2,
              borderWidth: strokeWidth,
              borderColor: backgroundColor,
            }
          ]} />
          
          {/* Progress circle overlay */}
          <View style={[
            styles.circleProgress,
            {
              width: size,
              height: size,
              borderRadius: size / 2,
              borderWidth: strokeWidth,
              borderColor: color,
              transform: [{ rotate: '-90deg' }],
            }
          ]} />
        </View>
        
        {icon && (
          <View style={styles.progressIconContainer}>
            <Image source={{ uri: icon }} style={[styles.progressIcon, { tintColor: color }]} />
          </View>
        )}
        
        <View style={styles.progressCenterText}>
          <Text style={[styles.progressValue, { color }]}>{Math.round(progress)}</Text>
          <Text style={[styles.progressSuffix, { color }]}>{valueSuffix}</Text>
        </View>
      </View>
      
      <Text style={styles.progressTitle}>{title}</Text>
      <Text style={styles.progressSubtitle}>{subtitle}</Text>
    </View>
  );
};

// 主应用组件
const App = () => {
  const [progressData, setProgressData] = useState([
    { 
      id: 'performance', 
      title: '系统性能', 
      subtitle: 'CPU负载', 
      progress: 75, 
      color: '#00d4ff', 
      icon: TECH_ICONS.performance,
      valueSuffix: "%"
    },
    { 
      id: 'battery', 
      title: '电池状态', 
      subtitle: '剩余电量', 
      progress: 45, 
      color: '#ff6b6b', 
      icon: TECH_ICONS.battery,
      valueSuffix: "%"
    },
    { 
      id: 'network', 
      title: '网络连接', 
      subtitle: '信号强度', 
      progress: 82, 
      color: '#4ecdc4', 
      icon: TECH_ICONS.network,
      valueSuffix: "%"
    },
    { 
      id: 'temperature', 
      title: '温度监控', 
      subtitle: '核心温度', 
      progress: 60, 
      color: '#ffd166', 
      icon: TECH_ICONS.temperature,
      valueSuffix: "°C"
    },
    { 
      id: 'security', 
      title: '安全等级', 
      subtitle: '防护强度', 
      progress: 95, 
      color: '#6a0572', 
      icon: TECH_ICONS.security,
      valueSuffix: "%"
    }
  ]);

  // 模拟进度变化
  useEffect(() => {
    const interval = setInterval(() => {
      setProgressData(prev => 
        prev.map(item => ({
          ...item,
          progress: Math.min(100, Math.max(0, item.progress + (Math.random() * 8 - 4)))
        }))
      );
    }, 4000);
    
    return () => clearInterval(interval);
  }, []);

  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.progressGrid}>
          {progressData.map((item) => (
            <TechCircularProgress
              key={item.id}
              progress={Math.round(item.progress)}
              color={item.color}
              title={item.title}
              subtitle={item.subtitle}
              icon={item.icon}
              valueSuffix={item.valueSuffix}
            />
          ))}
        </View>
        
        <View style={styles.statusPanel}>
          <Text style={styles.panelTitle}>系统状态概览</Text>
          <View style={styles.statusRow}>
            <Text style={styles.statusText}>运行时间:</Text>
            <Text style={styles.statusValue}>12 天 4 小时</Text>
          </View>
          <View style={styles.statusRow}>
            <Text style={styles.statusText}>平均负载:</Text>
            <Text style={styles.statusValue}>1.25</Text>
          </View>
          <View style={styles.statusRow}>
            <Text style={styles.statusText}>内存使用:</Text>
            <Text style={styles.statusValue}>6.2 GB / 16 GB</Text>
          </View>
        </View>
      </ScrollView>
      
      <View style={styles.footer}>
        <Text style={styles.footerText}>最后更新: {new Date().toLocaleTimeString()}</Text>
      </View>
    </SafeAreaView>
  );
};

const { width } = Dimensions.get('window');
const isTablet = width >= 768;

const styles = StyleSheet.create({
  container: {
    flex: 1,
    backgroundColor: '#0f0c29',
  },
  header: {
    backgroundColor: '#1a1a2e',
    paddingTop: 20,
    paddingBottom: 25,
    paddingHorizontal: 20,
    borderBottomWidth: 1,
    borderBottomColor: '#16213e',
  },
  headerTitle: {
    fontSize: 26,
    fontWeight: '700',
    color: '#e6e6ff',
    textAlign: 'center',
    marginBottom: 5,
    textShadowColor: 'rgba(0, 212, 255, 0.5)',
    textShadowOffset: { width: 0, height: 0 },
    textShadowRadius: 10,
  },
  headerSubtitle: {
    fontSize: 15,
    color: '#a0a0c0',
    textAlign: 'center',
  },
  contentContainer: {
    padding: 20,
  },
  progressGrid: {
    flexDirection: 'row',
    flexWrap: 'wrap',
    justifyContent: 'space-between',
  },
  progressContainer: {
    width: isTablet ? '18%' : '48%',
    backgroundColor: '#16213e',
    borderRadius: 16,
    padding: 15,
    marginBottom: 15,
    alignItems: 'center',
    borderWidth: 1,
    borderColor: '#1f4068',
  },
  progressCircleWrapper: {
    position: 'relative',
    marginVertical: 10,
  },
  circleBackground: {
    position: 'relative',
  },
  circleBase: {
    position: 'absolute',
  },
  circleProgress: {
    position: 'absolute',
    borderLeftColor: 'transparent',
    borderRightColor: 'transparent',
    borderTopColor: 'transparent',
    borderBottomColor: 'transparent',
    clipPath: 'inset(0 0 50% 0)', // 只显示上半部分
  },
  progressIconContainer: {
    position: 'absolute',
    top: '50%',
    left: '50%',
    width: 40,
    height: 40,
    borderRadius: 20,
    justifyContent: 'center',
    alignItems: 'center',
    transform: [{ translateX: -20 }, { translateY: -40 }],
    backgroundColor: 'rgba(26, 26, 46, 0.7)',
    borderWidth: 1,
    borderColor: '#1f4068',
  },
  progressIcon: {
    width: 24,
    height: 24,
  },
  progressCenterText: {
    position: 'absolute',
    top: '50%',
    left: '50%',
    transform: [{ translateX: -20 }, { translateY: -10 }],
    flexDirection: 'row',
    alignItems: 'flex-end',
  },
  progressValue: {
    fontSize: 20,
    fontWeight: '800',
  },
  progressSuffix: {
    fontSize: 12,
    fontWeight: '600',
    marginBottom: 2,
    marginLeft: 2,
  },
  progressTitle: {
    fontSize: 16,
    fontWeight: '700',
    color: '#e6e6ff',
    marginTop: 15,
    textAlign: 'center',
  },
  progressSubtitle: {
    fontSize: 13,
    color: '#a0a0c0',
    textAlign: 'center',
    marginBottom: 5,
  },
  statusPanel: {
    backgroundColor: '#16213e',
    borderRadius: 16,
    padding: 20,
    marginTop: 20,
    borderWidth: 1,
    borderColor: '#1f4068',
  },
  panelTitle: {
    fontSize: 18,
    fontWeight: '700',
    color: '#e6e6ff',
    marginBottom: 15,
    textAlign: 'center',
  },
  statusRow: {
    flexDirection: 'row',
    justifyContent: 'space-between',
    paddingVertical: 8,
    borderBottomWidth: 1,
    borderBottomColor: '#1f4068',
  },
  statusText: {
    fontSize: 14,
    color: '#a0a0c0',
  },
  statusValue: {
    fontSize: 14,
    color: '#00d4ff',
    fontWeight: '600',
  },
  footer: {
    paddingVertical: 15,
    alignItems: 'center',
    borderTopWidth: 1,
    borderTopColor: '#16213e',
    backgroundColor: '#1a1a2e',
  },
  footerText: {
    fontSize: 14,
    color: '#a0a0c0',
    fontWeight: '600',
  },
});

export default App;

这段React Native代码实现了一个科技监控中心的界面,其设计理念与鸿蒙系统的架构思想有着深度的契合。从原理层面来看,该代码采用了声明式UI和数据驱动渲染的模式,这与鸿蒙ArkUI的响应式编程范式高度一致。组件通过useState和useEffect来管理状态和副作用,实现了状态变化自动触发界面更新的机制,这种单向数据流的设计正是鸿蒙分布式能力在微观层面的体现。

在组件架构方面,代码通过TechCircularProgress组件的复用展示了鸿蒙原子化服务的设计思想。每个进度组件都是一个独立的功能单元,拥有自己的状态和显示逻辑,这种模块化的设计使得组件可以在不同的上下文中被灵活调用,正如鸿蒙的Ability组件可以在不同设备间无缝流转。进度数据的动态更新机制通过setInterval实现了实时监控的效果,这种持续性的状态监听与鸿蒙的分布式数据管理能力有着相似的实现逻辑。

界面布局采用了分层结构设计,从顶部的SafeAreaView到底部的footer,中间包含ScrollView和多个View组件,这种清晰的层级关系体现了鸿蒙系统对界面结构的规范化要求。响应式设计通过Dimensions API检测屏幕宽度,实现了平板和手机的适配,这与鸿蒙一次开发多端部署的理念不谋而合。

数据模型的设计采用了标准化的接口规范,每个监控项都包含id、title、subtitle、progress等属性,这种统一的数据结构便于在鸿蒙生态中进行数据交换和共享。事件处理机制通过onDateSelect等回调函数实现了组件间的通信,这种松耦合的设计模式正是鸿蒙分布式架构的核心特征。

视觉设计方面,通过不同的颜色编码来区分不同类型的监控指标,这种信息可视化方式与鸿蒙的人机交互设计原则相吻合。动态进度变化通过Math.random()模拟真实环境中的数据波动,这种实时数据更新的机制反映了鸿蒙在物联网场景下的实时监控能力。


打包

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

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

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

相关推荐
爱吃大芒果2 小时前
Flutter 开发环境配置避坑指南:Windows/macOS/Linux 全平台
flutter·华为·harmonyos
花先锋队长2 小时前
华为Mate X7测评:折叠屏的尽头,是让你忘记它在折叠?
科技·华为·智能手机·harmonyos
赵财猫._.2 小时前
React Native鸿蒙开发实战(一):环境搭建与第一个应用
react native·react.js·华为·harmonyos
2401_860494702 小时前
在React Native鸿蒙跨平台开发中实现一个计数排序算法,如何使用一个额外的数组来统计每个值的出现次数,然后根据这个统计结果来重构原数组的顺序
javascript·react native·react.js·重构·ecmascript·排序算法
2401_860494703 小时前
在React Native鸿蒙跨平台开发中实现一个基数排序算法,如何进行找到最大数:遍历数组找到最大值呢?
javascript·算法·react native·react.js·排序算法·harmonyos
ElenaYu3 小时前
在 Mac 上用 scrcpy 投屏 Honor 300 Pro(鸿蒙/Android)并支持鼠标点击控制
android·macos·harmonyos
晚霞的不甘4 小时前
[鸿蒙2025领航者闯关]:Flutter + OpenHarmony 性能优化终极指南:从 30 FPS 到 60 FPS 的实战跃迁
flutter·性能优化·harmonyos
盐焗西兰花7 小时前
鸿蒙学习实战之路 - 网络重连最佳实践
网络·学习·harmonyos
2401_8603195213 小时前
在React Native鸿蒙跨平台开发中实现 二叉搜索树,如何实现一些基本的遍历方法,如中序遍历,中序遍历按顺序访问左子树、根节点、右子树
react native·react.js·harmonyos