Flutter, FieldValue.removeArray () не работает - PullRequest
0 голосов
/ 25 января 2020

Я пытаюсь удалить элементы из списка в базе данных firestore, но это не работает. Что я нахожу странным, так это то, что FieldValue.arrayUnion () и FieldValue.delete () работают нормально. Почему FieldValue.arrayRemove () не работает только один? Спасибо.

Служба БД

 import 'package:cloud_firestore/cloud_firestore.dart';
    import 'package:firebase_auth/firebase_auth.dart';
    import "package:innovative_world/models/list_model.dart";

    class DatabaseService {
      // Collection Reference
      CollectionReference listCollection = Firestore.instance.collection("list");

      // Get current users id
      final String uid;
      DatabaseService({this.uid});

      // Set data to firestore db
      Future setUserData(List<String> list) async {
        return await listCollection.document(uid).setData({ 
          "list": list
        });
      }

      // DOES NOT WANT TO REMOVE ELEMENTS
      Future deleteListArr(int index) async {
        return await listCollection.document(uid)
          .updateData({ "list": FieldValue.arrayRemove([index]) });
      }

      // UserList snapshot
      UserList _userListFromSnapshot(DocumentSnapshot snapshot) {
        return UserList(
          uid: uid, 
          list: snapshot.data["list"].cast<String>().toList()
          ); 
      } 
      // Stream for user's to do list
      Stream<UserList> get userListStream {
        return listCollection.document(uid).snapshots()
          .map(_userListFromSnapshot);
      }
    }

