在Flutter中选择下拉项时出错

问题描述 投票:5回答:3

我有一个Flutter下拉列表的问题。当我选择其中一个项时,它会抛出一个错误:

引发了另一个例外:'package:flutter / src / material / dropdown.dart':断言失败:第481行第15行:'value == null || items.where((DropdownMenuItem item)=> item.value == value)。length == 1':不是真的。

我正在搜索并且人们告诉我产生了这个错误,因为所选元素不属于原始列表,但经过一些调试我可以看到它。我无法找到此错误的来源,所以我将不胜感激任何帮助。

这是我的代码

FeedCategory模型

import 'package:meta/meta.dart';

class FeedCategory {
  static final dbId = "id";
  static final dbName = "name";

  int id;
  String name;

  FeedCategory({this.id, @required this.name});

  FeedCategory.fromMap(Map<String, dynamic> map)
      : this(
    id: map[dbId],
    name: map[dbName],
  );

  Map<String, dynamic> toMap() {
    return {
      dbId: id,
      dbName: name,
    };
  }

  @override
  String toString() {
    return 'FeedCategory{id: $id, name: $name}';
  }
}

窗口小部件

import 'package:app2date/repository/repository.dart';
import 'package:app2date/model/FeedSource.dart';
import 'package:app2date/model/FeedCategory.dart';
import 'package:app2date/util/ui.dart';
import 'package:flutter/material.dart';

class ManageFeedSource extends StatefulWidget {
  ManageFeedSource({Key key, this.feedSource}) : super(key: key);

  final FeedSource feedSource;

  @override
  _ManageFeedSource createState() => new _ManageFeedSource();
}

class _ManageFeedSource extends State<ManageFeedSource> {
  FeedCategory _feedCategory;

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text('New Feed'),
      ),
      body: new FutureBuilder(
        future: Repository.get().getFeedCategories(),
        builder: (BuildContext context, AsyncSnapshot snapshot) {
          List<FeedCategory> categoriesList = snapshot.data;
          if (categoriesList != null) {
            return new DropdownButton<FeedCategory>(
              hint: Text('Choose category...'),
              value: _feedCategory,
              items: categoriesList.map((FeedCategory category) {
                return DropdownMenuItem<FeedCategory>(
                  value: category,
                  child: Text(category.name),
                );
              }).toList(),
              onChanged: (FeedCategory category) {
                print('Selected: $category');
                setState(() {
                  _feedCategory = category;
                });
              },
            );
          } else {
            return Container(
              decoration: new BoxDecoration(color: Colors.white),
            );
          }
        },
      ),
    );
  }

  @override
  void initState() {
    super.initState();
  }
}

存储库getFeedCategories方法

Future<List<FeedCategory>> getFeedCategories() async {
  return await database.getFeedCategories();
}

数据库getFeedCategories方法

Future<List<FeedCategory>> getFeedCategories() async {
  var dbClient = await db;
  var query = "SELECT * FROM $feedCategoryTableName;";
  var result = await dbClient.rawQuery(query);
  List<FeedCategory> feedCategories = [];
  for (Map<String, dynamic> item in result) {
    feedCategories.add(new FeedCategory.fromMap(item));
  }
  return feedCategories;
}

categoriesList内容和所选类别(调试器)enter image description here

flutter dropdown selected
3个回答
6
投票

我想我已经找到了你的问题。它源于你如何使用FutureBuilder。

这就是我的想法:

  1. 你的应用程序正在运行它调用getFeedCategories()
  2. 未来完成,列表已建立,具有以下项目: obj 123 ==> FeedCategory(Prueba)obj 345 ==> FeedCategory(Categories 2)
  3. 从下拉列表中选择项目,调用setState() 您的_feedCategory现在等于obj 123 ==> FeedCategory(Prueba)
  4. 小部件重建。它再次调用getFeedCategories()
  5. 未来完成,列表建立等与以下项目 obj 567 ==> FeedCategory(Prueba)obj 789 ==> FeedCategory(Categories 2)
  6. Dropdown测试items.where((DropdownMenuItem item) => item.value == value).length == 1,但长度== 0因为找不到obj 123 ==> FeedCategory(Prueba)

您的问题有几种解决方案。一种方法是将equals operator添加到你的FeedCategory类中,比较类别和/或id。

另一种方法是将futurebuilder中使用的Future与更改的部分分开 - 你可以通过将未来保持为成员变量(也可以在initState中实例化它)或者通过将构建器的内部部分放入其中来实现这一点。自己的有状态小部件(在这种情况下,您可以使ManageFeedSource成为StatelessWidget)。我推荐最后一个选项。

如果这不能解决您的问题,请告诉我,但我很确定这就是它正在做的事情的原因。


2
投票

要完成rmtmckenzie answer,这里是您应该放在FeedCategory对象中的代码:

bool operator ==(o) => o is FeedCategory && o.name == name;
int get hashCode => name.hashCode;

编辑:一个link,它解释了在Flutter中为hashCode做了什么


0
投票

当我使用实际上不在下拉列表中的值初始化下拉列表时,我收到此错误,该错误消息中不清楚。

© www.soinside.com 2019 - 2024. All rights reserved.