main.dart 15 KB

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