main.dart 18 KB

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