Flutter for OpenHarmony 实战:分组列表实现
组件化开发采用组件化开发思想,将功能拆分为数据模型和UI组件,提高代码复用性和可维护性。使用实现无状态组件,简化状态管理,提高性能。Flutter 核心布局技术使用实现高效的列表渲染,支持大量数据的展示。运用ContainerRowColumn等基础布局组件,构建灵活的界面结构。使用等布局属性,实现精确的元素定位。数据结构设计设计了清晰的数据模型层次结构,包括和ItemInfo类。通过构造函数和不
前言:跨生态开发的新机遇
在移动开发领域,我们总是面临着选择与适配。今天,你的Flutter应用在Android和iOS上跑得正欢,明天可能就需要考虑一个新的平台:HarmonyOS(鸿蒙)。这不是一道选答题,而是很多团队正在面对的现实。
Flutter的优势很明确——写一套代码,就能在两个主要平台上运行,开发体验流畅。而鸿蒙代表的是下一个时代的互联生态,它不仅仅是手机系统,更着眼于未来全场景的体验。将现有的Flutter应用适配到鸿蒙,听起来像是一个“跨界”任务,但它本质上是一次有价值的技术拓展:让产品触达更多用户,也让技术栈覆盖更广。
不过,这条路走起来并不像听起来那么简单。Flutter和鸿蒙,从底层的架构到上层的工具链,都有着各自的设计逻辑。会遇到一些具体的问题:代码如何组织?原有的功能在鸿蒙上如何实现?那些平台特有的能力该怎么调用?更实际的是,从编译打包到上架部署,整个流程都需要重新摸索。
这篇文章想做的,就是把这些我们趟过的路、踩过的坑,清晰地摊开给你看。我们不会只停留在“怎么做”,还会聊到“为什么得这么做”,以及“如果出了问题该往哪想”。这更像是一份实战笔记,源自真实的项目经验,聚焦于那些真正卡住过我们的环节。
无论你是在为一个成熟产品寻找新的落地平台,还是从一开始就希望构建能面向多端的应用,这里的思路和解决方案都能提供直接的参考。理解了两套体系之间的异同,掌握了关键的衔接技术,不仅能完成这次迁移,更能积累起应对未来技术变化的能力。
混合工程结构深度解析
项目目录架构
当Flutter项目集成鸿蒙支持后,典型的项目结构会发生显著变化。以下是经过ohos_flutter插件初始化后的项目结构:
my_flutter_harmony_app/
├── lib/ # Flutter业务代码(基本不变)
│ ├── main.dart # 应用入口
│ ├── home_page.dart # 首页
│ └── utils/
│ └── platform_utils.dart # 平台工具类
├── pubspec.yaml # Flutter依赖配置
├── ohos/ # 鸿蒙原生层(核心适配区)
│ ├── entry/ # 主模块
│ │ └── src/main/
│ │ ├── ets/ # ArkTS代码
│ │ │ ├── MainAbility/
│ │ │ │ ├── MainAbility.ts # 主Ability
│ │ │ │ └── MainAbilityContext.ts
│ │ │ └── pages/
│ │ │ ├── Index.ets # 主页面
│ │ │ └── Splash.ets # 启动页
│ │ ├── resources/ # 鸿蒙资源文件
│ │ │ ├── base/
│ │ │ │ ├── element/ # 字符串等
│ │ │ │ ├── media/ # 图片资源
│ │ │ │ └── profile/ # 配置文件
│ │ │ └── en_US/ # 英文资源
│ │ └── config.json # 应用核心配置
│ ├── ohos_test/ # 测试模块
│ ├── build-profile.json5 # 构建配置
│ └── oh-package.json5 # 鸿蒙依赖管理
└── README.md
展示效果图片
flutter 实时预览 效果展示
运行到鸿蒙虚拟设备中效果展示
目录
功能代码实现
数据模型设计
数据模型是整个功能的基础,定义了分组和项目的数据结构。
核心设计
StickyHeaderGroup:表示一个分组,包含标题和项目列表StickyItemData:表示分组中的一个项目,包含标题和副标题ItemInfo:用于列表渲染时的辅助类,标识当前项是头部还是项目
代码实现
class StickyHeaderGroup {
final String title;
final List<StickyItemData> items;
StickyHeaderGroup({required this.title, required this.items});
}
class StickyItemData {
final String title;
final String subtitle;
StickyItemData({required this.title, required this.subtitle});
}
class ItemInfo {
final bool isHeader;
final int groupIndex;
final int itemIndex;
ItemInfo({required this.isHeader, required this.groupIndex, required this.itemIndex});
}
技术要点
-
不可变数据结构:所有字段都使用
final修饰,确保数据的不可变性和安全性。 -
清晰的层次结构:通过
StickyHeaderGroup包含StickyItemData列表,形成清晰的数据层次。 -
辅助类设计:
ItemInfo类专门用于列表渲染时的索引计算和类型判断,提高代码可读性。
粘性头部列表主组件
主组件是功能的核心,负责接收数据并渲染整个列表。
核心设计
- 使用
ListView.builder实现高效的列表渲染 - 通过
_calculateTotalItems()方法计算总项目数 - 通过
_getItemInfo()方法实现分组头部和项目的交替显示 - 直接在组件内部实现头部和项目的UI布局
代码实现
import 'package:flutter/material.dart';
import 'sticky_models.dart';
class StickyHeaderList extends StatelessWidget {
final List<StickyHeaderGroup> groups;
const StickyHeaderList({super.key, required this.groups});
Widget build(BuildContext context) {
return ListView.builder(
itemCount: _calculateTotalItems(),
itemBuilder: (context, index) {
final itemInfo = _getItemInfo(index);
if (itemInfo.isHeader) {
return Container(
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 12),
color: Colors.deepPurple,
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Text(
groups[itemInfo.groupIndex].title,
style: const TextStyle(
color: Colors.white,
fontWeight: FontWeight.bold,
fontSize: 16,
),
),
Text(
'${groups[itemInfo.groupIndex].items.length}项',
style: const TextStyle(
color: Colors.white70,
fontSize: 14,
),
),
],
),
);
} else {
final group = groups[itemInfo.groupIndex];
final item = group.items[itemInfo.itemIndex];
return Container(
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 14),
decoration: const BoxDecoration(
border: Border(
bottom: BorderSide(
color: Colors.grey,
width: 0.5,
),
),
),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
item.title,
style: const TextStyle(
fontSize: 16,
fontWeight: FontWeight.w500,
),
),
const SizedBox(height: 4),
Text(
item.subtitle,
style: TextStyle(
fontSize: 14,
color: Colors.grey[600],
),
),
],
),
);
}
},
);
}
int _calculateTotalItems() {
int total = 0;
for (var group in groups) {
total += 1; // 分组头部
total += group.items.length; // 分组项
}
return total;
}
ItemInfo _getItemInfo(int index) {
int currentIndex = 0;
for (int i = 0; i < groups.length; i++) {
// 检查是否是分组头部
if (currentIndex == index) {
return ItemInfo(isHeader: true, groupIndex: i, itemIndex: -1);
}
currentIndex++;
// 检查是否是分组项
for (int j = 0; j < groups[i].items.length; j++) {
if (currentIndex == index) {
return ItemInfo(isHeader: false, groupIndex: i, itemIndex: j);
}
currentIndex++;
}
}
throw IndexError(index, groups, "Index out of bounds");
}
}
技术要点
-
高效渲染:使用
ListView.builder实现懒加载,只渲染可见区域的项目,提高性能。 -
索引计算:通过
_getItemInfo()方法实现了分组头部和项目的精确索引映射,确保正确显示。 -
UI设计:
- 头部使用深紫色背景,白色文字,显示分组标题和项目数量
- 项目使用白色背景,带有底部边框,显示标题和副标题
- 通过字体大小、字重和颜色创建清晰的视觉层次
-
异常处理:添加了索引越界异常处理,提高代码健壮性。
-
布局优化:使用
MainAxisAlignment.spaceBetween实现头部标题和数量的两端对齐,使用CrossAxisAlignment.start实现项目文本的左对齐。
首页集成使用
在首页中集成分组列表组件,添加模拟数据并直接显示。
核心设计
- 在
_MyHomePageState中定义分组列表数据 - 在
initState方法中初始化数据 - 直接在
build方法中使用StickyHeaderList组件 - 提供丰富的模拟数据,展示不同分组和项目
代码实现
import 'package:flutter/material.dart';
import 'components/sticky_header_list.dart';
import 'components/sticky_models.dart';
void main() {
runApp(const MyApp());
}
class MyApp extends StatelessWidget {
const MyApp({super.key});
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter for openHarmony',
theme: ThemeData(
colorScheme: ColorScheme.fromSeed(seedColor: Colors.deepPurple),
useMaterial3: true,
),
debugShowCheckedModeBanner: false,
home: const MyHomePage(title: 'Flutter for openHarmony'),
);
}
}
class MyHomePage extends StatefulWidget {
const MyHomePage({super.key, required this.title});
final String title;
State<MyHomePage> createState() => _MyHomePageState();
}
class _MyHomePageState extends State<MyHomePage> {
// 粘性头部列表数据
late List<StickyHeaderGroup> _stickyGroups;
void initState() {
super.initState();
// 初始化数据
_stickyGroups = [
StickyHeaderGroup(
title: '电子产品',
items: [
StickyItemData(title: '智能手机', subtitle: '最新款智能手机'),
StickyItemData(title: '笔记本电脑', subtitle: '高性能笔记本电脑'),
StickyItemData(title: '平板电脑', subtitle: '轻薄平板电脑'),
StickyItemData(title: '智能手表', subtitle: '多功能智能手表'),
StickyItemData(title: '无线耳机', subtitle: '降噪无线耳机'),
],
),
StickyHeaderGroup(
title: '家居用品',
items: [
StickyItemData(title: '沙发', subtitle: '舒适布艺沙发'),
StickyItemData(title: '床', subtitle: '实木双人床'),
StickyItemData(title: '餐桌', subtitle: '现代简约餐桌'),
StickyItemData(title: '椅子', subtitle: '人体工学椅子'),
],
),
StickyHeaderGroup(
title: '服装鞋帽',
items: [
StickyItemData(title: 'T恤', subtitle: '纯棉舒适T恤'),
StickyItemData(title: '牛仔裤', subtitle: '修身牛仔裤'),
StickyItemData(title: '外套', subtitle: '时尚休闲外套'),
StickyItemData(title: '鞋子', subtitle: '百搭休闲鞋'),
StickyItemData(title: '帽子', subtitle: '潮流棒球帽'),
StickyItemData(title: '包包', subtitle: '实用单肩包'),
],
),
StickyHeaderGroup(
title: '食品饮料',
items: [
StickyItemData(title: '零食', subtitle: '各种美味零食'),
StickyItemData(title: '饮料', subtitle: '健康饮品'),
StickyItemData(title: '水果', subtitle: '新鲜水果'),
StickyItemData(title: '蔬菜', subtitle: '有机蔬菜'),
],
),
StickyHeaderGroup(
title: '运动健身',
items: [
StickyItemData(title: '跑步机', subtitle: '家用跑步机'),
StickyItemData(title: '哑铃', subtitle: '可调节哑铃'),
StickyItemData(title: '瑜伽垫', subtitle: '防滑瑜伽垫'),
StickyItemData(title: '运动服', subtitle: '透气运动服'),
],
),
];
}
Widget build(BuildContext context) {
return Scaffold(
body: StickyHeaderList(groups: _stickyGroups),
);
}
}
技术要点
-
数据管理:将分组列表数据定义为类成员变量,在
initState方法中初始化,便于后续扩展和修改。 -
组件使用:直接在
Scaffold的body中使用StickyHeaderList组件,简化布局结构。 -
数据模拟:提供了丰富的模拟数据,覆盖了不同类型的分组和项目,便于测试和展示。
-
应用配置:配置了
MaterialApp的主题和调试横幅,提高应用的美观度。 -
状态管理:使用
StatefulWidget和State管理页面状态,为后续可能的交互功能做准备。
开发中容易遇到的问题
-
数据模型设计不合理
- 问题描述:如果数据模型设计不合理,会导致数据管理混乱,组件之间数据传递困难。
- 解决方案:采用清晰的数据模型层次结构,例如
StickyHeaderGroup包含StickyItemData列表,便于数据的组织和传递。
-
索引计算错误
- 问题描述:在实现
_getItemInfo()方法时,容易出现索引计算错误,导致分组头部和项目显示位置不正确。 - 解决方案:仔细检查索引计算逻辑,确保每个分组头部和项目都有唯一的索引映射,可通过添加日志输出来调试索引计算过程。
- 问题描述:在实现
-
性能优化问题
- 问题描述:当分组数量和项目数量较大时,可能会出现列表滚动不流畅的问题。
- 解决方案:使用
ListView.builder实现懒加载,避免一次性渲染所有项目;确保组件的build方法简洁高效,避免不必要的计算。
-
视觉一致性问题
- 问题描述:不同组件之间的间距、字体样式、颜色等不一致,导致整体视觉效果不协调。
- 解决方案:统一设计规范,使用一致的内边距、字体大小、颜色值等,可考虑创建主题常量文件来管理这些样式值。
-
异常处理不完善
- 问题描述:缺少必要的异常处理,可能会导致应用崩溃。
- 解决方案:添加适当的异常处理,例如索引越界异常、空指针异常等,提高应用的稳定性。
-
代码组织问题
- 问题描述:代码组织混乱,文件结构不合理,导致后续维护困难。
- 解决方案:采用模块化的代码组织方式,将不同功能的代码分离到不同的文件中,例如将数据模型和UI组件分离。
总结开发中用到的技术点
-
组件化开发
- 采用组件化开发思想,将功能拆分为数据模型和UI组件,提高代码复用性和可维护性。
- 使用
StatelessWidget实现无状态组件,简化状态管理,提高性能。
-
Flutter 核心布局技术
- 使用
ListView.builder实现高效的列表渲染,支持大量数据的展示。 - 运用
Container、Row、Column等基础布局组件,构建灵活的界面结构。 - 使用
MainAxisAlignment、CrossAxisAlignment等布局属性,实现精确的元素定位。
- 使用
-
数据结构设计
- 设计了清晰的数据模型层次结构,包括
StickyHeaderGroup、StickyItemData和ItemInfo类。 - 通过构造函数和不可变字段,确保数据的安全性和一致性。
- 设计了清晰的数据模型层次结构,包括
-
视觉设计技术
- 使用颜色、字体大小、字重等视觉元素,创建清晰的视觉层次和区分。
- 通过边框、间距、背景色等设计元素,提高界面的美观度和可读性。
- 选择了深紫色作为头部背景色,与应用主题保持一致,增强品牌识别度。
-
异常处理
- 添加了索引越界异常处理,提高代码的健壮性。
- 考虑了各种边界情况,确保组件在不同数据状态下都能正常工作。
-
性能优化
- 采用懒加载技术,只渲染可见区域的项目,提高列表滚动性能。
- 优化组件的
build方法,避免不必要的计算和渲染。 - 使用
const构造函数和常量,减少不必要的重建。
-
代码组织
- 创建了专门的
components目录,集中管理所有组件文件,提高代码组织结构的清晰度。 - 将数据模型和UI组件分离到不同的文件中,提高代码的可维护性。
- 使用清晰的命名规范,便于代码的理解和维护。
- 创建了专门的
-
状态管理
- 使用
StatefulWidget和State管理页面状态,为后续可能的交互功能做准备。 - 在
initState方法中初始化数据,确保数据在组件构建前准备就绪。
- 使用
-
应用配置
- 配置了
MaterialApp的主题和调试横幅,提高应用的美观度。 - 使用
ColorScheme.fromSeed创建主题,确保应用整体风格的一致性。
- 配置了
通过本次开发,我们成功实现了一个功能完整、性能优良的分组列表组件,展示了 Flutter for OpenHarmony 平台上的组件化开发能力。该组件不仅可以直接应用于实际项目中,也为类似功能的开发提供了参考和借鉴。
欢迎加入开源鸿蒙跨平台社区: https://openharmonycrossplatform.csdn.net
更多推荐


所有评论(0)