如何在React Native中,开发一个类似于鸿蒙组件(Hong Kong component)的NoticeBar(通知栏)组件呢?

在React Native中,开发一个类似于鸿蒙组件(Hong Kong component)的NoticeBar(通知栏)组件,可以通过以下几个步骤来实现:

  1. 创建新的React Native组件

首先,创建一个新的React Native组件NoticeBar。你可以在React Native项目中创建一个新的文件,例如NoticeBar.js

  1. 设计组件结构

NoticeBar.js中,你可以使用ViewTextTouchableOpacity等组件来构建你的通知栏。例如:

jsx 复制代码
import React from 'react';
import { View, Text, StyleSheet, TouchableOpacity } from 'react-native';

const NoticeBar = ({ message, onPress }) => {
  return (
    <TouchableOpacity style={styles.container} onPress={onPress}>
      <Text style={styles.text}>{message}</Text>
    </TouchableOpacity>
  );
};

const styles = StyleSheet.create({
  container: {
    backgroundColor: 'FFCC00', // 黄色背景,类似于鸿组件的样式
    padding: 10,
    justifyContent: 'center',
    alignItems: 'center',
  },
  text: {
    color: '000', // 黑色文字
    fontSize: 16,
  },
});

export default NoticeBar;
  1. 使用组件

在你的应用中,例如在App.js或者其他页面中,你可以这样使用NoticeBar组件:

jsx 复制代码
import React from 'react';
import { SafeAreaView, StyleSheet } from 'react-native';
import NoticeBar from './NoticeBar'; // 确保路径正确

const App = () => {
  const handlePress = () => {
    console.log('NoticeBar pressed!');
  };

  return (
    <SafeAreaView style={styles.container}>
      <NoticeBar message="这是一条重要通知!" onPress={handlePress} />
      {/* 其他组件 */}
    </SafeAreaView>
  );
};

const styles = StyleSheet.create({
  container: {
    flex: 1,
    justifyContent: 'flex-start', // 让通知栏位于顶部
    paddingTop: 20, // 根据需要调整顶部空间,避免遮挡状态栏等
  },
});

export default App;
  1. 自定义样式和功能

你可以根据需要自定义NoticeBar的样式和行为。例如,你可以添加关闭按钮、滑动隐藏功能等。以下是一个添加关闭按钮的示例:

jsx 复制代码
import React from 'react';
import { View, Text, StyleSheet, TouchableOpacity } from 'react-native';
import Icon from 'react-native-vector-icons/FontAwesome'; // 引入图标库,例如FontAwesome

const NoticeBar = ({ message, onPress, onClose }) => {
  return (
    <View style={styles.container}>
      <Text style={styles.text}>{message}</Text>
      <TouchableOpacity onPress={onClose} style={styles.closeButton}>
        <Icon name="close" size={20} color="000" /> {/* 使用图标库显示关闭图标 */}
      </TouchableOpacity>
    </View>
  );
};

并更新样式和事件处理:

