Flutter-仿京东项目总结

1、tab与路由配置

在routers/router.dart中, 配置路由:

import 'dart:js';

import 'package:flutter/material.dart';
import 'package:newsmth/pages/test/tabs/tabs.dart';

// 配置路由
final routes = {
  '/': (context) => const Tabs(),
  '/search': (context) => const SearchPage(),
};

//固定写法
var onGenerateRoute = (RouteSettings settings) {
  //统一处理
  final String? name = settings.name;
  final Function pageContentBuilder = routes[name] as Function;
  if (pageContentBuilder != null) {
    if (settings.arguments != null) {
      final Route route = MaterialPageRoute(
          builder: (context) =>
              pageContentBuilder(context, arguments: settings.arguments));
      return route;
    } else {
      final Route route =
          MaterialPageRoute(builder: (context) => pageContentBuilder(context));
      return route;
    }
  }
};

在tabs/tabs.dart中, 设置tab:

import 'package:flutter/material.dart';
import 'package:newsmth/pages/favorite/index.dart';
import 'package:newsmth/pages/home/index.dart';
import 'package:newsmth/pages/message/index.dart';
import 'package:newsmth/pages/my/index.dart';

class Tabs extends StatefulWidget {
  const Tabs({Key? key}) : super(key: key);

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

class _TabsState extends State<Tabs> {
  int _currentIndex = 0;
  final List _pageList = [
    const HomeView(),
    const FavoriteView(),
    const MessageView(),
    const MyView(),
  ];
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: const Text("jdshop"),
      ),
      body: _pageList[_currentIndex], //tab对应页面
      bottomNavigationBar: BottomNavigationBar(
        currentIndex: _currentIndex,
        onTap: (index) => {
          setState(() {
            _currentIndex = index;
          })
        },
        type: BottomNavigationBarType.fixed, //此配置可以实现展示多个tab
        fixedColor: Colors.red,//tab选中颜色
        items: const [
          BottomNavigationBarItem(
            icon: Icon(Icons.home),
            label: "首页",
          ),
          BottomNavigationBarItem(
            icon: Icon(Icons.category),
            label: "分类",
          ),
          BottomNavigationBarItem(
            icon: Icon(Icons.shopping_cart_outlined),
            label: "购物车",
          ),
          BottomNavigationBarItem(
            icon: Icon(Icons.people),
            label: "我的",
          )
        ],
      ),
    );
  }
}

在main.dart中使用Tab和路由配置:

import 'routers/router.dart';
...
class _myAppState extends State<MyApp>{
  @override
  Widget build(BuildContext context){
    return MaterialApp(
      initialRoute: '/', //初始路由
        onGenerateRoute: onGenerateRoute
    )
  }
}

在homeView.dart中, 测试路由跳转:

RaisedButton(
    child: Text("跳转到搜索"),
  onPressed:(){
    Navigator.pushName(context, '/search');
  }
)

2、首页布局

使用插件flutter_swiper, 来实现轮播图

dependencies:
    flutter_swiper: ^1.16 # 轮播图
  flutter_screenutil: ^5.0.3 # 屏幕适配

在home.dart 中, 使用

import 'package:flutter/material.dart';
import 'package:flutter_screenutil/flutter_screenutil.dart';
import 'package:flutter_swiper/flutter_swiper.dart';

class CartPage extends StatefulWidget {
  const CartPage({Key? key}) : super(key: key);

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

class _CartPageState extends State<CartPage> {
  //轮播图
  Widget _swiperWidget() {
    List<Map> imgList = [
      {"url": "https://www.ityiing.com/images/flutter/slide01.jpg"},
      {"url": "https://www.ityiing.com/images/flutter/slide02.jpg"},
      {"url": "https://www.ityiing.com/images/flutter/slide03.jpg"},
    ];
    return Container(
      child: AspectRatio(
        aspectRatio: 2 / 1, //设置宽高比
        child: Swiper(
          itemBuilder: (BuildContext context, int index) {
            return Image.network(
              imgList[index]["url"],
              fit: BoxFit.fill,
            );
          },
          itemCount: imgList.length,
          pagination: const SwiperPagination(),
          control: const SwiperControl(),
          autoplay: true, //自动轮播
        ),
      ),
    );
  }

