Flutter实现可循环轮播图

效果图

壹、控件分解图

控件拆解图

贰、代码实现

贰点壹、构建根布局

新建AdPictureWidget继承自StatefulWidget,新建_AdPictureWidgetState类继承自State<AdPictureWidget>,根布局为Stack,代码如下:

class AdPictureWidget extends StatefulWidget {
  @override
  _AdPictureWidgetState createState() => _AdPictureWidgetState();
}

class _AdPictureWidgetState extends State<AdPictureWidget>{
  @override
  Widget build(BuildContext context) {
    return Stack(
      children: <Widget>[
       ...
      ],
    );
  }
}
贰点贰、构建PageView

PageView类似于Android里的ViewPager,我们可以使用PageController控制PageView 的滑动行为,比如设置滑动动画、令其滑动到指定的页面等等。可以通过设置onPageChanged来监听页面的滑动,相当于Android里的PageListener。每一个Page里的布局可以通过children属性进行设置,例子中每一个Page里包含一张图片,图片是通过网络来加载的。代码如下:

class _AdPictureWidgetState extends State<AdPictureWidget>{
  PageController _pageController = PageController();
  List _adPictures = [];
  
  @override
  void dispose() {
    _pageController.dispose();
    super.dispose();
  }
  
  @override
  Widget build(BuildContext context) {
    return Stack(
      children: <Widget>[
       PageView(
          children: _adPictures.map((json) {
            var adPicture = AdPicture.fromJson(json);//可以先忽略这个实体类
            return Image.network(
              adPicture.imageUrl,
              fit: BoxFit.fill,//使照片占满整个屏幕
            );
          }).toList(),
          onPageChanged: _onPageChanged,
          controller: _pageController,
        ),
      ],
    );
  }
  
  void _onPageChanged(int index) {
   ...
  }
}
贰点叁、构建下方的Indicator布局

屏幕下方的一行指示小圆点可以直接使用flutter的TabPageSelector搞定,使用Align控制其显示在屏幕的下方。我们只需要使用TabPageSelector的三个属性即可,通过color属性设置其未被选中时的颜色,通过selectedColor设置选中时的颜色,那如何控制选中还是未被选中呢,答案是它的controller属性,我们直接new出一个TabController类,将其赋值给controller属性即可,代码如下:

