HomePage.dart 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. import 'package:flutter/material.dart';
  2. class SelfHomePage extends StatelessWidget {
  3. const SelfHomePage({Key key}) : super(key: key);
  4. @override
  5. Widget build(BuildContext context) {
  6. double rpx=MediaQuery.of(context).size.width/750;
  7. return Scaffold(
  8. body: Container(
  9. child: HomeMain(rpx: rpx,),
  10. ),
  11. );
  12. }
  13. }
  14. class HomeMain extends StatefulWidget {
  15. HomeMain({Key key,@required this.rpx}) : super(key: key);
  16. final double rpx;
  17. _HomeMainState createState() => _HomeMainState();
  18. }
  19. class _HomeMainState extends State<HomeMain> with TickerProviderStateMixin {
  20. double extraPicHeight = 0;
  21. BoxFit fitType;
  22. double prev_dy;
  23. double rpx;
  24. AnimationController animationController;
  25. Animation<double> anim;
  26. @override
  27. void initState() {
  28. super.initState();
  29. prev_dy = 0;
  30. fitType = BoxFit.fitWidth;
  31. animationController =
  32. AnimationController(vsync: this, duration: Duration(milliseconds: 300));
  33. anim = Tween(begin: 0.0, end: 0.0).animate(animationController);
  34. }
  35. updatePicHeight(changed) {
  36. if (prev_dy == 0) {
  37. prev_dy = changed;
  38. }
  39. extraPicHeight += changed - prev_dy;
  40. if (extraPicHeight >= 200*widget.rpx) {
  41. fitType = BoxFit.fitHeight;
  42. } else {
  43. fitType = BoxFit.fitWidth;
  44. }
  45. setState(() {
  46. prev_dy = changed;
  47. extraPicHeight = extraPicHeight;
  48. fitType=fitType;
  49. });
  50. }
  51. runAnimate() {
  52. setState(() {
  53. anim = Tween(begin: extraPicHeight, end: 0.0).animate(animationController)
  54. ..addListener(() {
  55. if (extraPicHeight >= widget.rpx*200) {
  56. fitType = BoxFit.fitHeight;
  57. } else {
  58. fitType = BoxFit.fitWidth;
  59. }
  60. setState(() {
  61. extraPicHeight = anim.value;
  62. fitType=fitType;
  63. });
  64. });
  65. prev_dy = 0;
  66. });
  67. }
  68. @override
  69. Widget build(BuildContext context) {
  70. double rpx = MediaQuery.of(context).size.width / 750;
  71. return Listener(
  72. onPointerMove: (result) {
  73. updatePicHeight(result.position.dy);
  74. },
  75. onPointerUp: (_) {
  76. runAnimate();
  77. animationController.forward(from: 0);
  78. },
  79. child: CustomScrollView(
  80. physics: ClampingScrollPhysics(),
  81. slivers: <Widget>[
  82. SliverAppBar(
  83. pinned: true,
  84. floating: true,
  85. actions: <Widget>[
  86. IconButton(
  87. icon: Icon(Icons.search),
  88. onPressed: () {},
  89. ),
  90. IconButton(
  91. icon: Icon(Icons.more_vert),
  92. onPressed: () {},
  93. ),
  94. ],
  95. leading: IconButton(
  96. icon: Icon(Icons.arrow_back),
  97. onPressed: () {},
  98. ),
  99. expandedHeight: 510 * rpx + extraPicHeight,
  100. flexibleSpace: FlexibleSpaceBar(
  101. // title: Text("Flutter SliverAppBar"),
  102. background: SliverTopBar(
  103. extraPicHeight: extraPicHeight,
  104. fitType: fitType,
  105. ),
  106. ),
  107. ),
  108. SliverList(
  109. delegate: SliverChildBuilderDelegate((context, index) {
  110. return Container(
  111. height: 30,
  112. color: Colors.blueAccent,
  113. );
  114. }, childCount: 80),
  115. )
  116. ],
  117. ));
  118. }
  119. }
  120. class SliverTopBar extends StatelessWidget {
  121. const SliverTopBar(
  122. {Key key, @required this.extraPicHeight, @required this.fitType})
  123. : super(key: key);
  124. final double extraPicHeight;
  125. final BoxFit fitType;
  126. @override
  127. Widget build(BuildContext context) {
  128. double rpx = MediaQuery.of(context).size.width / 750;
  129. return Stack(
  130. children: <Widget>[
  131. Column(
  132. children: <Widget>[
  133. Image.asset(
  134. "lib/images/temple.jpg",
  135. width: 750 * rpx,
  136. height: 300 * rpx + extraPicHeight,
  137. fit: fitType,
  138. ),
  139. Container(
  140. height: 100 * rpx,
  141. ),
  142. Padding(
  143. padding: EdgeInsets.symmetric(horizontal: 20 * rpx),
  144. child: Divider(
  145. color: Colors.grey[700],
  146. ),
  147. ),
  148. Container(
  149. height: 100 * rpx,
  150. )
  151. ],
  152. ),
  153. Positioned(
  154. top: 240 * rpx + extraPicHeight,
  155. left: 30 * rpx,
  156. child: Container(
  157. width: 180 * rpx,
  158. height: 180 * rpx,
  159. child: CircleAvatar(
  160. backgroundImage: NetworkImage(
  161. "https://pic2.zhimg.com/v2-a88cd7618933272ca681f86398e6240d_xll.jpg"))),
  162. )
  163. ],
  164. );
  165. }
  166. }