  Widget _titleWidget(value) {
    return Container(
      height: 34.h,
      margin:  EdgeInsets.only(left: 10.w),
      padding: EdgeInsets.only(left: 10.w),
      //设置左侧边框
      decoration:  BoxDecoration(
        border: Border(
          left: BorderSide(
            color: Colors.red,
            width: 10.w,
          ),
        ),
      ),
      child: Text(
        value,
        style: const TextStyle(
          color: Colors.black54,
        ),
      ),
    );
  }

  @override
  Widget build(BuildContext context) {
    return ListView(
      children: [
        _swiperWidget(),
        _titleWidget("猜你喜欢"),
        SizedBox(height: 10.sp),
        _titleWidget("热门推荐"),
        SizedBox(height: 10.sp),
      ],
    );
  }
}

3、封装适配库以实现左右滑动Listview

首页-热门商品

//热门商品
  Widget _hotProductList() {
    return Container(
      height: 240.h,
      // width: double.infinity, //要占用整个宽度或高度
      padding: EdgeInsets.all(20.w),
      child: ListView.builder(
        scrollDirection: Axis.horizontal, //水平滚动
        itemBuilder: (context, index) {
          return Column(
            children: [
              Container(
                height: 140.h,
                width: 140.w,
                margin: EdgeInsets.only(right: 21.w),
                child: Image.network(
                  "https://www.itying.com/images/flutter/hot${index + 1}.jpg",
                  fit: BoxFit.cover, //图片适配容器宽高
                ),
              ),
              Container(
                padding: EdgeInsets.only(top: 10.h),
                child: Text("第$index条"),
                height: 44.h,
              )
            ],
          );
        },
        itemCount: 80,
      ),
    );
  }

4、网格布局

首页商品列表

    //推荐商品
  Widget _recProductItemWidget() {
    var itemWidth = (1.sw - 30) / 2;
    //没设置高度, 高度会自适应
    return Container(
      padding: const EdgeInsets.all(5),
      width: itemWidth,
      decoration: BoxDecoration(
          //边框
          border: Border.all(
        color: Colors.black12,
        width: 1,
      )),
      child: Column(
        children: [
          SizedBox(
            width: double.infinity,
            child: AspectRatio(
              aspectRatio: 1/1,//防止服务器返回图片宽度不一致,导致高度不一致
              child: Image.network(
                "https://www.itying.com/images/flutter/list1.jpg",
                fit: BoxFit.cover, //图片适配容器宽高
              ),
            ),
          ),
          Padding(
            padding: EdgeInsets.only(top: 10.h),
            child: const Text(
              "2019夏季新款",
              maxLines: 2,
              overflow: TextOverflow.ellipsis, //超出限制...
              style: TextStyle(color: Colors.black54),
            ),
          ),
          Padding(
            padding: EdgeInsets.only(top: 20.h),
            //左右布局
            child: Stack(
              children: [
                Align(
                  alignment: Alignment.centerLeft,
                  child: Text(
                    "¥188.0",
                    style: TextStyle(color: Colors.red, fontSize: 16.sp),
                  ),
                ),
                Align(
                  alignment: Alignment.centerRight,
                  child: Text(
                    "¥198.0",
                    style: TextStyle(
                        color: Colors.black54,
                        fontSize: 14.sp,
                        decoration: TextDecoration.lineThrough),//配置一个下划线
                  ),
                ),
              ],
            ),
          )
        ],
      ),
    );
  }

