多选复选框的 Flutter 问题 - 来自 Firestore 的数据

标签 flutter dart google-cloud-firestore

以下代码显示我的收藏 (Firestore) 中列出的项目 我正在尝试创建检查任何项目的能力,然后将这些项目存储到下一个屏幕上的“收藏夹”中。

目前,复选框是全有或全无。所有项目都未选中或点击后选中。

class _SelectScreenState extends State<SelectScreen> {
  bool _isChecked = false;

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: Text('Select Exercises')),
      body: _buildBody(context),
    );
   }

   Widget _buildBody(BuildContext context) {
    return StreamBuilder<QuerySnapshot>(
      stream: Firestore.instance.collection('exercises').snapshots(),
      builder: (context, snapshot) {
        if (!snapshot.hasData) return LinearProgressIndicator();

        return _buildList(context, snapshot.data.documents);
      },
    );
   }

   Widget _buildList(BuildContext context, List<DocumentSnapshot> snapshot) 
   {
    return ListView(
      padding: const EdgeInsets.only(top: 20.0),
      children: snapshot.map((data) => _buildListItem(context, 
   data)).toList(),
    );
  }

   Widget _buildListItem(BuildContext context, DocumentSnapshot data) {
    final record = Record.fromSnapshot(data);

    return Padding(
      key: ValueKey(record.name),
      padding: const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0),
      child: Container(
        decoration: BoxDecoration(
          border: Border.all(color: Colors.grey),
          borderRadius: BorderRadius.circular(5.0),
        ),
        child: ListTile(
          title: Text(record.name),
          trailing: Checkbox(
            value: _isChecked,
            onChanged: (bool value) {
            setState(() {
              _isChecked = value;
            });
          },
          )
        ),
        ),
       );
     }
    }

    class Record {
    final String name;
    final DocumentReference reference;

    Record(this.name, this.reference);

    Record.fromMap(Map<String, dynamic> map, {this.reference})
      : assert(map['name'] != null),
        name = map['name'];

   Record.fromSnapshot(DocumentSnapshot snapshot)
      : this.fromMap(snapshot.data, reference: snapshot.reference);

    @override
    String toString() => "Record<$name:>";
    }

最佳答案

这是因为您对所有复选框都使用了一个变量。

要解决这个问题,您可以创建一个专用的有状态小部件,它会独立于其他复选框处理每个复选框的状态。

所以你可以用类似的东西替换你的 ListTile

Exercise(
 title: record.name,
)

然后您可以按如下方式定义Exercise 小部件

class Exercise extends StatefulWidget {
  final String title;

  Exercise({this.title});

  @override
  _ExerciseState createState() => _ExerciseState();
}

class _ExerciseState extends State<Exercise> {
  bool selected = false;

  @override
  Widget build(BuildContext context) {
    return ListTile(
      title: Text(widget.title),
      trailing: Checkbox(
          value: selected,
          onChanged: (bool val) {
            setState(() {
              selected = val;
            });
          }),
    );
  }
}

这是一个完整的工作示例

import 'package:flutter/material.dart';

void main() => runApp(MyApp());

class MyApp extends StatelessWidget {
  // This widget is the root of your application.
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: 'Flutter Demo',
      theme: ThemeData(
        primarySwatch: Colors.blue,
      ),
      home: MyHomePage(title: 'Flutter Demo Home Page'),
    );
  }
}

class MyHomePage extends StatefulWidget {
  MyHomePage({Key key, this.title}) : super(key: key);

  final String title;

  @override
  _MyHomePageState createState() => _MyHomePageState();
}

class _MyHomePageState extends State<MyHomePage> {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text(widget.title),
      ),
      body: ListView(
        children: <Widget>[
          Exercise(
            title: "Exercises 1",
          ),
          Exercise(
            title: "Exercises 2",
          ),
        ],
      ), // This trailing comma makes auto-formatting nicer for build methods.
    );
  }
}

class Exercise extends StatefulWidget {
  final String title;

  Exercise({this.title});

  @override
  _ExerciseState createState() => _ExerciseState();
}

class _ExerciseState extends State<Exercise> {
  bool selected = false;

  @override
  Widget build(BuildContext context) {
    return ListTile(
      title: Text(widget.title),
      trailing: Checkbox(
          value: selected,
          onChanged: (bool val) {
            setState(() {
              selected = val;
            });
          }),
    );
  }
}

checkbox

关于多选复选框的 Flutter 问题 - 来自 Firestore 的数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56953911/

相关文章:

flutter 错误 : The constructor being called isn't a const constructor

flutter - 如何更改 flutter showAboutDialog 中的文本按钮颜色?

google-app-engine - 启动 Dart 托管 VM 时出现 gcloud dev_appserver.py 错误

Firebase Firestore : Is there an equivalent of @SerializedName?

flutter - 按日期分组的 ListView Dart

Flutter:转义字符串中的美元符号 '$' 字符

android - 如何覆盖应用程序的一部分主题?

flutter - 如何将静态变量转换为常量并将其传递给期望最终值的类构造函数

firebase - 我如何添加加载圆形图标,因为从Firestore提取数据需要时间?

java - 如何将快照监听器正确添加到 Java 中的 Firebase Firestore 文档引用?