main.dart 11 KB

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