在React Native中,开发一个类似于鸿蒙组件(Hong Kong component)的NoticeBar(通知栏)组件,可以通过以下几个步骤来实现:
- 创建新的React Native组件
首先,创建一个新的React Native组件NoticeBar。你可以在React Native项目中创建一个新的文件,例如NoticeBar.js。
- 设计组件结构
在NoticeBar.js中,你可以使用View、Text、TouchableOpacity等组件来构建你的通知栏。例如:
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;
- 使用组件
在你的应用中,例如在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;
- 自定义样式和功能
你可以根据需要自定义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工程目录去:

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