zoukankan      html  css  js  c++  java
  • Flutter 实现酷炫的3D效果

    老孟导读:此文讲解3个酷炫的3D动画效果。

     

    下面是要实现的效果:

    Flutter 中3D效果是通过 Transform 组件实现的,没有变换效果的实现:

    class TransformDemo extends StatelessWidget {
      @override
      Widget build(BuildContext context) {
        return Scaffold(
          appBar: AppBar(
            title: Text('3D 变换Demo'),
          ),
          body: Container(
            alignment: Alignment.center,
            color: Colors.white,
            child: Text('3D 变换Demo'),
          ),
        );
      }
    }

    通过 GestureDetector 组件添加滑动事件监听:

    @override
    Widget build(BuildContext context) {
      return Scaffold(
        appBar: AppBar(
          title: Text('3D 变换Demo'),
        ),
        body: GestureDetector(
          onPanUpdate: (details) {
            print('$details');
          },
          child: Container(
            alignment: Alignment.center,
            color: Colors.white,
            child: Text('3D 变换Demo'),
          ),
        ),
      );
    }

    添加 Transform 对组件进入旋转:

    @override
    Widget build(BuildContext context) {
      return Transform(
          transform: Matrix4.identity()
            ..setEntry(3, 2, 0.001)
            ..rotateX(pi/6)
            ..rotateY(pi/6),
          alignment: Alignment.center,
          child: Scaffold(
            appBar: AppBar(
              title: Text('3D 变换Demo'),
            ),
            body: GestureDetector(
              onPanUpdate: (details) {
              },
              child: Container(
                alignment: Alignment.center,
                color: Colors.white,
                child: Text('3D 变换Demo'),
              ),
            ),
          ));
    }

    将滑动的偏移和旋转进行关联:

    class TransformDemo extends StatefulWidget {
      @override
      _TransformDemoState createState() => _TransformDemoState();
    }

    class _TransformDemoState extends State<TransformDemo> {
      double _rotateX = .0;
      double _rotateY = .0;

      @override
      Widget build(BuildContext context) {
        return Transform(
            transform: Matrix4.identity()
              ..rotateX(_rotateX)
              ..rotateY(_rotateY),
            alignment: Alignment.center,
            child: Scaffold(
              appBar: AppBar(
                title: Text('3D 变换Demo'),
              ),
              body: GestureDetector(
                onPanUpdate: (details) {
                  setState(() {
                    _rotateX += details.delta.dy * .01;
                    _rotateY += details.delta.dx * -.01;
                  });
                },
                child: Container(
                  alignment: Alignment.center,
                  color: Colors.white,
                  child: Text('3D 变换Demo'),
                ),
              ),
            ));
      }
    }

    基本已经实现了3D效果,但效果比较生硬,尤其垂直方向旋转的时候远点和近点在屏幕上的宽度是一样,

    添加近大远小的效果:

    Transform(
        transform: Matrix4.identity()
          ..setEntry(3, 2, 0.001)
          ..rotateX(_rotateX)
          ..rotateY(_rotateY),
      ...

    翻书效果

    上面的效果类似于翻书的效果。

    实现的原理:

    将图片左右切割为两部分,两张图片共分割为4个新的组件,如下图,分别为1、2、3、4

    代码实现:

    _child1 ClipRect(
      child: Align(
        alignment: Alignment.centerLeft,
        widthFactor: 0.5,
        child: child1,
      ),
    );
    _child2 ClipRect(
      child: Align(
        alignment: Alignment.centerRight,
        widthFactor: 0.5,
        child: child1,
      ),
    );

    _child3 ClipRect(
      child: Align(
        alignment: Alignment.centerLeft,
        widthFactor: 0.5,
        child: child2,
      ),
    );

    _child4 ClipRect(
      child: Align(
        alignment: Alignment.centerRight,
        widthFactor: 0.5,
        child: child2,
      ),
    );

    将第一张图片放在第二种图片的上面,先旋转 组件2 从 0度到 90度,然后再旋转 组件3 从 -90度到0度,代码实现:

    Row(
      mainAxisAlignment: MainAxisAlignment.center,
      children: <Widget>[
        Stack(www.youy2zhuce.cn
          children: [
            _child1,
            Transform(
              alignment: Alignment.centerRight,
              transform: Matrix4.identity()
                ..setEntry(3, 2, 0.001)
                ..rotateY(www.jujinyule.com_animation1.value),
              child: _child3,
            ),
          ],
        ),
        Container(
           3,
          color: Colors.white,
        ),
        Stack(
          children: [
            _child4,
            Transform(
              alignment: Alignment.centerLeft,
              transform: Matrix4.identity()
                ..setEntry(3, 2, 0.001)
                ..rotateY(_animation.value),
              child: _child2,
            )
          ],
        )
      ],
    )

    动画控制器设置:

    @override
    void initState(www.gaodeyulept.cn) {
      init();
      _controller =
          AnimationController(vsync: thisduration: Duration(seconds: 5))
            ..addListener(() {
              setState(() {});
            });
      _animation Tween(begin: .0end: pi / 2)
          .animate(CurvedAnimation(parent: _controller, curve: Interval(.0, .5)));
      _animation1 Tween(begin: -pi www.lanboyulezc.cn/ 2end: 0.0).animate(
          CurvedAnimation(parent: _controller, curve: Interval(.5, 1.0)));
      _controller.forward();
      super.initState(www.shentuylgw.cn);
    }

    其中 child1, child2为两种图片,代码如下:

    _FlipUpDemoState(
        Container(
           300,
          height: 400,
          child: Image.asset(
            'assets/images/b.jpg',
            fit: BoxFit.cover,
          ),
        ),
        Container(
           300,
          height: 400,
          child: Image.asset(
            'assets/images/c.www.tengyao3zc.cn g',
            fit: BoxFit.cover,
          ),
        ))

    最后生成的效果就是开始的翻书效果。

    上面是左右翻页效果,同理换成上下翻页效果:

    @override
    Widget build(BuildContext context) {
      return Scaffold(
        appBar: AppBar(www.javachenglei.com ),
        body: Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: <Widget>[
            Stack(
              children: [
                _upperChild1,
                Transform(
                  alignment: Alignment.bottomCenter,
                  transform: Matrix4.identity()
                    ..setEntry(3, 2, 0.003)
                    ..rotateX(_animation1.value),
                  child: _upperChild2,
                ),
              ],
            ),
            SizedBox(
              height: 2,
            ),
            Stack(
              children:www.chuancenpt.com [
                _lowerChild2,
                Transform(
                  alignment: Alignment.topCenter,
                  transform: Matrix4.identity()
                    ..setEntry(3, www.jintianxuesha.com 2, 0.003)
                    ..rotateX(_animation.value),
                  child: _lowerChild1,
  • 相关阅读:
    一元2次方程求解
    react-native 相对项目路径导入组件
    appium页面元素封装(十一)
    pytest高级用法,跳过、失败函数,函数数据参数化(十)
    python之自定义分页代码模板
    pytest高级用法,参数、函数、自动、返回值引用(九)
    pytest.ini文件配置
    因安装pytest最新版本导致,运行pytest命令出错:pluggy.manager.PluginValidationError: unknown hook 'pytest_namespace' in plugin <module 'allure.pytest_plugin' from 'd:\anaconda3\lib\site-packages\allure\pytest_plug
    pytest的使用(八)
    运行pytest,提示编码错误
  • 原文地址:https://www.cnblogs.com/woshixiaowang/p/13369047.html
Copyright © 2011-2022 走看看