jsx 复制代码
const styles = StyleSheet.create({
  container: {
    backgroundColor: 'FFCC00', // 黄色背景,类似于鸿组件的样式
    padding: 10,
    flexDirection: 'row', // 使文本和关闭按钮并排显示
    justifyContent: 'space-between', // 两端对齐,文本在左侧,关闭按钮在右侧
    alignItems: 'center', // 垂直居中图标和文本
  },
  text: {
    color: '000', // 黑色文字,可根据需要调整字体大小等样式属性。例如:fontSize: 16, flex: 1, // 使文本填满剩余空间,避免图标遮挡文本。根据需要添加或修改。例如:flex: 1, marginRight: 10, // 在文本和关闭按钮之间添加一些空间。根据需要调整。例如:marginRight: 10, }, closeButton: { paddingHorizontal: 5, }, }); ```在使用组件时添加关闭事件处理: ```jsx const handleClose = () => { console.log('NoticeBar closed!'); // 在这里可以添加关闭后的逻辑,例如隐藏通知栏 };

真实案列代码演示效果:

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

// Base64 Icons for notice bar components
const NOTICEBAR_ICONS = {
  info: '......',
  warning: '......',
  success: '......',
  error: '......',
  close: '......'
};

// 通知栏组件
interface NoticeBarProps {
  type?: 'info' | 'warning' | 'success' | 'error';
  message: string;
  closable?: boolean;
  onClose?: () => void;
  marquee?: boolean;
  style?: object;
}

const NoticeBar: React.FC<NoticeBarProps> = ({
  type = 'info',
  message,
  closable = false,
  onClose,
  marquee = false,
  style
}) => {
  const [visible, setVisible] = useState(true);
  const translateX = useState(new Animated.Value(0))[0];
  const [textWidth, setTextWidth] = useState(0);
  const [containerWidth, setContainerWidth] = useState(0);

  const getTypeConfig = () => {
    switch (type) {
      case 'warning':
        return {
          backgroundColor: '#fffbeb',
          borderColor: '#facc15',
          textColor: '#854d0e',
          icon: NOTICEBAR_ICONS.warning
        };
      case 'success':
        return {
          backgroundColor: '#f0fdf4',
          borderColor: '#22c55e',
          textColor: '#14532d',
          icon: NOTICEBAR_ICONS.success
        };
      case 'error':
        return {
          backgroundColor: '#fef2f2',
          borderColor: '#ef4444',
          textColor: '#7f1d1d',
          icon: NOTICEBAR_ICONS.error
        };
      case 'info':
      default:
        return {
          backgroundColor: '#eff6ff',
          borderColor: '#3b82f6',
          textColor: '#1e3a8a',
          icon: NOTICEBAR_ICONS.info
        };
    }
  };

  const config = getTypeConfig();

  useEffect(() => {
    if (marquee && visible && textWidth > containerWidth) {
      const duration = (textWidth / containerWidth) * 5000;
      Animated.loop(
        Animated.sequence([
          Animated.timing(translateX, {
            toValue: -textWidth,
            duration,
            easing: Easing.linear,
            useNativeDriver: true
          }),
          Animated.timing(translateX, {
            toValue: containerWidth,
            duration: 0,
            useNativeDriver: true
          })
        ])
      ).start();
    }
  }, [marquee, visible, textWidth, containerWidth]);

  const handleClose = () => {
    setVisible(false);
    onClose && onClose();
  };

  if (!visible) return null;

  return (
    <View 
      style={[
        styles.noticeBar,
        {
          backgroundColor: config.backgroundColor,
          borderLeftColor: config.borderColor,
          borderLeftWidth: 4
        },
        style
      ]}
      onLayout={(e) => setContainerWidth(e.nativeEvent.layout.width - 50)}
    >
      <Image source={{ uri: config.icon }} style={[styles.noticeIcon, { tintColor: config.borderColor }]} />
      
      <Animated.Text
        style={[
          styles.noticeText,
          { color: config.textColor },
          marquee && textWidth > containerWidth && {
            transform: [{ translateX }],
            width: textWidth
          }
        ]}
        numberOfLines={marquee ? 1 : undefined}
        onTextLayout={(e) => setTextWidth(e.nativeEvent.lines[0]?.width || 0)}
      >
        {message}
      </Animated.Text>
      
      {closable && (
        <TouchableOpacity 
          style={styles.closeButton} 
          onPress={handleClose}
          activeOpacity={0.7}
        >
          <Image source={{ uri: NOTICEBAR_ICONS.close }} style={[styles.closeIcon, { tintColor: config.textColor }]} />
        </TouchableOpacity>
      )}
    </View>
  );
};

// 主应用组件
const App = () => {
  const [notices, setNotices] = useState([
    { id: 1, type: 'info', message: '欢迎使用我们的应用程序!这里有最新的功能更新。', closable: true },
    { id: 2, type: 'warning', message: '请注意,您的账户安全设置需要更新。建议立即修改密码以确保账户安全。', closable: true },
    { id: 3, type: 'success', message: '操作成功完成!您的订单已确认,预计将在3个工作日内送达。', closable: true },
    { id: 4, type: 'error', message: '系统维护通知:为了提供更好的服务,我们将在今晚00:00-02:00进行系统维护,请提前做好准备。', closable: true },
    { id: 5, type: 'info', message: '这是一条很长的通知消息,用于演示滚动效果。当文本内容超出容器宽度时,将会自动启用滚动动画效果,让用户能够完整阅读所有内容。', marquee: true }
  ]);

  const handleNoticeClose = (id: number) => {
    setNotices(notices.filter(notice => notice.id !== id));
  };

  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>
          {notices.map(notice => (
            <NoticeBar
              key={notice.id}
              type={notice.type as any}
              message={notice.message}
              closable={notice.closable}
              marquee={notice.marquee}
              onClose={() => handleNoticeClose(notice.id)}
              style={styles.noticeItem}
            />
          ))}
        </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',
  },
  noticeItem: {
    marginBottom: 15,
  },
  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',
  },
  // Notice Bar Styles
  noticeBar: {
    flexDirection: 'row',
    alignItems: 'center',
    paddingVertical: 12,
    paddingHorizontal: 15,
    borderRadius: 8,
    borderWidth: 1,
    borderColor: '#e2e8f0',
    backgroundColor: '#eff6ff',
  },
  noticeIcon: {
    width: 20,
    height: 20,
    marginRight: 10,
  },
  noticeText: {
    flex: 1,
    fontSize: 15,
    fontWeight: '500',
    color: '#1e3a8a',
  },
  closeButton: {
    marginLeft: 10,
    padding: 5,
  },
  closeIcon: {
    width: 16,
    height: 16,
  },
});

