直播带货源码,flutter 顶部滚动栏+页面
直播带货源码,flutter 顶部滚动栏+页面
tabPage.dart
import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart';
import 'package:flutter_trip/dao/travelDao.dart';
import 'package:flutter_trip/model/home/commonModel.dart';
import 'package:flutter_trip/model/travel/travelModel.dart';
import 'package:flutter_staggered_grid_view/flutter_staggered_grid_view.dart';
import 'package:flutter_trip/widget/applicationWebView.dart';
import 'package:flutter_trip/widget/loadingContainer.dart';
const _TRAVEL_URL =
'https://m.ctrip.com/restapi/soa2/16189/json/searchTripShootListForHomePageV2?_fxpcqlniredt=09031014111431397988&__gw_appid=99999999&__gw_ver=1.0&__gw_from=10650013707&__gw_platform=H5';
const PAGE_SIZE = 10;
class TabPage extends StatefulWidget {
final String travelUrl;
final String channelCode;
TabPage({Key? key, required this.travelUrl, required this.channelCode})
: super(key: key);
@override
State<StatefulWidget> createState() {
// TODO: implement createState
return _TabPageState();
}
}
class _TabPageState extends State<TabPage> with AutomaticKeepAliveClientMixin {
List<TravelItem> travelItem = [];
int pageIndex = 0;
bool isLoad = true;
ScrollController _scrollController = ScrollController();
@override
// TODO: implement wantKeepAlive
bool get wantKeepAlive => true;
@override
void initState() {
// TODO: implement initState
super.initState();
_loadData();
_scrollController.addListener(() {
if (_scrollController.position.pixels ==
_scrollController.position.maxScrollExtent) {
_loadData(loadMore: true);
print("滚动到最底部有");
}
});
}
@override
Widget build(BuildContext context) {
// TODO: implement build
return Scaffold(
body: LoadingContainer(
isLoading: isLoad,
cover: true,
child: RefreshIndicator(
onRefresh: _handleRefresh,
child: MediaQuery.removePadding(
removeTop: true,
context: context,
child: new StaggeredGridView.countBuilder(
controller: _scrollController,
crossAxisCount: 2,
itemCount: travelItem.length,
itemBuilder: (BuildContext context, int index) =>
_TraveItem(index: index, item: travelItem[index]),
staggeredTileBuilder: (int index) => new StaggeredTile.fit(1),
// mainAxisSpacing: 4.0,
// crossAxisSpacing: 4.0,
),
),
),
),
);
}
Future<void> _handleRefresh() async {
_loadData();
}
_loadData({bool loadMore = false}) {
if (loadMore) {
pageIndex++;
} else {
pageIndex = 0;
}
TravelDao.fetch(widget.travelUrl, widget.channelCode, pageIndex, PAGE_SIZE)
.then((value) {
setState(() {
List<TravelItem> items = _filterItems(value.resultList);
if (travelItem.length != 0) {
travelItem.addAll(items);
} else {
travelItem = items;
}
isLoad = false;
});
}).catchError((onError) {
print('网络请求出错 $onError');
});
}
List<TravelItem> _filterItems(List<TravelItem>? resultList) {
List<TravelItem> items = [];
if (resultList != null) {
resultList.forEach((element) {
if (element.article != null) {
items.add(element);
}
});
}
return items;
}
}
class _TraveItem extends StatelessWidget {
final int index;
final TravelItem item;
_TraveItem({Key? key, required this.index, required this.item})
: super(key: key);
@override
Widget build(BuildContext context) {
// TODO: implement build
return GestureDetector(
onTap: () {
if (item.article.urls.length > 0) {
// Navigator.push(
// context,
// MaterialPageRoute(
// builder: (context) => ApplicationWebView(
// commonModel: CommonModel(
// icon: '',
// title: '详情',
// url: item.article.urls[0].h5Url,
// statusBarColor: 'ffffff',
// hideAppBar: false),
// )));
}
},
child: Card(
child: PhysicalModel(
color: Colors.transparent,
clipBehavior: Clip.antiAlias,
borderRadius: BorderRadius.circular(5),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
_itemImage(),
Container(
padding: EdgeInsets.all(4),
child: Text(
item.article.articleTitle,
textAlign: TextAlign.left,
maxLines: 2,
overflow: TextOverflow.ellipsis,
style: TextStyle(fontSize: 14, color: Colors.black),
),
),
_infoText(),
],
),
),
),
);
}
_itemImage() {
return Stack(
children: [
Image.network(item.article.images[0].dynamicUrl),
Positioned(
bottom: 8,
left: 8,
child: Container(
padding: EdgeInsets.fromLTRB(5, 1, 5, 1),
decoration: BoxDecoration(
color: Colors.black54,
borderRadius: BorderRadius.circular(10),
),
child: Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(
Icons.location_on_outlined,
size: 12,
color: Colors.white,
),
LimitedBox(
maxWidth: 130,
child: Text(
_positonName(),
maxLines: 1,
overflow: TextOverflow.ellipsis,
style: TextStyle(fontSize: 12, color: Colors.white),
),
)
],
),
),
)
],
);
}
String _positonName() {
// return item.article.pois?.length == 0
// ? '未知'
// : item.article.pois![0]!.poiName;
return item.article.pois?.length == 0
? '未知'
: item.article.pois![0]?.poiName ?? '未知';
}
_infoText() {
return Container(
padding: EdgeInsets.fromLTRB(6, 0, 6, 10),
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Row(
children: [
PhysicalModel(
color: Colors.transparent,
clipBehavior: Clip.antiAlias,
borderRadius: BorderRadius.circular(12),
child: Image.network(
item.article.author!.coverImage!.dynamicUrl,
width: 24,
height: 24,
),
),
Container(
width: 90,
padding: EdgeInsets.all(5),
child: Text(
item.article.author!.nickName,
textAlign: TextAlign.left,
style: TextStyle(fontSize: 12),
maxLines: 1,
overflow: TextOverflow.ellipsis,
),
)
],
),
Row(
mainAxisAlignment: MainAxisAlignment.end,
children: [
Icon(
Icons.thumb_up,
color: Colors.grey,
size: 12,
),
Container(
padding: EdgeInsets.only(left: 3),
child: Text(
item.article.likeCount.toString(),
maxLines: 1,
style: TextStyle(
color: Colors.grey,
fontSize: 10,
),
),
)
],
)
],
),
);
}
}
tabTravel.dart
import 'package:flutter/material.dart';
import 'package:flutter_trip/dao/travelDao.dart';
import 'package:flutter_trip/dao/travelTabDao.dart';
import 'package:flutter_trip/model/travel/travelModel.dart';
import 'package:flutter_trip/model/travel/travelTabModel.dart';
import 'package:flutter_trip/pages/travel/tabPage.dart';
class TabTravel extends StatefulWidget {
@override
_TabTravelState createState() => _TabTravelState();
}
class _TabTravelState extends State<TabTravel>
with SingleTickerProviderStateMixin {
late TabController tabController;
late TravelTabModel travelTabModel;
bool isInit = false;
List<TravelTab> tabs = [];
@override
void initState() {
//tabController = TabController(length: 0, vsync: this);
// TODO: implement initState
super.initState();
TravelTabDao.fetch().then((TravelTabModel tabModel) {
tabController = TabController(length: tabModel.tabs.length, vsync: this);
setState(() {
tabs = tabModel.tabs;
travelTabModel = tabModel;
});
this.isInit = true;
}).catchError((error) {
print("error $error");
});
}
@override
Widget build(BuildContext context) {
// TODO: implement build
return Scaffold(
body: Column(
children: [
isInit
? Container(
color: Colors.white,
padding:
EdgeInsets.only(top: MediaQuery.of(context).padding.top),
child: TabBar(
padding: EdgeInsets.zero,
controller: tabController,
isScrollable: true,
labelColor: Colors.black87,
labelPadding: EdgeInsets.only(left: 20, right: 20),
indicator: UnderlineTabIndicator(
borderSide: BorderSide(
color: Colors.lightBlue,
width: 3,
),
insets: EdgeInsets.only(bottom: 10),
),
tabs:
tabs.map((item) => Tab(text: item.labelName)).toList(),
),
)
: Container(),
Flexible(
child: isInit
? TabBarView(
controller: tabController,
children: tabs
.map((item) => TabPage(
travelUrl: travelTabModel.url,
channelCode: item.groupChannelCode))
.toList(),
// children: tabs
// .map((item) => Tab(text: item.groupChannelCode))
// .toList(),
)
: Container(),
)
],
),
);
}
@override
void dispose() {
// TODO: implement dispose
tabController.dispose();
super.dispose();
}
以上就是 直播带货源码,flutter 顶部滚动栏+页面,更多内容欢迎关注之后的文章
直播带货源码,flutter 顶部滚动栏+页面的更多相关文章
- C# 带滚动栏的Label控件
C# 带滚动栏的Label控件,用鼠标选的时候还是有点闪烁: namespace 带滚动栏的Label控件 { public class TextBoxLabel : System.Windows.F ...
- 直播带货APP源码开发为什么选择云服务器
云服务器可以为直播带货APP源码提供弹性计算以及更高的运行效率,避免资源浪费,随着直播带货APP源码业务需求的变化,可以实时扩展或缩减计算资源.CVM支持按实际使用的资源计费,可以节约计算成本. 一. ...
- Bootstrap学习笔记上(带源码)
阅读目录 排版 表单 网格系统 菜单.按钮 做好笔记方便日后查阅o(╯□╰)o bootstrap简介: ☑ 简单灵活可用于架构流行的用户界面和交互接口的html.css.javascript工具集 ...
- 自己写一个jQuery垂直滚动栏插件(panel)
html中原生的滚动栏比較难看,所以有些站点,会自己实现滚动栏,导航站点hao123在一个側栏中,就自己定义了垂直滚动栏,效果比較好看,截图例如以下: watermark/2/text/aHR0cDo ...
- [置顶]
xamarin Tablayout+Viewpager+Fragment顶部导航栏
最近几天不忙,所以把项目中的顶部导航栏的实现归集一下.android中使用TabLayout+ViewPager+Fragment制作顶部导航非常常见,代码实现也比较简单.当然我这个导航栏是基于xam ...
- 使用PagerSlidingTabStrip实现顶部导航栏
使用PagerSlidingTabStrip配合ViewPager实现顶部导航栏. 效果图如下: PagerSlidingTabStrip是github上的一个开源项目,项目地址如下 ...
- Android之仿今日头条顶部导航栏效果
随着时间的推移现在的软件要求显示的内容越来越多,所以要在小的屏幕上能够更好的显示更多的内容,首先我们会想到底部菜单栏,但是有时候像今日头条新闻客户端要显示的内容太多,而且又想在主界面全部显示出来,所以 ...
- JS 带运动的返回顶部 小案例
带运动的返回顶部:当滚动条在滚动的时候,滚动鼠标的滚轮,应该让滚动条停止滚动,清掉定时器.下面的方法b 就是清掉的方法 <!DOCTYPE html PUBLIC "-//W3C//D ...
- iOS开发项目实战——Swift实现ScrollView滚动栏功能
手机作为一个小屏设备,须要显示的信息往往无法在一个屏幕上显示,此时就须要使用到滚动栏,当然除了像TableView这样能够自带滚动功能的. 假设一个界面上View较多,那就必须要使用到ScrollVi ...
- android 滚动栏下拉反弹的效果(相似微信朋友圈)
微信朋友圈上面的图片封面,QQ空间说说上面的图片封面都有下拉反弹的效果,这些都是使用滚动栏实现的.下拉,当松开时候.反弹至原来的位置.下拉时候能看到背景图片.那么这里简介一下这样的效果的实现. 本文源 ...
随机推荐
- golang使用JWX进行认证和加密
golang使用JWX进行认证和加密 最近看了一个名为go-auth的库,它将JWT作为HTTP cookie对用户进行验证,但这个例子中缺少了对JWT的保护,由此进行了一些针对JWX的研究. 下面描 ...
- Java基础语法:注释、数据类型、字节
Java基础语法:注释.数据类型.字节 注释 单行注释:// 多行注释:/* 注释 */ 文档注释:/** 注释 */ 数据类型分为两大类:基本类型和引用类型 八大基本数据类型 整数类型 byte(占 ...
- JZOJ 5372. 【NOIP2017提高A组模拟9.17】猫
题目大意 对于 \(m = [1,\lfloor \frac n 2 \rfloor]\) 要求在一个序列中恰好选出 \(m\) 个不相邻的数使得权值和最大 其中 \(1\) 的左边是 \(n\),\ ...
- CF1372D Omkar and Circle
题目传送门 思路 这是一道非常简单的 \(\mathcal *2100\). 既然他样例给的那么简单,说明这是一道结论题. 于是我们可以手玩几组数据试试. 例如 \(3,5,9,8,12\) 这组,发 ...
- 有理数四则运算 PTA1034
题目:https://pintia.cn/problem-sets/994805260223102976/problems/994805287624491008 本题要求编写程序,计算 2 个有理数的 ...
- 【多线程与高并发】- 浅谈volatile
浅谈volatile 目录 浅谈volatile 简介 JMM概述 volatile的特性 1.可见性 举个例子 总结 2.无法保证原子性 举个例子 分析 使用volatile对原子性测试 使用锁的机 ...
- python之定时任务APScheduler
一.APScheduler APScheduler全称Advanced Python Scheduler 作用为在指定的时间规则执行指定的作业. 指定时间规则的方式可以是间隔多久执行,可以是指定日期时 ...
- [C#]为debug添加DebuggerDisplay属性
最近才发现,DebuggerDisplay 非常弓虽大,给类添加DebuggerDisplay属性后可以让调试变得更简单.如何使用? 1.定义一个有DebuggerDisplay的类:在类的头部添加& ...
- fabric学习笔记11
fabric学习笔记10 20201303张奕博 2023.1.23 测试实践2 导入链码依赖包 package main import ( "github.com/hyperledger/ ...
- 关于xtr的一些基础
Q&A 如何找到QSEQ方式的xtt呢? 我们可以去版本的代码中寻找,具体位置在:(modem_proc/rf/rftarget_denali/mtp/xtt/etc/QSEQ/) 在QSEQ ...