class _AdPictureWidgetState extends State<AdPictureWidget>
    with SingleTickerProviderStateMixin {
  TabController _tabController;
  ...
  
  @override
  void initState() {
    _tabController = TabController(length: 0, vsync: this);
    super.initState();
  }

  @override
  void dispose() {
    ...
    _tabController.dispose();
    super.dispose();
  }
  
  @override
  Widget build(BuildContext context) {
    return Stack(
      children: <Widget>[
        PageView(
          ...
        ),
        Align(
          alignment: Alignment(0.0, 0.5),
          child: TabPageSelector(
            color: Colors.white,
            selectedColor: Colors.black,
            controller: _tabController,
          ),
        ),
      ],
    );
  }
贰点肆、PageViewTabPageSelector联动 & 定时自动翻页

二者的联动很简单,在PageView的滑动回调里调用_tabControlleranimateTo方法即可实现二者的联动。如果需要定时翻页,则需要使用到一个Timer的类,详细代码如下:

const timeout = const Duration(seconds: 2);
class _AdPictureWidgetState extends State<AdPictureWidget>
    with SingleTickerProviderStateMixin {
  ...
  Timer _timer;
  int _index = 0;

  @override
  void initState() {
    ...
    _timer = Timer.periodic(timeout, _handleTimeout);//一创建定时器就启动了,每过timeout时间就会调用_handleTimeout这个回调。
    super.initState();
  }

  @override
  void dispose() {
    ...
    _timer.cancel();
    super.dispose();
  }
  
  _handleTimeout(Timer timer) {
      _index++;
      _pageController.animateToPage(
        _index % (_adPictures.length),//跳转到的位置
        duration: Duration(milliseconds: 16),//跳转的间隔时间
        curve: Curves.fastOutSlowIn,//跳转动画
      );
      _tabController.animateTo(_index % (_adPictures.length));
  }
  
贰点五、循环翻页实现

在这里插入图片描述
在这里插入图片描述
假设只有三页,实现循环播放的原理是在原来的数据基础上,在最开始插入一张原本的尾页,在最末尾插入一张原本的首页(看上面两张图也许更形象),当用户滑动到现在的尾页时,程序自动的将其滑动到现在的第二页,滑动的很快对用户来说是无感之的,同理,当用户滑动到现在的首页时,程序自动滑动到现在的倒数第二页。这种方法在Android里也是挺常用的。

叁、可运行的完整代码

  • 依赖的第三方库:
  dio: 1.0.6  
  json_annotation: ^2.0.0
  • 代码及文件名:

///文件名:AdPictureWidget.dart
class AdPictureWidget extends StatefulWidget {
  @override
  _AdPictureWidgetState createState() => _AdPictureWidgetState();
}

const timeout = const Duration(seconds: 2);

class _AdPictureWidgetState extends State<AdPictureWidget>
    with SingleTickerProviderStateMixin {
  TabController _tabController;
  PageController _pageController = PageController();
  Timer _timer;

  List _adPictures = [];
  int _index = 0;

  @override
  void initState() {
    _tabController = TabController(length: 0, vsync: this);
    _timer = Timer.periodic(timeout, _handleTimeout);
    loadAdPictures();
    super.initState();
  }

  @override
  void dispose() {
    _tabController.dispose();
    _timer.cancel();
    _pageController.dispose();
    super.dispose();
  }

  @override
  Widget build(BuildContext context) {
    return Stack(
      children: <Widget>[
        PageView(
          children: _adPictures.map((json) {
            var adPicture = AdPicture.fromJson(json);
            return Image.network(adPicture.imageUrl, fit: BoxFit.fill);
          }).toList(),
          onPageChanged: _onPageChanged,
          controller: _pageController,
        ),
        Align(
          alignment: Alignment(0.0, 0.5),
          child: TabPageSelector(
            color: Colors.white,
            selectedColor: Colors.black,
            controller: _tabController,
          ),
        ),
      ],
    );
  }

  _handleTimeout(Timer timer) {
    if (_adPictures.length - 2 != 0) {
      _index++;
      _pageController.animateToPage(
        _index % (_adPictures.length - 2),
        duration: Duration(milliseconds: 16),
        curve: Curves.fastOutSlowIn,
      );
    }
  }

  void _onPageChanged(int index) {
    _index = index;
    if (index == 0) {
      _tabController.animateTo(_tabController.length - 1);
      _pageController.jumpToPage(_adPictures.length - 2);
    } else if (index == _adPictures.length - 1) {
      _tabController.animateTo(0);
      _pageController.jumpToPage(1);
    } else {
      _tabController.animateTo(index - 1);
    }
  }

  void loadAdPictures() async {
    Dio dio = Dio();
    Response<List> response = await dio
        .get("http://www.wanandroid.com/tools/mockapi/2511/getAdPictures");

    List res = response.data;
    if (res.length != 0) {
      res.insert(0, res[res.length - 1]);
      res.add(res[1]);

      setState(() {
        _adPictures = res;
        _pageController.jumpToPage(1);
        _tabController =
            TabController(length: _adPictures.length - 2, vsync: this);
      });
    }
  }
}

///文件名:AdPicture.dart

library adpicture;

import 'package:json_annotation/json_annotation.dart';

part 'AdPicture.g.dart';

///首页轮播图
@JsonSerializable()
class AdPicture {
  final String imageUrl; //图片链接

  AdPicture({
    this.imageUrl,
  });

  factory AdPicture.fromJson(Map<String, dynamic> json) =>
      _$AdPictureFromJson(json);
}


///文件名:AdPicture.g.dart
part of adpicture;

AdPicture _$AdPictureFromJson(Map<String, dynamic> json) {
  return AdPicture(imageUrl: json['imageUrl'] as String);
}

Map<String, dynamic> _$AdPictureToJson(AdPicture instance) => <String, dynamic>{
      'imageUrl': instance.imageUrl,
    };

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个基本的Flutter轮播图示例: ```dart import 'package:flutter/material.dart'; import 'package:carousel_slider/carousel_slider.dart'; class MyCarousel extends StatefulWidget { @override _MyCarouselState createState() => _MyCarouselState(); } class _MyCarouselState extends State<MyCarousel> { int _current = 0; List<String> _images = [ 'https://picsum.photos/id/1015/600/400', 'https://picsum.photos/id/1016/600/400', 'https://picsum.photos/id/1018/600/400', 'https://picsum.photos/id/1019/600/400' ]; @override Widget build(BuildContext context) { return Column( children: [ CarouselSlider( items: _images.map((image) { return Container( margin: EdgeInsets.all(5), child: ClipRRect( borderRadius: BorderRadius.all(Radius.circular(5)), child: Image.network(image, fit: BoxFit.cover, width: 1000), ), ); }).toList(), options: CarouselOptions( autoPlay: true, enlargeCenterPage: true, aspectRatio: 2.0, onPageChanged: (index, reason) { setState(() { _current = index; }); }), ), Row( mainAxisAlignment: MainAxisAlignment.center, children: _images.map((image) { int index = _images.indexOf(image); return Container( width: 8, height: 8, margin: EdgeInsets.symmetric(vertical: 10, horizontal: 2), decoration: BoxDecoration( shape: BoxShape.circle, color: _current == index ? Colors.blueAccent : Colors.grey[400], ), ); }).toList(), ), ], ); } } ``` 这个示例使用了一个第三方库`carousel_slider`来实现轮播图。首先在`pubspec.yaml`文件中添加依赖: ```yaml dependencies: flutter: sdk: flutter carousel_slider: ^4.0.0 ``` 然后在代码中导入: ```dart import 'package:carousel_slider/carousel_slider.dart'; ``` 在`_MyCarouselState`类中定义了一个`_current`变量来跟踪当前轮播图的索引。然后定义了一个包含图像URL的列表`_images`。在`build`方法中,使用`CarouselSlider`小部件来创建轮播图。`items`属性接受一个包含所有要显示的小部件的列表。在这个示例中,我们将图像映射到一个包含图像的容器中,并将它们放在一个`ClipRRect`小部件中,以便将它们的边角切成圆角。 `options`属性接受一个`CarouselOptions`对象,其中包括自动播放、放大中心页面、宽高比和页面更改回调等选项。 在`onPageChanged`回调中,我们使用`setState`方法来更新`_current`变量。最后,我们创建了一个包含小圆点指示器的行,以便用户可以看到当前显示的页面。这个小圆点指示器使用`_images`列表中的图像来创建,并基于当前索引来高亮显示当前页面。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值