main.dart 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557
  1. import 'package:douyin_demo/pages/RecommendPage/BottomSheet.dart';
  2. import 'package:douyin_demo/providers/RecommendProvider.dart';
  3. import 'package:douyin_demo/widgets/BottomBar.dart';
  4. // import 'package:douyin_demo/widgets/FavAnimation.dart' as prefix0;
  5. import 'package:flutter/material.dart';
  6. // import 'package:marquee/marquee.dart';
  7. import 'package:marquee_flutter/marquee_flutter.dart';
  8. import 'package:provider/provider.dart';
  9. import 'package:video_player/video_player.dart';
  10. import 'widgets/FavAnimation.dart';
  11. void main() {
  12. runApp(MyApp());
  13. }
  14. class MyApp extends StatelessWidget {
  15. const MyApp({Key key}) : super(key: key);
  16. @override
  17. Widget build(BuildContext context) {
  18. return MaterialApp(
  19. title: "某音",
  20. theme: ThemeData(primaryColor: Color(0xff121319)),
  21. home: RecommendPage(selIndex: 0,),
  22. );
  23. }
  24. }
  25. class RecommendPage extends StatelessWidget {
  26. const RecommendPage({Key key,@required this.selIndex}) : super(key: key);
  27. final int selIndex;
  28. @override
  29. Widget build(BuildContext context) {
  30. return MultiProvider(
  31. providers: [
  32. ChangeNotifierProvider(
  33. builder: (context) => RecommendProvider(),
  34. )
  35. ],
  36. child: Scaffold(
  37. resizeToAvoidBottomInset: false,
  38. body: Container(
  39. decoration: BoxDecoration(color: Colors.black),
  40. child: Stack(children: [
  41. CenterImage(),
  42. Home(),
  43. ]),
  44. ),
  45. bottomNavigationBar: BottomSafeBar(selIndex: selIndex,),
  46. ));
  47. }
  48. }
  49. class BottomSafeBar extends StatelessWidget {
  50. const BottomSafeBar({Key key,@required this.selIndex}) : super(key: key);
  51. final int selIndex;
  52. @override
  53. Widget build(BuildContext context) {
  54. RecommendProvider provider = Provider.of<RecommendProvider>(context);
  55. // double toBottom=MediaQuery.of(context).viewInsets.bottom;
  56. return Container(
  57. // padding: EdgeInsets.only(top:toBottom),
  58. decoration: BoxDecoration(color: Colors.black),
  59. child: SafeArea(
  60. child: BottomAppBar(
  61. child: Container(
  62. decoration: BoxDecoration(color: Colors.black),
  63. height: 60,
  64. // decoration: BoxDecoration(color: Colors.black),
  65. child: BtmBar(selectIndex: selIndex,),
  66. ),
  67. )),
  68. );
  69. }
  70. }
  71. class CenterImage extends StatelessWidget {
  72. const CenterImage({Key key}) : super(key: key);
  73. @override
  74. Widget build(BuildContext context) {
  75. double bottom = MediaQuery.of(context).viewInsets.bottom;
  76. RecommendProvider provider = Provider.of<RecommendProvider>(context);
  77. return Center(
  78. child: Container(
  79. // padding: EdgeInsets.only(top: bottom),
  80. child: Image.asset(provider.mainInfo.videoPath)),
  81. );
  82. }
  83. }
  84. class VideoBack extends StatefulWidget {
  85. VideoBack({Key key}) : super(key: key);
  86. _VideoBackState createState() => _VideoBackState();
  87. }
  88. class _VideoBackState extends State<VideoBack> {
  89. VideoPlayerController _controller;
  90. bool _isPlaying = false;
  91. String url =
  92. "https://www.guojio.com/video/07a7faa1-3696-4af7-aeac-2d6cf6bf25f9.mp4";
  93. @override
  94. void initState() {
  95. // TODO: implement initState
  96. super.initState();
  97. _controller = VideoPlayerController.network(this.url)
  98. // 播放状态
  99. ..addListener(() {
  100. final bool isPlaying = _controller.value.isPlaying;
  101. if (isPlaying != _isPlaying) {
  102. setState(() {
  103. _isPlaying = isPlaying;
  104. });
  105. }
  106. })
  107. // 在初始化完成后必须更新界面
  108. ..initialize().then((_) {
  109. setState(() {});
  110. });
  111. }
  112. @override
  113. Widget build(BuildContext context) {
  114. return Container(
  115. child: _controller.value.initialized
  116. // 加载成功
  117. ? new AspectRatio(
  118. aspectRatio: _controller.value.aspectRatio,
  119. child: VideoPlayer(_controller),
  120. )
  121. : new Container(),
  122. );
  123. }
  124. }
  125. class Home extends StatelessWidget {
  126. const Home({Key key}) : super(key: key);
  127. @override
  128. Widget build(BuildContext context) {
  129. double screenWidth = MediaQuery.of(context).size.width;
  130. double screenHeight = MediaQuery.of(context).size.height;
  131. RecommendProvider provider = Provider.of<RecommendProvider>(context);
  132. double rpx = screenWidth / 750;
  133. return Stack(children: [
  134. Positioned(
  135. top: 20*rpx,
  136. // height: 120,
  137. width: screenWidth,
  138. child: SafeArea(
  139. child: Container(
  140. // decoration: BoxDecoration(color: Colors.pinkAccent),
  141. child: TopTab(),
  142. )),
  143. ),
  144. Positioned(
  145. bottom: 0,
  146. width: 0.70 * screenWidth,
  147. height: 260 * rpx,
  148. child: Container(
  149. // decoration: BoxDecoration(color: Colors.redAccent),
  150. child: BtnContent(),
  151. ),
  152. ),
  153. Positioned(
  154. right: 0,
  155. width: 0.2 * screenWidth,
  156. height: 500 * rpx,
  157. top: 0.45 * screenHeight,
  158. child: Container(
  159. // decoration: BoxDecoration(color: Colors.orangeAccent),
  160. child: ButtonList(),
  161. ),
  162. ),
  163. Positioned(
  164. bottom: 20 * rpx,
  165. right: 0,
  166. width: 0.2 * screenWidth,
  167. height: 0.2 * screenWidth,
  168. child: Container(
  169. // decoration: BoxDecoration(color: Colors.purpleAccent),
  170. child: RotateAlbum(),
  171. ),
  172. )
  173. ]);
  174. }
  175. }
  176. class TopTab extends StatefulWidget {
  177. TopTab({Key key}) : super(key: key);
  178. _TopTabState createState() => _TopTabState();
  179. }
  180. class _TopTabState extends State<TopTab> with SingleTickerProviderStateMixin {
  181. TabController _controller;
  182. @override
  183. void initState() {
  184. // TODO: implement initState
  185. super.initState();
  186. _controller = TabController(vsync: this, length: 2, initialIndex: 1);
  187. }
  188. @override
  189. Widget build(BuildContext context) {
  190. double rpx = MediaQuery.of(context).size.width / 750;
  191. return Row(
  192. crossAxisAlignment: CrossAxisAlignment.center,
  193. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  194. children: [
  195. // BottomNavigationBar(items: [BottomNavigationBarItem(icon: null,)],)
  196. SizedBox(
  197. width: 17 * rpx,
  198. ),
  199. Icon(
  200. Icons.search,
  201. size: 50 * rpx,
  202. color: Colors.white,
  203. ),
  204. Container(
  205. child: Container(
  206. padding: EdgeInsets.symmetric(horizontal: 90 * rpx),
  207. width: 500 * rpx,
  208. child: TabBar(
  209. indicatorColor: Colors.white,
  210. labelStyle: TextStyle(color: Colors.white, fontSize: 20),
  211. indicatorPadding: EdgeInsets.symmetric(horizontal: 30),
  212. unselectedLabelStyle:
  213. TextStyle(color: Colors.grey[700], fontSize: 18),
  214. controller: _controller,
  215. tabs: <Widget>[Text("关注"), Text("推荐")],
  216. ))),
  217. Icon(
  218. Icons.live_tv,
  219. size: 30,
  220. color: Colors.white,
  221. ),
  222. SizedBox(
  223. width: 10,
  224. ),
  225. ],
  226. );
  227. }
  228. }
  229. class BtnContent extends StatelessWidget {
  230. const BtnContent({Key key}) : super(key: key);
  231. @override
  232. Widget build(BuildContext context) {
  233. RecommendProvider provider = Provider.of<RecommendProvider>(context);
  234. return Container(
  235. child: Column(
  236. mainAxisSize: MainAxisSize.min,
  237. children: <Widget>[
  238. ListTile(
  239. title: Text(
  240. "@${provider.mainInfo.userName}",
  241. style: TextStyle(color: Colors.white, fontSize: 16),
  242. ),
  243. subtitle: Text(
  244. "${provider.mainInfo.content}",
  245. style: TextStyle(color: Colors.white, fontSize: 16),
  246. maxLines: 3,
  247. overflow: TextOverflow.ellipsis,
  248. ),
  249. ),
  250. Row(
  251. children: <Widget>[
  252. SizedBox(
  253. width: 10,
  254. ),
  255. Icon(
  256. Icons.music_note,
  257. color: Colors.white,
  258. ),
  259. // Marquee(text: "",),
  260. // Container(
  261. // width: 200,
  262. // height: 20,
  263. // child: MarqueeWidget(
  264. // text: '${provider.mainInfo.desc}',
  265. // textStyle: TextStyle(
  266. // fontWeight: FontWeight.bold,
  267. // color: Colors.white,
  268. // fontSize: 16),
  269. // // scrollAxis: Axis.horizontal,
  270. // // crossAxisAlignment: CrossAxisAlignment.start,
  271. // // blankSpace: 20.0,
  272. // // velocity: 100.0,
  273. // // pauseAfterRound: Duration(seconds: 1),
  274. // // startPadding: 10.0,
  275. // // accelerationDuration: Duration(seconds: 1),
  276. // // accelerationCurve: Curves.linear,
  277. // // decelerationDuration: Duration(milliseconds: 500),
  278. // // decelerationCurve: Curves.easeOut,
  279. // ))
  280. ],
  281. )
  282. ],
  283. ),
  284. );
  285. }
  286. }
  287. class RotateAlbum extends StatefulWidget {
  288. RotateAlbum({Key key}) : super(key: key);
  289. _RotateAlbumState createState() => _RotateAlbumState();
  290. }
  291. class _RotateAlbumState extends State<RotateAlbum>
  292. with SingleTickerProviderStateMixin {
  293. AnimationController _controller;
  294. var animation;
  295. @override
  296. void initState() {
  297. super.initState();
  298. _controller =
  299. AnimationController(vsync: this, duration: Duration(seconds: 4));
  300. animation = RotationTransition(
  301. turns: Tween(begin: 0.0, end: 1.0).animate(_controller)
  302. ..addStatusListener((status) {
  303. if (status == AnimationStatus.completed) {
  304. // _controller.forward(from: 0.0);
  305. }
  306. }),
  307. child: Container(
  308. child: CircleAvatar(
  309. backgroundImage: NetworkImage(
  310. "https://gss1.bdstatic.com/9vo3dSag_xI4khGkpoWK1HF6hhy/baike/s%3D500/sign=dde475320ee9390152028d3e4bec54f9/d009b3de9c82d1586d8294a38f0a19d8bc3e42a4.jpg"),
  311. )),
  312. );
  313. _controller.forward(from: 0.0);
  314. }
  315. @override
  316. void dispose() {
  317. // TODO: implement dispose
  318. _controller.dispose();
  319. super.dispose();
  320. }
  321. @override
  322. Widget build(BuildContext context) {
  323. return Container(
  324. padding: EdgeInsets.all(18),
  325. child: animation,
  326. );
  327. }
  328. }
  329. class ButtonList extends StatefulWidget {
  330. ButtonList({Key key}) : super(key: key);
  331. _ButtonListState createState() => _ButtonListState();
  332. }
  333. class _ButtonListState extends State<ButtonList> {
  334. @override
  335. Widget build(BuildContext context) {
  336. double rpx = MediaQuery.of(context).size.width / 750;
  337. RecommendProvider provider = Provider.of<RecommendProvider>(context);
  338. List<IconAnimationStage> stages1 = List<IconAnimationStage>();
  339. stages1.add(IconAnimationStage(
  340. color: Colors.grey[100],
  341. start: 1.0,
  342. end: 0.0,
  343. duration: Duration(milliseconds: 200)));
  344. stages1.add(IconAnimationStage(
  345. color: Colors.redAccent,
  346. start: 0.0,
  347. end: 1.3,
  348. duration: Duration(milliseconds: 300)));
  349. stages1.add(IconAnimationStage(
  350. color: Colors.redAccent,
  351. start: 1.3,
  352. end: 1.0,
  353. duration: Duration(milliseconds: 100)));
  354. List<IconAnimationStage> stages2 = List<IconAnimationStage>();
  355. stages2.add(IconAnimationStage(
  356. color: Colors.grey[100],
  357. start: 1.0,
  358. end: 1.2,
  359. duration: Duration(milliseconds: 200)));
  360. stages2.add(IconAnimationStage(
  361. color: Colors.grey[100],
  362. start: 1.2,
  363. end: 1.0,
  364. duration: Duration(milliseconds: 200)));
  365. List<IconAnimationStage> stages3 = List<IconAnimationStage>();
  366. stages3.add(IconAnimationStage(
  367. color: Colors.redAccent,
  368. start: 1.0,
  369. end: 1.2,
  370. duration: Duration(milliseconds: 200)));
  371. stages3.add(IconAnimationStage(
  372. color: Colors.grey[100],
  373. start: 1.2,
  374. end: 1.0,
  375. duration: Duration(milliseconds: 200)));
  376. double iconSize = 70 * rpx;
  377. return Container(
  378. child: Column(
  379. mainAxisAlignment: MainAxisAlignment.spaceAround,
  380. crossAxisAlignment: CrossAxisAlignment.center,
  381. children: <Widget>[
  382. Container(
  383. width: 90 * rpx,
  384. height: 105 * rpx,
  385. child: Stack(
  386. children: <Widget>[
  387. Container(
  388. // decoration: BoxDecoration(c),
  389. width: 90 * rpx,
  390. height: 90 * rpx,
  391. child: CircleAvatar(
  392. backgroundImage:
  393. NetworkImage("${provider.mainInfo.avatarUrl}"),
  394. )),
  395. Positioned(
  396. bottom: 0,
  397. left: 25 * rpx,
  398. child: Container(
  399. width: 40 * rpx,
  400. height: 40 * rpx,
  401. decoration: BoxDecoration(
  402. color: Colors.redAccent,
  403. borderRadius: BorderRadius.circular(25)),
  404. child: Icon(
  405. Icons.add,
  406. size: 20,
  407. color: Colors.white,
  408. ),
  409. ),
  410. )
  411. ],
  412. )),
  413. IconText(
  414. text: "${provider.mainInfo.favCount}",
  415. icon: !provider.mainInfo.ifFaved
  416. ? AnimatedIconWidget(
  417. key: UniqueKey(),
  418. animationList: stages1,
  419. icon: Icons.favorite,
  420. size: iconSize,
  421. provider: provider,
  422. callback: () {
  423. provider.tapFav();
  424. })
  425. : AnimatedIconWidget(
  426. key: UniqueKey(),
  427. animationList: stages3,
  428. icon: Icons.favorite,
  429. size: iconSize,
  430. provider: provider,
  431. callback: () {
  432. provider.tapFav();
  433. },
  434. ),
  435. ),
  436. IconText(
  437. text: "${provider.mainInfo.replyCount}",
  438. icon: AnimatedIconWidget(
  439. animationList: stages2,
  440. icon: Icons.comment,
  441. size: iconSize,
  442. callbackDelay: Duration(milliseconds: 200),
  443. callback: () {
  444. showBottom(context, provider);
  445. },
  446. ),
  447. ),
  448. IconText(
  449. text: "${provider.mainInfo.shareCount}",
  450. icon: AnimatedIconWidget(
  451. animationList: stages2,
  452. icon: Icons.reply,
  453. size: iconSize,
  454. ),
  455. ),
  456. ],
  457. ),
  458. );
  459. }
  460. }
  461. // class ButtonList extends StatelessWidget {
  462. // const ButtonList({Key key}) : super(key: key);
  463. // @override
  464. // }
  465. // getButtonList(double rpx, RecommendProvider provider, BuildContext context) {
  466. // return
  467. // }
  468. class IconText extends StatelessWidget {
  469. const IconText({Key key, this.icon, this.text}) : super(key: key);
  470. final AnimatedIconWidget icon;
  471. final String text;
  472. @override
  473. Widget build(BuildContext context) {
  474. double rpx = MediaQuery.of(context).size.width / 750;
  475. return Container(
  476. child: Column(
  477. mainAxisSize: MainAxisSize.min,
  478. mainAxisAlignment: MainAxisAlignment.center,
  479. crossAxisAlignment: CrossAxisAlignment.center,
  480. children: <Widget>[
  481. icon,
  482. Container(
  483. // alignment: Alignment.center,
  484. child: Text(
  485. text,
  486. style: TextStyle(color: Colors.white, fontSize: 25 * rpx),
  487. )),
  488. ],
  489. ),
  490. );
  491. }
  492. }
  493. showBottom(context, provider) {
  494. // RecommendProvider provider = Provider.of<RecommendProvider>(context);
  495. double height = MediaQuery.of(context).size.height;
  496. provider.setScreenHeight(height);
  497. provider.hideBottomBar();
  498. showModalBottomSheet(
  499. shape: RoundedRectangleBorder(
  500. borderRadius: BorderRadiusDirectional.circular(10)),
  501. context: context,
  502. builder: (_) {
  503. return Container(
  504. height: 600,
  505. child: GestureDetector(
  506. onTap: () {
  507. FocusScope.of(context).requestFocus(FocusNode());
  508. },
  509. child: ReplyFullList(pCtx:context)));
  510. });
  511. }