    Widget _recProductListWidget() {
    return Container(
      padding: EdgeInsets.all(10),
      child: Wrap(
        runSpacing: 10, //主轴间距
        spacing: 10, //副轴间距
        children: [
          _recProductItemWidget(),
        ],
      ),
    );
  }

5、JSON转对象

一、在线方式

1、JSON to Dart
2、quicktype (推荐)

二、插件工具 (推荐)

1、FlutterJsonBeanFactory
2、json_serializable 和 build_runner

7、商品分类页面布局--左右菜单

import 'package:flutter/material.dart';
import 'package:flutter_screenutil/flutter_screenutil.dart';

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

  @override
  State<Category> createState() => _CategoryState();
}

class _CategoryState extends State<Category> {
  @override
  Widget build(BuildContext context) {
    var _selectedIndex = 0;

    //计算左侧宽度
    var leftWidth = 1.sw / 4;
    //右侧每一项宽度=(总宽度-左测宽度-GridView外侧原生左右的Padding值-GridView中间的间距)/3
    var rightItemWidth = (1.sw - leftWidth - 20 - 20) / 3;
    rightItemWidth = rightItemWidth.w;
    var rightItemHeight = rightItemWidth + 28.h;

    return Row(children: [
      //左侧视图
      Container(
        width: leftWidth,
        height: double.infinity,
        color: Colors.red,
        child: ListView.builder(
          itemCount: 18,
          itemBuilder: (context, index) {
            return Column(
              children: [
                // InkWell当做"按钮"组件用
                InkWell(
                  onTap: () {
                    setState(() {
                      _selectedIndex = index;
                    });
                  },
                  child: Container(
                    width: double.infinity,
                    height: 84.h,
                    padding: EdgeInsets.only(top: 24.h),
                    child: Text("第$index条", textAlign: TextAlign.center),
                    color: _selectedIndex == index ? Colors.red : Colors.white,
                  ),
                ),
                const Divider(height: 1) //如果不设置高度, 默认高度是16
              ],
            );
          },
        ),
      ),
      //右侧视图
      Expanded(
        flex: 1,
        child: Container(
          padding: EdgeInsets.all(10.h),
          height: double.infinity,
          color: const Color.fromRGBO(240, 240, 240, 0.9),
          child: GridView.builder(
            gridDelegate: SliverGridDelegateWithFixedCrossAxisCount(
              crossAxisCount: 3, //每行数量
              childAspectRatio: rightItemWidth / rightItemHeight, //宽高比
              crossAxisSpacing: 10, //纵轴间距
              mainAxisSpacing: 10, //主轴间距
            ),
            itemBuilder: (context, index) {
              return Container(
                child: Column(
                  children: [
                    AspectRatio(
                      aspectRatio: 1 / 1,
                      child: Image.network(
                        "https://www.itying.com/images/flutter/list1.jpg",
                        fit: BoxFit.cover,
                      ),
                    ),
                    Container(
                      height: 28.h,
                      child:  Text("女装"),
                    )
                  ],
                ),
              );
            },
            itemCount: 18,
          ),
        ),
      ),
    ]);
  }
}

9、底部Tab切换保持页面状态

IndexedStack 保持页面状态

IndexedStack 和 Stack 一样,都是层布局控件, 可以在一个控件上面放置另一 个控件,但唯一不同的是 IndexedStack 在同一时刻只能显示子控件中的一个控 件,通过 Index 属性来设置显示的控件。

IndexedStack 来保持页面状态的优点就是配置简单。IndexedStack 保持页面状 态的缺点就是不方便单独控制每个页面的状态。

IndexedStack 用法:

Container(
  width: double.infinity,
  height: double.infinity,
  child: new IndexedStack(
    index: 0,
    alignment: Alignment.center,
    children: <Widget>[
      Image.network(
        "https://www.itying.com/images/flutter/list1.jpg",
        fit: BoxFit.cover,
      ),
      Image.network("https://www.itying.com/images/flutter/list2.jpg",
          fit: BoxFit.cover)
    ],
  ),
);

