main.dart 18 KB

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