【精通篇】打造React Native鸿蒙跨平台开发高级复合组件库开发系列:Badge 徽标(在右上角展示徽标数字或小红点)

在 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';
import Svg, { Circle, Path } from 'react-native-svg';

// Base64 Icons for circular progress indicators
const PROGRESS_ICONS = {
  health: '......',
  battery: '......',
  storage: '......',
  network: '......',
  memory: '......'
};

// 环形进度条组件
interface CircularProgressProps {
  size?: number;
  strokeWidth?: number;
  progress: number;
  color: string;
  backgroundColor?: string;
  icon?: string;
  title: string;
  subtitle: string;
}

const CircularProgress: React.FC<CircularProgressProps> = ({
  size = 120,
  strokeWidth = 10,
  progress,
  color,
  backgroundColor = "#E0E0E0",
  icon,
  title,
  subtitle
}) => {
  const radius = (size - strokeWidth) / 2;
  const circumference = radius * 2 * Math.PI;
  const strokeDashoffset = circumference - (progress / 100) * circumference;

  return (
    <View style={styles.progressContainer}>
      <View style={styles.progressCircleWrapper}>
        <Svg width={size} height={size}>
          {/* Background circle */}
          <Circle
            cx={size / 2}
            cy={size / 2}
            r={radius}
            stroke={backgroundColor}
            strokeWidth={strokeWidth}
            fill="none"
          />
          
          {/* Progress circle */}
          <Circle
            cx={size / 2}
            cy={size / 2}
            r={radius}
            stroke={color}
            strokeWidth={strokeWidth}
            fill="none"
            strokeDasharray={`${circumference} ${circumference}`}
            strokeDashoffset={strokeDashoffset}
            strokeLinecap="round"
            transform={`rotate(-90 ${size / 2} ${size / 2})`}
          />
        </Svg>
        
        {icon && (
          <View style={[styles.progressIconContainer, { backgroundColor: `${color}20` }]}>
            <Image source={{ uri: icon }} style={[styles.progressIcon, { tintColor: color }]} />
          </View>
        )}
      </View>
      
      <Text style={styles.progressTitle}>{title}</Text>
      <Text style={styles.progressSubtitle}>{subtitle}</Text>
      <Text style={[styles.progressPercent, { color }]}>{progress}%</Text>
    </View>
  );
};

// 主应用组件
const App = () => {
  const [progressData, setProgressData] = useState([
    { id: 'health', title: '健康指数', subtitle: '身体状况', progress: 75, color: '#4CAF50', icon: PROGRESS_ICONS.health },
    { id: 'battery', title: '电池电量', subtitle: '设备续航', progress: 45, color: '#FF9800', icon: PROGRESS_ICONS.battery },
    { id: 'storage', title: '存储空间', subtitle: '磁盘使用', progress: 82, color: '#2196F3', icon: PROGRESS_ICONS.storage },
    { id: 'network', title: '网络速度', subtitle: '连接质量', progress: 60, color: '#9C27B0', icon: PROGRESS_ICONS.network },
    { id: 'memory', title: '内存使用', subtitle: 'RAM占用', progress: 35, color: '#F44336', icon: PROGRESS_ICONS.memory }
  ]);

  // 模拟进度变化
  useEffect(() => {
    const interval = setInterval(() => {
      setProgressData(prev => 
        prev.map(item => ({
          ...item,
          progress: Math.min(100, Math.max(0, item.progress + (Math.random() * 10 - 5)))
        }))
      );
    }, 3000);
    
    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) => (
            <CircularProgress
              key={item.id}
              progress={Math.round(item.progress)}
              color={item.color}
              title={item.title}
              subtitle={item.subtitle}
              icon={item.icon}
            />
          ))}
        </View>
        
        <View style={styles.infoSection}>
          <Text style={styles.infoTitle}>性能监控说明</Text>
          <Text style={styles.infoText}>
            环形进度条直观展示各项系统指标的当前状态。
            绿色表示正常,黄色表示需要注意,红色表示警告。
            数值会实时更新以反映最新状态。
          </Text>
        </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: '#F5F7FA',
  },
  header: {
    backgroundColor: '#FFFFFF',
    paddingTop: 20,
    paddingBottom: 25,
    paddingHorizontal: 20,
    borderBottomWidth: 1,
    borderBottomColor: '#E1E5EB',
    shadowColor: '#000',
    shadowOffset: { width: 0, height: 2 },
    shadowOpacity: 0.05,
    shadowRadius: 4,
    elevation: 2,
  },
  headerTitle: {
    fontSize: 26,
    fontWeight: '700',
    color: '#2D3748',
    textAlign: 'center',
    marginBottom: 5,
  },
  headerSubtitle: {
    fontSize: 15,
    color: '#718096',
    textAlign: 'center',
  },
  contentContainer: {
    padding: 20,
  },
  progressGrid: {
    flexDirection: 'row',
    flexWrap: 'wrap',
    justifyContent: 'space-between',
  },
  progressContainer: {
    width: isTablet ? '18%' : '48%',
    backgroundColor: '#FFFFFF',
    borderRadius: 16,
    padding: 15,
    marginBottom: 15,
    alignItems: 'center',
    shadowColor: '#000',
    shadowOffset: { width: 0, height: 2 },
    shadowOpacity: 0.08,
    shadowRadius: 6,
    elevation: 3,
  },
  progressCircleWrapper: {
    position: 'relative',
    marginVertical: 10,
  },
  progressIconContainer: {
    position: 'absolute',
    top: '50%',
    left: '50%',
    width: 40,
    height: 40,
    borderRadius: 20,
    justifyContent: 'center',
    alignItems: 'center',
    transform: [{ translateX: -20 }, { translateY: -20 }],
  },
  progressIcon: {
    width: 20,
    height: 20,
  },
  progressTitle: {
    fontSize: 16,
    fontWeight: '700',
    color: '#2D3748',
    marginTop: 10,
    textAlign: 'center',
  },
  progressSubtitle: {
    fontSize: 13,
    color: '#718096',
    textAlign: 'center',
    marginBottom: 5,
  },
  progressPercent: {
    fontSize: 18,
    fontWeight: '800',
    marginTop: 5,
  },
  infoSection: {
    backgroundColor: '#FFFFFF',
    borderRadius: 16,
    padding: 20,
    marginTop: 20,
    shadowColor: '#000',
    shadowOffset: { width: 0, height: 2 },
    shadowOpacity: 0.08,
    shadowRadius: 6,
    elevation: 3,
  },
  infoTitle: {
    fontSize: 18,
    fontWeight: '700',
    color: '#2D3748',
    marginBottom: 10,
  },
  infoText: {
    fontSize: 14,
    color: '#718096',
    lineHeight: 22,
  },
  footer: {
    paddingVertical: 15,
    alignItems: 'center',
    borderTopWidth: 1,
    borderTopColor: '#E1E5EB',
    backgroundColor: '#FFFFFF',
  },
  footerText: {
    fontSize: 14,
    color: '#718096',
    fontWeight: '600',
  },
});