注: IndexedStack会一次性加载所有页面, 不方便控制所有页面状态

AutomaticKeepAliveClientMixin 保持页面状态 ✅

AutomaticKeepAliveClientMixin 结合 tab 切换保持页面状态相比 IndexedStack 而言配置起来稍 微有些复杂。它结合底部 BottomNavigationBar 保持页面状态的时候需要进行如下配置。

import 'package:flutter/material.dart';
import 'package:newsmth/pages/favorite/index.dart';
import 'package:newsmth/pages/home/index.dart';
import 'package:newsmth/pages/message/index.dart';
import 'package:newsmth/pages/my/index.dart';

class Tabs extends StatefulWidget {
  const Tabs({Key? key}) : super(key: key);

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

class _TabsState extends State<Tabs> {
  //sp1.初始化PageController
  late PageController _pageController;
  int _currentIndex = 0;
  @override
  void initState() {
    super.initState();
    _pageController = PageController(initialPage: _currentIndex);
  }

  final List<Widget> _pageList = [
    const HomeView(),
    const FavoriteView(),
    const MessageView(),
    const MyView(),
  ];
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: const Text("jdshop"),
      ),
      body: PageView(//sp2.设置tab对应页面
        controller: _pageController,
        children: _pageList,
      ),
      bottomNavigationBar: BottomNavigationBar(
        currentIndex: _currentIndex,
        onTap: (index) => {
          setState(() {
            _currentIndex = index;
            _pageController.jumpToPage(index);//sp3.设置页面切换
          })
        },
        type: BottomNavigationBarType.fixed, //此配置可以实现展示多个tab
        fixedColor: Colors.red, //选中颜色
        items: const [
          BottomNavigationBarItem(
            icon: Icon(Icons.home),
            label: "首页",
          ),
          BottomNavigationBarItem(
            icon: Icon(Icons.category),
            label: "分类",
          ),
          BottomNavigationBarItem(
            icon: Icon(Icons.shopping_cart_outlined),
            label: "购物车",
          ),
          BottomNavigationBarItem(
            icon: Icon(Icons.people),
            label: "我的",
          )
        ],
      ),
    );
  }
}

需要持久化的页面加入如下代码:

class HomePage extends StatefulWidget {
  HomePage({Key? key}) : super(key: key);
  _HomePageState createState() => _HomePageState();
}

class _HomePageState extends State<HomePage>
    with AutomaticKeepAliveClientMixin { //sp5.继承AutomaticKeepAliveClientMixin 来保持状态
  @override
  bool get wantKeepAlive => true;//sp6. 返回true
}

10、商品列表布局

注意:如果Container里面加上decoration属性, 这个时候color属性必须放在BoxDecoration里面

class _ProductListPageState extends State<ProductListPage> {
  Widget _productListWidget() {
    return Container(
      padding: EdgeInsets.all(10),
      margin: EdgeInsets.only(top: 80.h),
      child: ListView.builder(
        itemBuilder: (context, index) {
          //每一个元素
          return Column(
            children: [
              //左边图片
              Container(
                width: 180.w,
                height: 180.h,
                child: Image.network("图片地址"),
              ),
              //右侧内容
              Expanded(
                  flex: 1,
                  child: Container(
                    height: 180.h, //此时要设置高度, 否则子组件内容无法撑满容器
                    margin: EdgeInsets.only(left: 10),
                    child: Column(
                      mainAxisAlignment: MainAxisAlignment.spaceBetween,
                      crossAxisAlignment: CrossAxisAlignment.start,
                      children: [],
                    ),
                  ))
            ],
          );
        },
        itemCount: 10,
      ),
    );
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
        appBar: AppBar(
          title: Text("商品列表"),
        ),
        body: _productListWidget()
    );
  }
}

11、商品列表页面二级筛选导航布局

