Flutter列表管理插件bloc_list的使用
Flutter列表管理插件bloc_list的使用
准备工作
一个灵活且可定制的列表小部件,用于使用BLoC模式进行状态管理的Flutter应用程序。它提供了动态数据加载、状态相关的渲染、自定义加载器支持以及错误处理,非常适合创建响应式和用户友好的列表视图。
使用方法
示例代码
void main() {
runApp(MultiBlocProvider(
providers: [
BlocProvider(create: (_) => TodoBloc(DataService())),
],
child: const MainApp(),
));
}
主页面构建
[@override](/user/override)
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(title: const Text('Todo List')),
floatingActionButton: FloatingActionButton(
shape: const CircleBorder(),
onPressed: () => _addOrUpdateTodoItem(context),
backgroundColor: Theme.of(context).primaryColor,
child: const Icon(Icons.add, color: Colors.white),
),
body: BlocList<TodoModel, TodoBloc, ListState>(
emptyBuilder: (context, state) {
return const Center(child: Text('请添加一个待办事项'));
},
loadBuilder: (context, state) {
return const Center(
child: CircularProgressIndicator(
color: Colors.black,
));
},
onItemError: (errorType, item, errorMessage) {
item.isBusy = false;
_displaySnackBar(context, errorMessage);
},
onItemAdding: (addingItem) {
addingItem.isBusy = true;
},
onItemDeleting: (deletingItem) {
deletingItem.isBusy = true;
},
onItemUpdating: (newItem, oldItem) {
oldItem.isBusy = true;
},
onItemDeleted: (deletedItem) {
_displaySnackBar(context, "已删除 ${deletedItem.description}");
},
onItemUpdated: (newItem, oldItem) {
_displaySnackBar(context,
"更新了 ${oldItem.description} 为 ${newItem.description}");
},
onItemAdded: (addedItem) {
addedItem.isBusy = false;
_displaySnackBar(context, "添加了 ${addedItem.description}");
},
itemBuilder: (context, index, item) {
var outputFormat = DateFormat('dd MMM yyyy HH:mm');
var date = outputFormat.format(item.created);
return ListTile(
title: Text(
item.description,
style:
TextStyle(color: item.isBusy ? Colors.grey : Colors.black),
),
trailing: Row(
mainAxisSize: MainAxisSize.min,
children: [
IconButton(
icon: Icon(Icons.edit,
color: item.isBusy ? Colors.grey : Colors.black),
onPressed: () {
_addOrUpdateTodoItem(context, index: index, item: item);
},
),
IconButton(
icon: Icon(Icons.delete,
color: item.isBusy ? Colors.grey : Colors.black),
onPressed: () {
_deleteDialog(context, () {
BlocProvider.of<TodoBloc>(context)
.add(DeleteDataEvent(item));
});
},
),
],
),
subtitle: Padding(
padding: const EdgeInsets.only(top: 5.0),
child: Text(
date,
style: TextStyle(
color: item.isBusy ? Colors.grey : Colors.black54),
),
),
);
},
loadData: () async {
BlocProvider.of<TodoBloc>(context).add(LoadDataEvent());
},
stateCondition: (state) => state,
));
}
TodoBloc类定义
class TodoBloc extends ListBloc<TodoModel> {
final DataService dataService;
TodoBloc(this.dataService)
: super(
dataProvider: ([id]) => dataService.getTodoList(id),
dataAdder: (item) => dataService.addTodo(item),
dataDeleter: (item) => dataService.deleteTodo(item),
dataSorter: (items, [compare]) =>
items.sort((a, b) => b.created.compareTo(a.created)),
dataUpdater: (item) => dataService.updateTodo(item),
);
}
获取待办事项列表的方法
Future<BlocResponse<List<TodoModel>>> getTodoList([int? id]) async {
final response = await http
.get(Uri.parse('https://example.com/todos'));
if (response.statusCode == 200) {
List<TodoModel> todos = (jsonDecode(response.body) as List<dynamic>)
.map((e) => TodoModel.fromJson(e as Map<String, dynamic>))
.toList();
return BlocResponse<List<TodoModel>>(success: true, data: todos);
} else {
return BlocResponse<List<TodoModel>>(
success: false, message: "无法加载待办事项");
}
}
示例演示
完整示例代码
import 'package:example/blocs/chat_bloc.dart';
import 'package:example/blocs/todo_bloc.dart';
import 'package:example/data/data_service.dart';
import 'package:example/screens/chat.dart';
import 'package:example/screens/home_page.dart';
import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
void main() {
runApp(MultiBlocProvider(
providers: [
BlocProvider(create: (_) => TodoBloc(DataService())),
BlocProvider(create: (_) => ChatBloc(DataService())),
],
child: const MainApp(),
));
}
class MainApp extends StatelessWidget {
const MainApp({super.key});
[@override](/user/override)
Widget build(BuildContext context) {
// return const MaterialApp(home: TodoListPage());
return const MaterialApp(home: HomePage());
}
}
更多关于Flutter列表管理插件bloc_list的使用的实战系列教程也可以访问 https://www.itying.com/category-92-b0.html
更多关于Flutter列表管理插件bloc_list的使用的实战系列教程也可以访问 https://www.itying.com/category-92-b0.html
当然,以下是一个关于如何使用 bloc_list
插件在 Flutter 中管理列表的示例代码。bloc_list
是一个用于将 Bloc 模式与 Flutter 列表视图结合的插件,使得状态管理和列表渲染更加高效和简洁。
首先,确保你已经在 pubspec.yaml
文件中添加了 flutter_bloc
和 bloc
依赖:
dependencies:
flutter:
sdk: flutter
flutter_bloc: ^8.0.0 # 请使用最新版本
bloc: ^8.0.0 # 请使用最新版本
然后,你可以按照以下步骤创建一个简单的列表管理应用。
1. 定义 Cubit 和状态
首先,我们需要定义一个 Cubit 来管理列表的状态。假设我们要管理一个待办事项列表。
import 'package:bloc/bloc.dart';
class Todo {
final String title;
final bool isCompleted;
Todo({required this.title, required this.isCompleted});
}
class TodoCubit extends Cubit<List<Todo>> {
TodoCubit() : super([]);
void addTodo(String title) {
emit([...state, Todo(title: title, isCompleted: false)]);
}
void toggleTodoCompletion(int index) {
final newTodos = List.from(state);
if (index >= 0 && index < newTodos.length) {
newTodos[index] = newTodos[index].copyWith(isCompleted: !newTodos[index].isCompleted);
}
emit(newTodos);
}
void deleteTodo(int index) {
emit(state.where((todo, i) => i != index).toList());
}
}
2. 创建 UI
接下来,我们创建一个 Flutter 应用,使用 BlocProvider
提供 TodoCubit
,并使用 BlocBuilder
来监听状态变化并更新 UI。
import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:your_app/cubit/todo_cubit.dart'; // 确保路径正确
void main() {
runApp(MyApp());
}
class MyApp extends StatelessWidget {
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo',
theme: ThemeData(
primarySwatch: Colors.blue,
),
home: BlocProvider(
create: (context) => TodoCubit(),
child: TodoListScreen(),
),
);
}
}
class TodoListScreen extends StatelessWidget {
@override
Widget build(BuildContext context) {
final todoCubit = context.read<TodoCubit>();
return Scaffold(
appBar: AppBar(
title: Text('Todo List'),
actions: [
IconButton(
icon: Icon(Icons.add),
onPressed: () async {
final title = await showDialog<String>(
context: context,
builder: (context) => AlertDialog(
title: Text('Add Todo'),
content: TextField(
decoration: InputDecoration(hintText: 'Enter todo title'),
onEditingComplete: () => Navigator.of(context).pop(textFieldController.text),
controller: TextEditingController()..text = "",
),
actions: [
TextButton(
onPressed: () => Navigator.of(context).pop(),
child: Text('Cancel'),
),
TextButton(
onPressed: () {
final text = textFieldController.text.trim();
if (text.isNotEmpty) {
Navigator.of(context).pop(text);
}
},
child: Text('Add'),
),
],
),
);
if (title != null) {
todoCubit.addTodo(title);
}
},
),
],
),
body: BlocBuilder<TodoCubit, List<Todo>>(
builder: (context, todos) {
return ListView.builder(
itemCount: todos.length,
itemBuilder: (context, index) {
final todo = todos[index];
return Dismissible(
key: Key(todo.title),
direction: DismissDirection.horizontal,
onDismissed: (direction) {
todoCubit.deleteTodo(index);
},
child: ListTile(
leading: Checkbox(
value: todo.isCompleted,
onChanged: (value) {
todoCubit.toggleTodoCompletion(index);
},
),
title: Text(todo.title),
),
);
},
);
},
),
);
}
}
3. 运行应用
确保所有文件路径正确,然后运行你的 Flutter 应用。你应该会看到一个简单的待办事项列表应用,可以添加、删除和切换待办事项的完成状态。
这个示例展示了如何使用 bloc_list
(尽管实际上没有特定的 bloc_list
插件,但 flutter_bloc
插件提供了类似的功能)来管理列表状态,并结合 BlocProvider
和 BlocBuilder
来构建响应式 UI。