Home

  import "package:flutter/material.dart";
    import 'package:flutter/rendering.dart';
    import "package:innovative_world/services/auth_service.dart";
    import "package:innovative_world/models/list_model.dart";
    import 'package:innovative_world/services/database_service.dart';
    import "package:innovative_world/shared/decoration.dart";
    import 'package:innovative_world/shared/loading.dart';
    import 'package:provider/provider.dart';
    import "package:innovative_world/models/user_model.dart";
    import "package:innovative_world/models/list_model.dart";


    class Home extends StatefulWidget {
      @override
      _HomeState createState() => _HomeState();
    }

    class _HomeState extends State<Home> {
      String _text;
      final _formKey = GlobalKey<FormState>();
      List<String> userDoc = [];


      @override
      Widget build(BuildContext context) {
        final user = Provider.of<UserId>(context);

        return StreamBuilder<UserList>(
            stream: DatabaseService(uid: user.uid).userListStream,
            builder: (context, snapshot) {
              if (snapshot.hasData) {
                UserList userList = snapshot.data;
                // List<String> userDoc = [userList.list.toString()];
                return Scaffold(
                  backgroundColor: Colors.blue[200],
                  appBar: AppBar(
                    title: Text(
                      "Create List",
                      style: TextStyle(
                        fontSize: 23.0,
                      ),
                    ),
                    elevation: 0.0,
                    actions: <Widget>[
                      FlatButton.icon(
                        onPressed: () {
                          AuthService().signOut();
                        },
                        icon: Icon(Icons.person),
                        label: Text("Sign out"),
                      ),
                    ],
                  ),
                  body: Padding(
                    padding: const EdgeInsets.fromLTRB(20.0, 25.0, 20.0, 0.0),
                    child: Column(
                      children: <Widget>[
                        Form(
                          key: _formKey,
                          child: Column(
                            children: <Widget>[
                              TextFormField(
                                validator: (val) =>
                                    val.isEmpty ? "Enter text" : null,
                                onChanged: (val) => setState(() => _text = val),
                                decoration: InputDecorationConst.copyWith(
                                    hintText: "Create List..."),
                              ),
                              RaisedButton(
                                onPressed: () async {
                                  if (_formKey.currentState.validate()) {
                                    userDoc.add(_text);

                                    _formKey.currentState.reset();
                                    await DatabaseService(uid: user.uid)
                                        .setUserData(userDoc ?? userList.list);
                                  }
                                },
                                color: Colors.pink,
                                child: Text(
                                  "Add to list",
                                  style: TextStyle(
                                      fontSize: 15.0, color: Colors.white),
                                ),
                              ),
                            ],
                          ),
                        ),
                        SizedBox(height: 15.0),
                        ListView.builder(
                          shrinkWrap: true,
                          itemCount: userList.list.length,
                          itemBuilder: (context, index) {
                            return ForList(index: index, theList: userList.list);
                          },
                        ),
                      ],
                    ),
                  ),
                );
              } else {
                return Loading();
              }
            });
      }
    }

    class ForList extends StatefulWidget {
      final int index;
      final List<String> theList;

      ForList({this.index, this.theList});

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

    class _ForListState extends State<ForList> {
      bool isSelected = false;

      @override
      Widget build(BuildContext context) {
        final user = Provider.of<UserId>(context);

        return Column(
          children: <Widget>[
            Card(
              child: CheckboxListTile(
                title: Text(widget.theList[widget.index]),
                value: isSelected,
                onChanged: (bool val)  {
                  setState(() {
                    isSelected = val;
                    if (isSelected) {
                    DatabaseService(uid: user.uid).deleteListArr(widget.index);
                    }
                  });
                },
              ),
            ),
          ],
        );
      }
    }

Справа выше у меня есть отдельный виджет ForList, который содержит snapshot.data и индекс мне кажется, что все хорошо, но я не могу понять, почему он не удалит элемент из БД

1 Ответ

1 голос
/ 28 января 2020

Причина, по которой arrayRemove () не работала, заключалась в том, что ему требовалось не только индекс, но и строка, с которой он соотносился.

Служба БД

import 'package:cloud_firestore/cloud_firestore.dart';
import 'package:firebase_auth/firebase_auth.dart';
import "package:innovative_world/models/list_model.dart";

class DatabaseService {
  // Collection Reference
  CollectionReference listCollection = Firestore.instance.collection("list");

  // Get current users id
  final String uid;
  DatabaseService({this.uid});

  // Set data to firestore db
  Future setUserData(List<String> list) async {
    return await listCollection.document(uid).setData({ 
      "list": list
    });
  }

  // For deleting the lists
  Future deleteListArr(String index) async {
    return await listCollection.document(uid)
      .updateData({ "list": FieldValue.arrayRemove([index]) });
  }



  // UserList snapshot
  UserList _userListFromSnapshot(DocumentSnapshot snapshot) {
    return UserList(
      uid: uid, 
      list: snapshot.data["list"].cast<String>().toList()
      ); 
  } 
  // Stream for user's to do list
  Stream<UserList> get userListStream {
    return listCollection.document(uid).snapshots()
      .map(_userListFromSnapshot);
  }
}

Home

import "package:flutter/material.dart";
import 'package:flutter/rendering.dart';
import "package:innovative_world/services/auth_service.dart";
import "package:innovative_world/models/list_model.dart";
import 'package:innovative_world/services/database_service.dart';
import "package:innovative_world/shared/decoration.dart";
import 'package:innovative_world/shared/loading.dart';
import 'package:provider/provider.dart';
import "package:innovative_world/models/user_model.dart";
import "package:innovative_world/models/list_model.dart";
import 'dart:async';

class Home extends StatefulWidget {
  @override
  _HomeState createState() => _HomeState();
}

class _HomeState extends State<Home> {
  String _text;
  final _formKey = GlobalKey<FormState>();
  List<String> userDoc = [];
  // On every refresh the userDoc starts from 0, so it overrides the data

  @override
  Widget build(BuildContext context) {
    final user = Provider.of<UserId>(context);

    return StreamBuilder<UserList>(
        stream: DatabaseService(uid: user.uid).userListStream,
        builder: (context, snapshot) {
          if (snapshot.hasData) {
            UserList userList = snapshot.data;
            // List<String> userDoc = [userList.list.toString()];
            return Scaffold(
              backgroundColor: Colors.blue[200],
              appBar: AppBar(
                title: Text(
                  "Create List",
                  style: TextStyle(
                    fontSize: 23.0,
                  ),
                ),
                elevation: 0.0,
                actions: <Widget>[
                  FlatButton.icon(
                    onPressed: () {
                      AuthService().signOut();
                    },
                    icon: Icon(Icons.person),
                    label: Text("Sign out"),
                  ),
                ],
              ),
              body: Padding(
                padding: const EdgeInsets.fromLTRB(20.0, 25.0, 20.0, 0.0),
                child: Column(
                  children: <Widget>[
                    Form(
                      key: _formKey,
                      child: Column(
                        children: <Widget>[
                          TextFormField(
                            validator: (val) =>
                                val.isEmpty ? "Enter text" : null,
                            onChanged: (val) => setState(() => _text = val),
                            decoration: InputDecorationConst.copyWith(
                                hintText: "Create List..."),
                          ),
                          RaisedButton(
                            onPressed: () async {
                              if (_formKey.currentState.validate()) {
                                userDoc.add(_text);
                                _formKey.currentState.reset();
                                await DatabaseService(uid: user.uid)
                                    .setUserData(userDoc ?? userList.list);
                              }
                            },
                            color: Colors.pink,
                            child: Text(
                              "Add to list",
                              style: TextStyle(
                                  fontSize: 15.0, color: Colors.white),
                            ),
                          ),
                        ],
                      ),
                    ),
                    SizedBox(height: 15.0),
                    ListView.builder(
                      shrinkWrap: true,
                      itemCount: userList.list.length,
                      itemBuilder: (context, index) {
                        return ForList(listsIndex: userList.list[index], index: index, theList: userList.list);
                      },
                    ),
                  ],
                ),
              ),
            );
          } else {
            return Loading();
          }
        });
  }
}

class ForList extends StatefulWidget {
  final int index;
  final String listsIndex;
  final List<String> theList;

  ForList({ this.listsIndex, this.index, this.theList });

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

class _ForListState extends State<ForList> {
  bool isSelected = false;

  @override
  Widget build(BuildContext context) {
    final user = Provider.of<UserId>(context);

    return Column(
      children: <Widget>[
        Card(
          child: CheckboxListTile(
            title: Text(widget.theList[widget.index]),
            value: isSelected,
            onChanged: (bool val) {
              setState(() {
                isSelected = val;
                Timer(Duration(seconds: 1), () {
                  setState(() async {
                  if (isSelected) {
                    await DatabaseService(uid: user.uid).deleteListArr(widget.listsIndex);

                    }
                  });
                });
              });
            },
          ),
        ),
      ],
    );
  }
}

Теперь я добавил новый аргумент в экземпляр ForList, listsIndex, который содержит строку рядом с индексом.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...