Трепетать, как избегать построения виджетов снова и снова при добавлении данных на основе страниц в FutureBuilder - PullRequest
0 голосов
/ 27 апреля 2020

Я работаю над виджетом будущего строителя, который показывает список ресторанов. Поскольку список огромен, у меня есть данные на основе страниц, поэтому, когда контроллер Scroll достигает конца представления в мобильном устройстве, я увеличиваю счетчик страниц. В этом случае мой API вызывается и данные добавляются, но весь виджет перестраивается и это не правильно. Ниже приведен код:


import 'package:flutter/material.dart';
import 'package:timeout_dubai/podo/RestaurantsData.dart';
import 'package:timeout_dubai/screens/ArticleDetails.dart';
import 'package:timeout_dubai/screens/BarsFliterScreen.dart';
import 'package:timeout_dubai/screens/TopPicsRestoScreen.dart';
import 'package:timeout_dubai/services/getRestaurantsData.dart';
import 'package:timeout_dubai/utils/RatingWidget.dart';
import 'package:timeout_dubai/constants/constants.dart' as Constants;

/*
* @author Sagar Chavan
* @since 14-feb-2020
* @version 1.0.0
* @class RestaurantFragment shows list of Restaurants
* */

class RastaurantsFragment extends StatefulWidget {
  RastaurantsFragment({Key key}) : super(key: key);

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

class RastaurantsFragmentState extends State<RastaurantsFragment> {
  List data;
  final GetRestaurantsData httpService = GetRestaurantsData();

  var placeholder = "assets/images/placeholderimage.jpg";
  ScrollController _sc = new ScrollController();
  static int page = 0;
  static int tagid = 0;

  String toppickstext = "TIME OUT TOP PICKS";
  Future<List<RestaurantsData>> filteredrestodata;

  String SortType = "PUB_DATE";
  Future<List<RestaurantsData>> restodata;
  final GlobalKey<State> list1Key = new GlobalKey<State>();
  bool loadmore = false;

  @override
  void initState() {
    //getSortOption("PUB_DATE");
    restodata = httpService.getRestaurantsData(page, tagid, SortType, loadmore);

    // TODO: implement initState
    super.initState();

    _sc.addListener(() {
      if (_sc.position.pixels == _sc.position.maxScrollExtent) {
        loadmore = true;
        _getMoreData(page,loadmore);
      }
    });
  }

  Future<List<RestaurantsData>> _getMoreData(int index, bool loadmore) async {
    restodata = httpService.getRestaurantsData(index, tagid, SortType, loadmore);
    setState(() {
      //loadmore = loadmore;
      page++;
      restodata = httpService.getRestaurantsData(page, tagid, SortType, loadmore);

    });
  }

  goToArticleDetails(String nid, String image, String category, int rating) {
    Navigator.push(
        context,
        MaterialPageRoute(
            builder: (context) => ArticleDetails(nid, image, category, rating)));
  }

  getSortOption(String sortingtype) async {

    setState(() {
      SortType = sortingtype;
      restodata = httpService.getRestaurantsData(page, tagid, sortingtype, loadmore);
    });

  }

  @override
  void dispose() {
//    _bannerAd?.dispose();
//    _interstitialAd?.dispose();
    _sc.dispose();
    super.dispose();
  }

