main.dart 12 KB

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