Анимировать ListView элемент на весь экран в Flutter - PullRequest
0 голосов
/ 22 мая 2018

Я бы хотел, чтобы элементы моего списка выполняли эту анимацию ( mp4 ) при нажатии.Я попытался использовать AnimatedCrossFade, но для этого требуется, чтобы два его потомка были на одном уровне, например, детальный вид пересекается с ListView, а не с элементом, к которому прикоснулись.На самом деле кажется, что анимация Hero - единственная, которая может анимировать различные виджеты.

У меня проблемы с использованием Hero.Должен ли он обернуть элемент списка?Имеет ли значение, что поддерево Виджета существенно отличается в источнике / назначении героя?Кроме того, можно ли использовать анимацию героев с LocalHistoryRoute с или смещенными анимациями ?

Редактировать

Теперь это похоже на то, что мне нужно сделатьэто использовать Overlay, сложная часть в том, что мне нужно добавить выбранный элемент в оверлей в той же позиции на экране, где он был нажат, тогда анимация будет легкой.Здесь возможно использование паттерна цели / последователя, например CompositedTransformTarget

Ответы [ 2 ]

0 голосов
/ 29 мая 2018

Я бы просто обманул и обернул все это в стек - нижний слой был бы страницей с AppBar, а верхний слой был бы прозрачным до тех пор, пока не будет нарисован.

onTap, дублируйте ListTile на верхнюю поверхность, а затем анимация героя заполняет весь экран.Это не очень элегантно, но фреймворк (пока) не обеспечивает легкого покрытия AppBar, поэтому готовый холст для рисования для других сложных анимаций может быть изобретательным.

0 голосов
/ 28 мая 2018

Вы можете просто использовать виджет Hero для создания такой анимации.Вот мой пример:

enter image description here

и исходный код:

import 'package:flutter/material.dart';

void main() => runApp(new MyApp());

class MyApp extends StatelessWidget {
  // This widget is the root of your application.
  @override
  Widget build(BuildContext context) {
    return new MaterialApp(
      title: 'Flutter Demo',
      theme: new ThemeData(
        primarySwatch: Colors.blue,
      ),
      home: new FirstPage(title: 'Color Palette'),
    );
  }
}

class FirstPage extends StatefulWidget {
  FirstPage({Key key, this.title}) : super(key: key);

  final String title;

  @override
  _FirstPageState createState() => new _FirstPageState();
}

class _FirstPageState extends State<FirstPage> {
  final palette = [
    {'#E53935': 0xFFE53935},
    {'#D81B60': 0xFFD81B60},
    {'#8E24AA': 0xFF8E24AA},
    {'#5E35B1': 0xFF5E35B1},
    {'#3949AB': 0xFF3949AB},
    {'#1E88E5': 0xFF1E88E5},
    {'#039BE5': 0xFF039BE5},
    {'#00ACC1': 0xFF00ACC1},
    {'#00897B': 0xFF00897B},
    {'#43A047': 0xFF43A047},
    {'#7CB342': 0xFF7CB342},
    {'#C0CA33': 0xFFC0CA33},
  ];

  @override
  Widget build(BuildContext context) {
    return new Scaffold(
      appBar: new AppBar(
        title: new Text(widget.title),
      ),
      body: new Container(
        child: new ListView.builder(
            itemCount: palette.length,
            itemBuilder: (context, index) => new Hero(
                  tag: palette[index].keys.first,
                  child: new GestureDetector(
                    onTap: () {
                      Navigator
                          .of(context)
                          .push(new ColorPageRoute(palette[index]));
                    },
                    child: new Container(
                      height: 64.0,
                      width: double.infinity,
                      color: new Color(palette[index].values.first),
                      child: new Center(
                        child: new Hero(
                          tag: 'text-${palette[index].keys.first}',
                          child: new Text(
                            palette[index].keys.first,
                            style: Theme.of(context).textTheme.title.copyWith(
                                  color: Colors.white,
                                ),
                          ),
                        ),
                      ),
                    ),
                  ),
                )),
      ),
    );
  }
}

class SecondPage extends StatelessWidget {
  final Map<String, int> color;

  SecondPage({this.color});

  @override
  Widget build(BuildContext context) {
    return new Scaffold(
      appBar: new AppBar(
        title: new Text('Color'),
      ),
      body: new Hero(
        tag: color.keys.first,
        child: new Container(
          color: new Color(color.values.first),
          child: new Center(
            child: new Hero(
              tag: 'text-${color.keys.first}',
              child: new Text(
                color.keys.first,
                style:
                    Theme.of(context).textTheme.title.copyWith(color: Colors.white),
              ),
            ),
          ),
        ),
      ),
    );
  }
}

class ColorPageRoute extends MaterialPageRoute {
  ColorPageRoute(Map<String, int> color)
      : super(
            builder: (context) => new SecondPage(
                  color: color,
                ));

  @override
  Widget buildTransitions(BuildContext context, Animation<double> animation,
      Animation<double> secondaryAnimation, Widget child) {
    return FadeTransition(opacity: animation, child: child);
  }
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...