  @override
  Widget build(BuildContext context) {
    print("RESTOLISTBUILD ${restodata.then((value) => print(value.length))}");


    return new Scaffold(
      body: FutureBuilder(
        future: restodata,
        builder: (BuildContext context,
            AsyncSnapshot<List<RestaurantsData>> snapshot) {
          print("DataTable:  ${snapshot.hasData}");
          if (snapshot.connectionState == ConnectionState.waiting) {
            return Center(child: CircularProgressIndicator());
          } else {
            if (snapshot.data?.isEmpty ?? true)
              return Center(
                  child: Text(
                'No data found',
                style: TextStyle(
                    color: Colors.black,
                    fontSize: 25,
                    fontWeight: FontWeight.w700),
              ));
            else
              List<RestaurantsData> posts = snapshot.data;
            //print("posts:  ${snapshot.data[index].template}");
            return NotificationListener<ScrollNotification>(
              // ignore: missing_return
              onNotification: (ScrollNotification scrollInfo) {
                if (scrollInfo.metrics.pixels ==
                    scrollInfo.metrics.maxScrollExtent) {
                  print("Scroll End");
                }
              },
              child: ListView.builder(
                  controller: _sc,
                  itemCount: snapshot.data.length,
                  itemBuilder: (BuildContext context, int index) {
                    return Container(
                      padding: EdgeInsets.fromLTRB(0, 0, 0, 2),
                      child: GestureDetector(
                        child: Stack(
                          children: <Widget>[
                            ShaderMask(
                                shaderCallback: (rect) {
                                  return LinearGradient(
                                    begin: Alignment.topCenter,
                                    end: Alignment.bottomCenter,
                                    colors: [
                                      Constants.gradientStart,
                                      Constants.gradientEnd
                                    ],
                                  ).createShader(Rect.fromLTRB(
                                      0, 20, rect.width, rect.height - 20));
                                },
                                blendMode: BlendMode.darken,
                                child: Container(
                                  width: double.infinity,
                                  child: FadeInImage.assetNetwork(
                                    placeholder: placeholder,
                                    image: snapshot.data[index].image,
                                    fit: BoxFit.cover,
                                  ),
                                )),
                            if (snapshot.data[index].sponsored == 1)
                              Positioned(
                                  top: 20,
                                  left: 20,
                                  child: Container(
                                    padding: EdgeInsets.all(5),
                                    decoration: new BoxDecoration(
                                        border: new Border.all(
                                            width: 0,
                                            color: Colors
                                                .transparent), //color is transparent so that it does not blend with the actual color specified
                                        borderRadius: const BorderRadius.all(
                                            const Radius.circular(30.0)),
                                        color: Colors.black.withOpacity(
                                            0.4) // Specifies the background color and the opacity
                                        ),
                                    child: Text(
                                      "SPONSORED",
                                      style: TextStyle(color: Colors.white),
                                    ),
                                  )),
                            Positioned(
                              top: 20,
                              right: 20,
                              child: IconButton(
                                  icon: Icon(Icons.favorite, color:Colors.white ,),
                                  color: Colors.white
                                  //onPressed: () => removeFavourites(snapshot.data[index].nid),
                                  ),
                            ),
                            Positioned(
                              bottom: 20,
                              left: 20,
                              child: Column(
                                crossAxisAlignment: CrossAxisAlignment.start,
                                children: <Widget>[
                                  Container(
                                    width: 300,
                                    child: Text(
                                      "${snapshot.data[index].title}",
                                      softWrap: true,
                                      style: TextStyle(
                                        fontSize: 24,
                                        fontFamily: 'franklin-gothic-demi',
                                        fontWeight: FontWeight.bold,
                                        color: Color(0xffFFFFFF),
                                      ),
                                    ),
                                  ),
                                  Padding(
                                    padding: EdgeInsets.only(top: 5),
                                  ),
                                  if (snapshot.data[index].editor_rating !=
                                      null)
                                    RatingWidget(
                                        data:
                                            "${snapshot.data[index].editor_rating}"),
                                  Padding(
                                    padding: EdgeInsets.only(top: 5),
                                  ),
                                  if (snapshot
                                      .data[index].oarea_name.isNotEmpty)
                                    Row(
                                      children: <Widget>[
                                        Icon(
                                          Icons.location_on,
                                          color: Colors.white,
                                          size: 15,
                                        ),
                                        Text(
                                          "${snapshot.data[index].oarea_name}",
                                          style: TextStyle(
                                            fontSize: 18,
                                            fontFamily:
                                                'franklin-gothic-book-cnd.otf',
                                            color: Colors.white,
                                          ),
                                        ),
                                      ],
                                    ),
                                ],
                              ),
                            ),
                          ],
                        ),
                        onTap: () => goToArticleDetails(
                            "${snapshot.data[index].nid}",
                            snapshot.data[index].image,
                            "Restaurants",
                            snapshot.data[index].editor_rating,
                        ),
                      ),
                    );
                  }),
            ); // snapshot.data  :- get your object which is pass from your downloadData() function
          }
        },
      ),
      floatingActionButton: Row(
        mainAxisAlignment: MainAxisAlignment.center,
        children: <Widget>[
          FloatingActionButton(
            heroTag: "btn1",
            onPressed: () {
              Navigator.push(
                context,
                // Create the SelectionScreen in the next step.
                MaterialPageRoute(builder: (context) => BarsFilters()),
              );
            },
            child: Icon(
              Icons.filter_list,
              size: 30,
            ),
            backgroundColor: Color(0xff94C2D4),
            foregroundColor: Colors.white,
            elevation: 10,
            shape: RoundedRectangleBorder(
                borderRadius: new BorderRadius.circular(30.0),
                side: BorderSide(color: Colors.white)),
          ),
          SizedBox(
            width: 20,
          ),
          FloatingActionButton.extended(
            heroTag: "btn2",
            onPressed: () {
              _navigateAndDisplaySelection(context);
            },
            label: Text(
                "$toppickstext"
            ),
            backgroundColor: Color(0xff94C2D4),
            foregroundColor: Colors.white,
            elevation: 10,
            shape: RoundedRectangleBorder(
                borderRadius: new BorderRadius.circular(0.0),
                side: BorderSide(color: Colors.white)),
          ),
        ],
      ),
      floatingActionButtonLocation: FloatingActionButtonLocation.centerFloat,
    );
  }

