Flutter列表管理插件bloc_list的使用

发布于 1周前 作者 nodeper 来自 Flutter

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

1 回复

更多关于Flutter列表管理插件bloc_list的使用的实战系列教程也可以访问 https://www.itying.com/category-92-b0.html


当然,以下是一个关于如何使用 bloc_list 插件在 Flutter 中管理列表的示例代码。bloc_list 是一个用于将 Bloc 模式与 Flutter 列表视图结合的插件,使得状态管理和列表渲染更加高效和简洁。

首先,确保你已经在 pubspec.yaml 文件中添加了 flutter_blocbloc 依赖:

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 插件提供了类似的功能)来管理列表状态,并结合 BlocProviderBlocBuilder 来构建响应式 UI。

回到顶部