main.dart 17 KB

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