  _navigateAndDisplaySelection(BuildContext context) async {
    // Navigator.push returns a Future that completes after calling
    // Navigator.pop on the Selection Screen.
    final result = await Navigator.push(
      context,
      // Create the SelectionScreen in the next step.
      MaterialPageRoute(builder: (context) => TopPicksRestoItems()),
    );
    if(result == null){
      setState(() {
        toppickstext = toppickstext;
        restodata = httpService.getRestaurantsData(page, 0, SortType, loadmore);
      });
    }else{
      setState(() {
        toppickstext = result["tagname"];
        restodata = httpService.getRestaurantsData(page, result["tagid"], SortType, loadmore);
      });
    }
  }
}

Вызов API похож на:

import 'dart:convert';
import 'package:http/http.dart';
import 'package:timeout_dubai/podo/BarsData.dart';
import 'package:timeout_dubai/podo/CategoriesList.dart';
import 'package:timeout_dubai/podo/RestaurantsData.dart';
/*
* @author Sagar Chavan
* @since 14-feb-2020
* @version 1.0.0
* @class GetRestaurantsData calls api to get restaurants section data
* */

class GetRestaurantsData {

  List<RestaurantsData> rsetata = new List();
  String sections;

  Future<List<RestaurantsData>> getRestaurantsData(int pageno, int tagid, String sortType, bool loadmore) async {

    if(tagid == 0){
      sections = '44';
    }else{
      sections = '0';

    }

    final uri = '';
    final headers = {'Content-Type': 'application/json'};
    Map<String, dynamic> body = {
      'site_key': 'timeoutdubai_en',
      'page': pageno,
      'sort': sortType,
      'sort_option': 'desc',
      'sections': '44',
      'tags': tagid,

    };
    String jsonBody = json.encode(body);
    final encoding = Encoding.getByName('utf-8');

    Response response = await post(
      uri,
      headers: headers,
      body: jsonBody,
      encoding: encoding,
    );

    Map<String, dynamic> map = jsonDecode(response.body);

    print('RESPONSEMAP: $map');

    List<dynamic> resbody = map["items"];
    print('RESPONSEBODY: $resbody');

    List<RestaurantsData> restaurantsobj = resbody
        .map(
          (dynamic item) => RestaurantsData.fromJson(item),
    )
        .toList();



    if(loadmore == true){
      rsetata.addAll(restaurantsobj);
    }else{
      rsetata = restaurantsobj;
    }

    print("RestorantdataCount ${rsetata.length}");


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