HomePage.dart 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416
  1. import 'package:after_layout/after_layout.dart';
  2. import 'package:flutter/material.dart';
  3. class SelfHomePage extends StatelessWidget {
  4. const SelfHomePage({Key key}) : super(key: key);
  5. @override
  6. Widget build(BuildContext context) {
  7. double rpx = MediaQuery.of(context).size.width / 750;
  8. return Scaffold(
  9. body: Container(
  10. child: HomeMain(
  11. rpx: rpx,
  12. ),
  13. ),
  14. );
  15. }
  16. }
  17. class HomeMain extends StatefulWidget {
  18. HomeMain({Key key, @required this.rpx}) : super(key: key);
  19. final double rpx;
  20. _HomeMainState createState() => _HomeMainState();
  21. }
  22. class _HomeMainState extends State<HomeMain> with TickerProviderStateMixin {
  23. double extraPicHeight = 0;
  24. BoxFit fitType;
  25. double prev_dy;
  26. double rpx;
  27. AnimationController animationController;
  28. Animation<double> anim;
  29. TabController tabController;
  30. double expanedHeight=300;
  31. @override
  32. void initState() {
  33. super.initState();
  34. tabController=TabController(vsync: this,length: 3);
  35. prev_dy = 0;
  36. fitType = BoxFit.fitWidth;
  37. animationController =
  38. AnimationController(vsync: this, duration: Duration(milliseconds: 300));
  39. anim = Tween(begin: 0.0, end: 0.0).animate(animationController);
  40. }
  41. updatePicHeight(changed) {
  42. if (prev_dy == 0) {
  43. prev_dy = changed;
  44. }
  45. extraPicHeight += changed - prev_dy;
  46. if (extraPicHeight >= 200 * widget.rpx) {
  47. fitType = BoxFit.fitHeight;
  48. } else {
  49. fitType = BoxFit.fitWidth;
  50. }
  51. setState(() {
  52. prev_dy = changed;
  53. extraPicHeight = extraPicHeight;
  54. fitType = fitType;
  55. });
  56. }
  57. updateExpandedHeight(height){
  58. setState(() {
  59. expanedHeight=height;
  60. });
  61. }
  62. runAnimate() {
  63. setState(() {
  64. anim = Tween(begin: extraPicHeight, end: 0.0).animate(animationController)
  65. ..addListener(() {
  66. if (extraPicHeight >= widget.rpx * 200) {
  67. fitType = BoxFit.fitHeight;
  68. } else {
  69. fitType = BoxFit.fitWidth;
  70. }
  71. setState(() {
  72. extraPicHeight = anim.value;
  73. fitType = fitType;
  74. });
  75. });
  76. prev_dy = 0;
  77. });
  78. }
  79. @override
  80. Widget build(BuildContext context) {
  81. double rpx = MediaQuery.of(context).size.width / 750;
  82. return Listener(
  83. onPointerMove: (result) {
  84. updatePicHeight(result.position.dy);
  85. },
  86. onPointerUp: (_) {
  87. runAnimate();
  88. animationController.forward(from: 0);
  89. },
  90. child: CustomScrollView(
  91. physics: ClampingScrollPhysics(),
  92. slivers: <Widget>[
  93. SliverAppBar(
  94. pinned: true,
  95. floating: true,
  96. actions: <Widget>[
  97. IconButton(
  98. icon: Icon(Icons.search),
  99. onPressed: () {},
  100. ),
  101. IconButton(
  102. icon: Icon(Icons.more_vert),
  103. onPressed: () {},
  104. ),
  105. ],
  106. leading: IconButton(
  107. icon: Icon(Icons.arrow_back),
  108. onPressed: () {},
  109. ),
  110. bottom: PreferredSize(
  111. preferredSize: Size.fromHeight(50), child:TabBar(controller: tabController,tabs: <Widget>[
  112. Text("作品 91"),
  113. Text("动态 91"),
  114. Text("喜欢 91"),
  115. ],)),
  116. // expandedHeight: 510 * rpx + extraPicHeight,
  117. expandedHeight: expanedHeight+extraPicHeight,
  118. flexibleSpace: Container(
  119. child: TopBarWithCallback(
  120. extraPicHeight: extraPicHeight,
  121. fitType: fitType,
  122. updateHeight: updateExpandedHeight,
  123. ),
  124. ),
  125. ),
  126. SliverList(
  127. delegate: SliverChildBuilderDelegate((context, index) {
  128. return Container(
  129. height: 30,
  130. alignment: Alignment.centerLeft,
  131. color: Colors.blueAccent,
  132. child: Text("This is itm $index"),
  133. margin: EdgeInsets.symmetric(
  134. horizontal: 20 * rpx, vertical: 10 * rpx),
  135. );
  136. }, childCount: 80),
  137. )
  138. ],
  139. ));
  140. }
  141. }
  142. class TopBarWithCallback extends StatefulWidget {
  143. TopBarWithCallback({Key key,@required this.extraPicHeight, @required this.fitType, @required this.updateHeight}) : super(key: key);
  144. final double extraPicHeight;
  145. final BoxFit fitType;
  146. final Function(double) updateHeight;
  147. _TopBarWithCallbackState createState() => _TopBarWithCallbackState();
  148. }
  149. class _TopBarWithCallbackState extends State<TopBarWithCallback> with AfterLayoutMixin {
  150. @override
  151. Widget build(BuildContext context) {
  152. return Container(
  153. child: SliverTopBar(extraPicHeight: widget.extraPicHeight,fitType: widget.fitType,),
  154. );
  155. }
  156. @override
  157. void afterFirstLayout(BuildContext context) {
  158. RenderBox box=context.findRenderObject();
  159. double height=box.getMaxIntrinsicHeight(MediaQuery.of(context).size.width);
  160. widget.updateHeight(height);
  161. }
  162. }
  163. class SliverTopBar extends StatelessWidget {
  164. const SliverTopBar(
  165. {Key key, @required this.extraPicHeight, @required this.fitType})
  166. : super(key: key);
  167. final double extraPicHeight;
  168. final BoxFit fitType;
  169. @override
  170. Widget build(BuildContext context) {
  171. double rpx = MediaQuery.of(context).size.width / 750;
  172. return Stack(
  173. children: <Widget>[
  174. Column(
  175. mainAxisSize: MainAxisSize.min,
  176. children: <Widget>[
  177. Image.asset(
  178. "lib/images/temple.jpg",
  179. width: 750 * rpx,
  180. height: 300 * rpx + extraPicHeight,
  181. fit: fitType,
  182. ),
  183. Container(
  184. padding: EdgeInsets.only(top: 20 * rpx),
  185. height: 120 * rpx,
  186. child: Row(
  187. mainAxisAlignment: MainAxisAlignment.end,
  188. children: <Widget>[
  189. Container(
  190. height: 80 * rpx,
  191. width: 330 * rpx,
  192. child: RaisedButton(
  193. color: Color(0xffdc3254),
  194. child: Text(
  195. "+关注",
  196. style: TextStyle(
  197. fontSize: 30 * rpx,
  198. color: Colors.white,
  199. letterSpacing: 3 * rpx),
  200. ),
  201. onPressed: () {},
  202. )),
  203. SizedBox(
  204. width: 10 * rpx,
  205. ),
  206. Container(
  207. decoration: BoxDecoration(
  208. borderRadius: BorderRadius.circular(2),
  209. color: Color(0xff3b3c49),
  210. ),
  211. height: 80 * rpx,
  212. child: IconButton(
  213. icon: Center(
  214. child: Icon(
  215. Icons.arrow_drop_down,
  216. color: Colors.white,
  217. size: 50 * rpx,
  218. )),
  219. onPressed: () {},
  220. ),
  221. ),
  222. SizedBox(
  223. width: 30 * rpx,
  224. )
  225. ],
  226. ),
  227. ),
  228. SizedBox(
  229. height: 100 * rpx,
  230. ),
  231. Container(
  232. width: 750 * rpx,
  233. padding: EdgeInsets.symmetric(horizontal: 30 * rpx),
  234. child: Column(
  235. crossAxisAlignment: CrossAxisAlignment.start,
  236. children: <Widget>[
  237. Text(
  238. "马友发",
  239. style: TextStyle(
  240. fontSize: 55 * rpx,
  241. color: Colors.white,
  242. fontWeight: FontWeight.bold),
  243. ),
  244. Text("抖音号:1234567",
  245. style: TextStyle(
  246. fontSize: 27 * rpx,
  247. color: Colors.white,
  248. )),
  249. SizedBox(
  250. height: 15 * rpx,
  251. )
  252. ],
  253. )),
  254. Padding(
  255. padding: EdgeInsets.symmetric(horizontal: 20 * rpx),
  256. child: Divider(
  257. color: Colors.grey[700],
  258. ),
  259. ),
  260. Container(
  261. padding: EdgeInsets.symmetric(horizontal: 20 * rpx),
  262. child: Row(
  263. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  264. children: <Widget>[
  265. Row(
  266. children: <Widget>[
  267. Icon(
  268. Icons.shop,
  269. color: Color(0xffeacd3f),
  270. ),
  271. Text(
  272. "商品橱窗",
  273. style: TextStyle(color: Color(0xffeacd3f)),
  274. )
  275. ],
  276. ),
  277. Icon(Icons.keyboard_arrow_right, color: Color(0xffeacd3f))
  278. ],
  279. )),
  280. Padding(
  281. padding: EdgeInsets.symmetric(horizontal: 20 * rpx),
  282. child: Divider(
  283. color: Colors.grey[700],
  284. ),
  285. ),
  286. Container(
  287. padding: EdgeInsets.symmetric(horizontal: 20 * rpx),
  288. height: 100 * rpx,
  289. width: 750 * rpx,
  290. child: Text(
  291. "爱神的箭发;黑色大力开发哈的\n阿萨德饭还是电话费拉开始的计划发",
  292. style: TextStyle(color: Colors.white, fontSize: 30 * rpx),
  293. ),
  294. ),
  295. Container(
  296. padding: EdgeInsets.symmetric(horizontal: 20*rpx,vertical: 10*rpx),
  297. child: Row(
  298. children: <Widget>[
  299. Tag(text:"深圳" ,),
  300. Tag(text: "世界之窗",),
  301. Tag(text: "深圳大学",)
  302. ],
  303. ),
  304. ),
  305. Container(padding: EdgeInsets.symmetric(horizontal: 20*rpx,vertical: 30*rpx),child: Row(children: <Widget>[
  306. NumWithDesc(numm: "100.2w",desc: "获赞",),
  307. NumWithDesc(numm: "15",desc: "关注",),
  308. NumWithDesc(numm: "10.8w",desc: "粉丝",),
  309. ],),)
  310. ],
  311. ),
  312. Positioned(
  313. top: 250 * rpx + extraPicHeight,
  314. left: 30 * rpx,
  315. child: Container(
  316. decoration: BoxDecoration(
  317. color: Theme.of(context).primaryColor,
  318. borderRadius: BorderRadius.circular(220 * rpx)),
  319. width: 220 * rpx,
  320. height: 220 * rpx,
  321. padding: EdgeInsets.all(10 * rpx),
  322. child: CircleAvatar(
  323. backgroundImage: NetworkImage(
  324. "https://pic2.zhimg.com/v2-a88cd7618933272ca681f86398e6240d_xll.jpg"))),
  325. )
  326. ],
  327. );
  328. }
  329. }
  330. class Tag extends StatelessWidget {
  331. const Tag({Key key, @required this.text}) : super(key: key);
  332. final String text;
  333. @override
  334. Widget build(BuildContext context) {
  335. double rpx = MediaQuery.of(context).size.width / 750;
  336. return Container(
  337. child: Text(
  338. text,
  339. style: TextStyle(fontSize: 26 * rpx,color: Color(0xff64626e)),
  340. ),
  341. color: Color(0xff3b3c49),
  342. padding: EdgeInsets.all(10 * rpx),
  343. margin: EdgeInsets.only(right: 10*rpx),
  344. );
  345. }
  346. }
  347. class NumWithDesc extends StatelessWidget {
  348. const NumWithDesc({Key key,@required this.numm,@required this.desc}) : super(key: key);
  349. final String numm;
  350. final String desc;
  351. @override
  352. Widget build(BuildContext context) {
  353. double rpx=MediaQuery.of(context).size.width/750;
  354. double textSize=35*rpx;
  355. return Padding(
  356. padding: EdgeInsets.only(right: 20*rpx),
  357. child: Row(
  358. children: <Widget>[
  359. Text(numm,style: TextStyle(fontSize: textSize,color: Colors.white,fontWeight: FontWeight.bold),),
  360. SizedBox(width: 10*rpx,),
  361. Text(desc,style: TextStyle(fontSize: textSize,color: Color(0xff3b3c49)))
  362. ],
  363. )
  364. );
  365. }
  366. }
  367. // import 'package:flutter/material.dart';
  368. // class SelfHomePage extends StatelessWidget {
  369. // const SelfHomePage({Key key}) : super(key: key);
  370. // @override
  371. // Widget build(BuildContext context) {
  372. // return Scaffold(
  373. // body: CustomScrollView(
  374. // physics: ClampingScrollPhysics(),
  375. // slivers: <Widget>[
  376. // SliverAppBar(
  377. // leading: IconButton(
  378. // icon: Icon(Icons.arrow_back),
  379. // onPressed: () {},
  380. // ),
  381. // floating: true,
  382. // pinned: false,
  383. // snap: true,
  384. // expandedHeight: 250,
  385. // flexibleSpace: FlexibleSpaceBar(
  386. // title: Text("This is Sliver App Bar"),
  387. // background: Image.asset("lib/images/temple.jpg",height: 250,fit: BoxFit.fitWidth,),
  388. // ),
  389. // ),
  390. // SliverList(delegate: SliverChildBuilderDelegate((context,index){
  391. // return Container(child: Text("This is item $index",style: TextStyle(fontSize: 20),),color: Colors.redAccent,);
  392. // },))
  393. // ],
  394. // ),
  395. // );
  396. // }
  397. // }