[译] 挑战 Flutter 之 YouTube(画中画)
![[译] 挑战 Flutter 之 YouTube(画中画)](https://static.blogweb.cn/article/1e833cba55ad4287ae32e686d38e5e77.webp) 
挑战 Flutter 尝试在 Flutter 中重新创建特定应用的 UI 或设计。
此挑战将尝试实现 YouTube 的主页和视频详情页(视频实际播放的页面),包括动画。
这个挑战将比我以前的挑战稍微复杂一些,但结果却更好。
开始
YouTube 应用包括:
a)主页包括:
- AppBar 中有三个 action
- 用户订阅视频
- 底部导航栏
b)视频详情页包括:
- 可缩小的主播放器,能让用户查看他们的订阅信息(PIP)
- 基于当前视频的用户推荐
建立项目
让我们创建一个名为 youtube_flutter 的 Flutter 项目,并删除所有默认代码,只留下一个带有默认 appBar 的空白页面。
import 'package:flutter/material.dart';
void main() => runApp(new MyApp());
class MyApp extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return new MaterialApp(
      title: 'Flutter Demo',
      theme: new ThemeData(
        primarySwatch: Colors.blue,
      ),
      home: new MyHomePage(),
    );
  }
}
class MyHomePage extends StatefulWidget {
  @override
  _MyHomePageState createState() => new _MyHomePageState();
}
class _MyHomePageState extends State<MyHomePage> {
  @override
  Widget build(BuildContext context) {
    return new Scaffold(
      appBar: new AppBar(
        title: new Text(""),
      ),
      body: new Center(
        child: new Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: <Widget>[
          ],
        ),
      ),
    );
  }
}
制作 AppBar
AppBar 左侧有 YouTube 的 logo 和名称,右侧有三个 action,即记录、搜索和打开配置文件。
重新创建 AppBar:
appBar: new AppBar(
  backgroundColor: Colors.white,
  title: Row(
    mainAxisSize: MainAxisSize.min,
    children: <Widget>[
      Icon(FontAwesomeIcons.youtube, color: Colors.red,),
      Padding(
        padding: const EdgeInsets.only(left: 8.0),
        child: Text("YouTube", style: TextStyle(color: Colors.black, letterSpacing: -1.0, fontWeight: FontWeight.w700),),
      ),
    ],
  ),
  actions: <Widget>[
    Padding(
      padding: const EdgeInsets.symmetric(horizontal: 12.0),
      child: Icon(Icons.videocam, color: Colors.black54,),
    ),
    Padding(
      padding: const EdgeInsets.symmetric(horizontal: 12.0),
      child: Icon(Icons.search, color: Colors.black54,),
    ),
    Padding(
      padding: const EdgeInsets.symmetric(horizontal: 12.0),
      child: Icon(Icons.account_circle, color: Colors.black54,),
    ),
  ],
),
这就是重新创建的 AppBar 的样子:
![[译] 挑战 Flutter 之 YouTube(画中画)](https://static.blogweb.cn/article/41846ae1372a413a9bc3700241581e67.webp) 
注意:对于 YouTube 的 logo,我使用了 Dart pub FontFlutterAwesome 图标。
接着制作底部导航栏,
创建 BottomNavigationBar
底部导航栏有5项,在 Flutter 中重新创建非常简单。我们使用 Scaffold 的 bottomNavigationBar 参数。
bottomNavigationBar: BottomNavigationBar(items: [
  BottomNavigationBarItem(icon: Icon(Icons.home, color: Colors.black54,), title: Text("Home", style: TextStyle(color: Colors.black54),),),
  BottomNavigationBarItem(icon: Icon(FontAwesomeIcons.fire, color: Colors.black54,), title: Text("Home", style: TextStyle(color: Colors.black54),),),
  BottomNavigationBarItem(icon: Icon(Icons.subscriptions, color: Colors.black54,), title: Text("Home", style: TextStyle(color: Colors.black54),),),
  BottomNavigationBarItem(icon: Icon(Icons.email, color: Colors.black54,), title: Text("Home", style: TextStyle(color: Colors.black54),),),
  BottomNavigationBarItem(icon: Icon(Icons.folder, color: Colors.black54,), title: Text("Home", style: TextStyle(color: Colors.black54),),),
], type: BottomNavigationBarType.fixed,),
注意:对于4个以上的项目我们需要指定一个固定的 BottomNavigationBarType,因为为了避免拥挤默认类型是 shifting。
结果是:
![[译] 挑战 Flutter 之 YouTube(画中画)](https://static.blogweb.cn/article/372b881fcc7d4553a8287581c1a5a916.webp) 
重新创建的 YouTube 底部导航栏
用户订阅视频
用户订阅视频是由推荐视频组成的项目列表。我们来看看列表项:
![[译] 挑战 Flutter 之 YouTube(画中画)](https://static.blogweb.cn/article/85e91c334dba405c9ca577ed5bc3bb96.webp) 
列表项由一个带有一张图片的 Column 和一个有关视频信息的 Raw 组成。该 Row 由一张图片,一个包含标题、发布者和菜单按钮的 Column 组成。
要在 Flutter 中创建列表,我们可以使用 ListView.builder()。重新创建列表项,如下:
ListView.builder(
  itemCount: 3,
  itemBuilder: (context, position) {
    return Column(
      children: <Widget>[
        Row(
          children: <Widget>[
            Expanded(child: Image.asset(videos[position].imagePath, fit: BoxFit.cover,)),
          ],
        ),
        Padding(
          padding: const EdgeInsets.all(12.0),
          child: Row(
            crossAxisAlignment: CrossAxisAlignment.start,
            children: <Widget>[
              Expanded(child: Icon(Icons.account_circle, size: 40.0,), flex: 2,),
              Expanded(
                child: Column(
                  children: <Widget>[
                    Padding(
                      padding: const EdgeInsets.only(bottom: 4.0),
                      child: Text(videos[position].title, style: TextStyle(fontSize: 18.0),),
                    ),
                    Text(videos[position].publisher, style: TextStyle(color: Colors.black54),)
                  ],
                  crossAxisAlignment: CrossAxisAlignment.start,
                ),
                flex: 9,
              ),
              Expanded(child: Icon(Icons.more_vert), flex: 1,),
            ],
          ),
        )
      ],
    );
  },
),
这里的视频只是包含由标题和发布者等视频详情的列表。
这是重新创建的主页的样子:
![[译] 挑战 Flutter 之 YouTube(画中画)](https://static.blogweb.cn/article/29b9632997df4792924c2a16b8c73677.webp) 
我们重新创建的主页
现在,我们将继续讨论稍微难一点的部分,视频详情页。
创建视频详情页
视频详情页才是在 YouTube 中真正展示视频的页面。页面的亮点是我们可以缩小视频,并在屏幕的右下角继续播放。对于本文,我们将专注于缩小动画而不是实际播放视频。
请注意,这并不是一个特别的页面,而是在现有屏幕上叠加覆盖层。因此,我们将使用 Stack 组件来覆盖屏幕。
所以在背后,将有我们的主页,而顶部将是我们的视频页面。
构建浮动视频播放器(画中画)
为了构建可以扩大至填充整个屏幕的浮动视频播放器,我们使用 LayoutBuilder 来完美地适配屏幕。
在继续之前,我们先定义一些值,即缩小和扩大时视频播放器的大小。我们不用为扩大的播放器设置宽度,而是从布局构建器中获取。
var currentAlignment = Alignment.topCenter;
var minVideoHeight = 100.0;
var minVideoWidth = 150.0;
var maxVideoHeight = 200.0;
// 这是一个任意的值,当构建布局时会改变。
var maxVideoWidth = 250.0;
var currentVideoHeight = 200.0;
var currentVideoWidth = 200.0;
bool isInSmallMode = false;
这里, “small mode” 指视频播放器缩小的时候。
构建视频详情页的 LayoutBuilder 可以写成:
LayoutBuilder(
  builder: (context, constraints) {
    maxVideoWidth = constraints.biggest.width;
    if(!isInSmallMode) {
      currentVideoWidth = maxVideoWidth;
    }
    return Column(
      crossAxisAlignment: CrossAxisAlignment.end,
      children: <Widget>[
        Expanded(
          child: Align(
            child: Padding(
              padding: EdgeInsets.all(isInSmallMode? 8.0 : 0.0),
              child: GestureDetector(
                child: Container(
                  width: currentVideoWidth,
                  height: currentVideoHeight,
                  child: Image.asset(
                    videos[videoIndexSelected].imagePath,
                    fit: BoxFit.cover,),
                  color: Colors.blue,
                ),
                onVerticalDragEnd: (details) {
                  if(details.velocity.pixelsPerSecond.dy > 0) {
                    setState(() {
                      isInSmallMode = true;
                    });
                  }else if (details.velocity.pixelsPerSecond.dy < 0){
                    setState(() {
                    });
                  }
                },
              ),
            ),
            alignment: currentAlignment,
          ),
          flex: 3,
        ),
        currentAlignment == Alignment.topCenter ?
        Expanded(
          flex: 6,
          child: Container(
            child: Column(
              children: <Widget>[
                Row(),
                Padding(
                  padding: const EdgeInsets.all(8.0),
                  child: Card(
                    child: Padding(
                      padding: const EdgeInsets.all(8.0),
                      child: Text("Video Recommendation"),
                    ),
                  ),
                ),
                Padding(
                  padding: const EdgeInsets.all(8.0),
                  child: Card(
                    child: Padding(
                      padding: const EdgeInsets.all(8.0),
                      child: Text("Video Recommendation"),
                    ),
                  ),
                ),
                Padding(
                  padding: const EdgeInsets.all(8.0),
                  child: Card(
                    child: Padding(
                      padding: const EdgeInsets.all(8.0),
                      child: Text("Video Recommendation"),
                    ),
                  ),
                )
              ],
            ),
            color: Colors.white,
          ),
        )
            :Container(),
        Row(),
      ],
    );
  },
)
注意我们是如何获得最大屏幕宽度然后使用,而不是使用我们的第一个任意值来作为最大屏幕宽度。
我们附加了一个 GestureDetector 来检测屏幕上的滑动,以便我们可以相应地缩小和扩大它。让我们创建动画。
为视频详情页添加动画
当我们制作动画时,需要处理两件事:
- 将视频从右上角移动到右下角。
- 更改视频的大小并使其变小。
对于这些东西,我们使用两个 Tweens,一个 AlignmentTween 和一个 Tween,并构造两个同时运行的独立动画。
AnimationController alignmentAnimationController;
Animation alignmentAnimation;
AnimationController videoViewController;
Animation videoViewAnimation;
var currentAlignment = Alignment.topCenter;
@override
void initState() {
  super.initState();
  alignmentAnimationController = AnimationController(vsync: this, duration: Duration(seconds: 1))
    ..addListener(() {
      setState(() {
        currentAlignment = alignmentAnimation.value;
      });
    });
  alignmentAnimation = AlignmentTween(begin: Alignment.topCenter, end: Alignment.bottomRight).animate(CurvedAnimation(parent: alignmentAnimationController, curve: Curves.fastOutSlowIn));
  videoViewController = AnimationController(vsync: this, duration: Duration(seconds: 1))
    ..addListener(() {
      setState(() {
        currentVideoWidth = (maxVideoWidth*videoViewAnimation.value) + (minVideoWidth*(1.0-videoViewAnimation.value));
        currentVideoHeight = (maxVideoHeight*videoViewAnimation.value) + (minVideoHeight*(1.0-videoViewAnimation.value));
      });
    });
  videoViewAnimation = Tween<double>(begin: 1.0, end: 0.0).animate(videoViewController);
}
当我们的视频播放器向上或向下滑动时,会触发这些动画。
onVerticalDragEnd: (details) {
  if(details.velocity.pixelsPerSecond.dy > 0) {
    setState(() {
      isInSmallMode = true;
      alignmentAnimationController.forward();
      videoViewController.forward();
    });
  }else if (details.velocity.pixelsPerSecond.dy < 0){
    setState(() {
      alignmentAnimationController.reverse();
      videoViewController.reverse().then((value) {
        setState(() {
          isInSmallMode = false;
        });
      });
    });
  }
},
这是代码的最终结果:
![[译] 挑战 Flutter 之 YouTube(画中画)](https://static.blogweb.cn/article/10e60b600ac54ff99a5cb198c6029aa2.webp) 
最终重新创建的 YouTube 应用
以下是该应用的视频:
- YouTube 视频链接:youtu.be/dTpZ1BtNy4w
最终应用的 iOS 视频
这是该项目的 GitHub 链接:github.com/deven98/You…
感谢阅读此 Flutter 挑战。可以留言告诉我任何你想要在 Flutter 中重新创建的应用。喜欢请给个 star,下次见。
不要错过:
转载自:https://juejin.cn/post/6844903655007584269




