main.dart 20 KB

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