export default App;

这段React Native代码实现了一个环形进度条组件,用于构建系统监控中心的仪表盘界面。其核心原理是通过SVG图形和数学计算来可视化进度数据。

该环形进度条组件采用双圆环设计,底层圆环作为背景轨道,上层圆环通过动态计算strokeDashoffset属性来实现进度填充效果。关键计算包括根据组件尺寸和描边宽度确定圆环半径,再根据半径计算圆环周长,最后根据进度百分比计算需要隐藏的弧长。通过将圆环旋转-90度,使进度从顶部开始顺时针填充,符合用户对进度条的传统认知。

在主应用组件中,通过状态管理维护多个系统指标的进度数据,包括健康指数、电池电量、存储空间等,每个指标都有对应的颜色编码和图标。使用useEffect Hook模拟实时数据更新,每隔3秒随机调整各项进度值,创造出动态监控效果。

在鸿蒙系统适配方面,这段代码面临几个关键挑战。鸿蒙的ArkUI框架采用声明式UI范式,其图形绘制机制与React Native的SVG组件有本质区别。React Native依赖于第三方react-native-svg库,而鸿蒙内置了更高效的图形绘制能力。

鸿蒙的Progress组件虽然提供了环形进度条的基础实现,但自定义能力有限,无法直接实现这种带有中心图标和复杂样式的设计。如果要在鸿蒙上实现相同效果,需要使用Canvas组件进行自定义绘制,这需要完全不同的实现思路。

布局系统方面,React Native的Flexbox布局与鸿蒙的Flex、Grid等布局组件在具体属性和行为上存在差异。特别是响应式设计部分,React Native通过Dimensions API检测屏幕尺寸,而鸿蒙提供了更完善的响应式布局方案。

性能方面,React Native的动画和状态更新需要通过JavaScript桥接层,而鸿蒙的声明式UI和状态管理在Native层执行,具有更好的性能表现。特别是当需要频繁更新多个进度条时,鸿蒙的渲染效率优势会更加明显。

数据流架构上,React Native使用单向数据流和状态提升,而鸿蒙推荐使用MVVM模式,通过@State、@Link等装饰器管理状态,这导致组件间通信模式完全不同。


打包

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

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

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

相关推荐
HONG````5 小时前
鸿蒙异步编程深度解析:async/await 原理、使用与实战
华为·harmonyos
马剑威(威哥爱编程)5 小时前
【鸿蒙开发案例篇】NAPI 实现 ArkTS 与 C++ 间的复杂对象传递
c++·华为·harmonyos
国服第二切图仔5 小时前
Electron for鸿蒙PC封装的步骤进度指示器组件
microsoft·electron·harmonyos·鸿蒙pc
北辰alk5 小时前
React Router 路由模式详解:HashRouter vs BrowserRouter
react.js
北辰alk5 小时前
React Native 工作原理深度解析:Bridge 机制与核心架构
react native
赵财猫._.6 小时前
【Flutter x 鸿蒙】第八篇:打包发布、应用上架与运营监控
flutter·华为·harmonyos
晚霞的不甘6 小时前
[鸿蒙2025领航者闯关]: Flutter + OpenHarmony 安全开发实战:从数据加密到权限管控的全链路防护
安全·flutter·harmonyos
灰灰勇闯IT6 小时前
[鸿蒙2025领航者闯关] 鸿蒙6.0星盾安全架构实战:打造金融级支付应用的安全防护
安全·harmonyos·安全架构
禁默6 小时前
[鸿蒙2025领航者闯关] 鸿蒙 6 特性实战闯关:金融支付应用的安全升级之路
安全·金融·harmonyos·鸿蒙2025领航者闯关·鸿蒙6实战