export default App;

这段React Native通知栏组件代码实现了一个高度可配置的消息提示系统。组件通过类型配置系统区分不同级别的通知,包括信息、警告、成功和错误四种状态,每种类型都有对应的背景色、边框色、文字颜色和图标,通过getTypeConfig函数实现类型到视觉样式的映射。组件内部维护可见性状态、容器宽度和文本宽度三个关键状态,通过useEffect监听这些状态变化来控制滚动动画的启停。

在鸿蒙系统适配方面,这段代码面临着根本性的架构差异。React Native的通知栏依赖于Animated API和状态管理来实现滚动效果,而鸿蒙的ArkUI框架提供了Marquee跑马灯组件作为系统级优化实现。鸿蒙的Marquee组件专门设计用于处理长文本在有限空间内的滚动展示,其底层实现直接调用系统图形引擎,避免了JavaScript桥接带来的性能损耗。

鸿蒙的Notification系统与React Native的组件级实现有本质区别。鸿蒙提供了完整的通知服务框架,包括通知发布、管理和显示的一整套解决方案,这与在应用内自定义通知组件的思路完全不同。在鸿蒙生态中,通知更倾向于通过系统服务统一管理,而不是在每个应用中单独实现。

动画系统的实现机制存在显著差异。React Native的滚动动画通过JavaScript计算动画值,然后通过桥接传递给原生组件渲染。而鸿蒙的Marquee动画在Native层执行,能够实现更流畅的滚动效果和更精确的性能控制。

布局计算方面,React Native需要通过onLayout事件手动获取容器尺寸,而鸿蒙的布局系统提供了更自动化的尺寸管理机制,减少了手动计算的需要。

事件处理流程上,React Native的触摸事件需要在JavaScript层处理,而鸿蒙的手势识别系统直接在Native层完成,这影响了关闭按钮的响应性能。

资源管理机制方面,React Native通过URI加载图标资源,而鸿蒙使用ResourceManager统一管理应用资源,这种差异导致图片加载路径需要重新设计。

状态同步机制上,React Native的setState触发重新渲染,而鸿蒙的状态变更通过响应式系统直接更新相关UI组件,这种差异影响了组件的更新效率。


打包

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

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

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

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

相关推荐
BlackWolfSky1 分钟前
React中文网课程笔记1—快速入门
前端·笔记·react.js
借个火er7 分钟前
用 Tauri 2.0 + React + Rust 打造跨平台文件工具箱
react.js·rust
俩毛豆12 分钟前
【毛豆工具集】【UI】【多设备适配】实现与屏幕密度等倍的图片加载
华为·harmonyos
码界奇点28 分钟前
基于React与TypeScript的后台管理系统设计与实现
前端·c++·react.js·typescript·毕业设计·源代码管理
Summer不秃44 分钟前
使用 SnapDOM + jsPDF 生成高质量 PDF (含多页分页, 附源码)
前端·javascript·vue.js·pdf·node.js
l134062082351 小时前
344.在鸿蒙上使用 animations Flutter 包的指南
flutter·华为·harmonyos
灯把黑夜烧了一个洞1 小时前
2026年跨年倒计时网页版
javascript·css·html·2026跨年代码·新年代码
前端小咸鱼一条1 小时前
Redux
react.js·前端框架
zhenryx1 小时前
React Native 横向滚动指示器组件库(淘宝|京东...&旧版|新版)
javascript·react native·react.js
POLITE32 小时前
Leetcode 54.螺旋矩阵 JavaScript (Day 8)
javascript·leetcode·矩阵