自定义Tab导航:

  Widget _subHeaderWidget() {
    return Positioned(
      top: 0,
      height: 80.h,
      width: 750.w,
      child: Container(
        height: 80.h,
        width: 750.w,
        color: Colors.red,
        child: Row(
          children: [
            Expanded(
              flex: 1,
              child: InkWell(
                child: Padding(
                  padding: EdgeInsets.fromLTRB(0, 16.h, 0, 16.h),
                  child: Text("综合", textAlign: TextAlign.center),
                ),
                onTap: () {},
              ),
            ),
            Expanded(
              flex: 1,
              child: InkWell(
                child: Padding(
                  padding: EdgeInsets.fromLTRB(0, 16.h, 0, 16.h),
                  child: Text("价格", textAlign: TextAlign.center),
                ),
                onTap: () {},
              ),
            ),
            Expanded(
              flex: 1,
              child: InkWell(
                child: Padding(
                  padding: EdgeInsets.fromLTRB(0, 16.h, 0, 16.h),
                  child: Text("筛选", textAlign: TextAlign.center),
                ),
                onTap: () {},
              ),
            ),
          ],
        ),
      ),
    );
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text("商品列表"),
      ),
      body: Stack(
        children: [
          _productListWidget(),
          _subHeaderWidget(),
        ],
      ),
    );
  }

实现筛选功能--侧边栏弹框:

final GlobalKey<ScaffoldState> _scaffoldKey = new GlobalKey<ScaffoldState>();   //sp1
@override
  Widget build(BuildContext context) {
    return Scaffold(
      key: _scaffoldKey,//sp2
      appBar: AppBar(
        title: Text("商品列表"),
        actions: [
          Text("")//sp4.去掉导航栏默认添加的调出侧边栏按钮
        ],
      ),
      endDrawer: Drawer(//sp3.侧边栏弹框
        child: Container(
          child: Text("侧边栏弹框内容布局"),
        ),
      ),
      body: Stack(
        children: [
          _productListWidget(),
          _subHeaderWidget(),
        ],
      ),
    );
  }

通过事件打开侧边栏

InkWell(
  onTap: () {
  //注意:新版本的 Flutter 中 ScaffoldState? 为可空类型 注意判断 
  if(_scaffoldKey.currentState!=null){
    _scaffoldKey.currentState!.openEndDrawer();
  },
  child: Text("筛选", textAlign: TextAlign.center),
),

12、上拉刷新监听

class _ProductListPageState extends State<ProductListPage> {
  Widget _productListWidget() {
    return Container(
      padding: EdgeInsets.all(10),
      margin: EdgeInsets.only(top: 80.h),
      child: ListView.builder(
        controller: _scrollController, //sp1.
        ...
   }
        
        
   final GlobalKey<ScaffoldState> _scaffoldKey = GlobalKey<ScaffoldState>();

  //sp2.用于上拉分页
  ScrollController _scrollController = ScrollController(); //listview的控制器
  //分页
  int _page = 1;
  //数据
  List _productList = [];
  //排序
  String _sort = "";
  //解决重复请求的问题
  bool flag = true;

  @override
  void initState() {
    // TODO: implement initState
    super.initState();

    _scrollController.addListener(() {//sp3.监听滚动
      //_scrollController.position.pixels //获取滚动条滚动高度
      //_scrollController.position.maxScrollExtent //获取页面高度
      if (_scrollController.position.pixels >
          _scrollController.position.maxScrollExtent - 20) {
        if (flag) {
          _getProductListData();
        }
      }
    });
  }
}

注: _scrollController.jumpTo(0); //回到顶部

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 158,736评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,167评论 1 291
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,442评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,902评论 0 204
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,302评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,573评论 1 216
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,847评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,562评论 0 197
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,260评论 1 241
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,531评论 2 245
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,021评论 1 258
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,367评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,016评论 3 235
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,068评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,827评论 0 194
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,610评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,514评论 2 269

推荐阅读更多精彩内容