main.dart 15 KB

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