animated_to 0.1.0 copy "animated_to: ^0.1.0" to clipboard
animated_to: ^0.1.0 copied to clipboard

AnimateTo animates its child to a new position requiring no calculation.

example/lib/main.dart

import 'package:animated_to/animated_to.dart';
import 'package:flutter/material.dart';

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

class MyApp extends StatelessWidget {
  const MyApp({super.key});

  @override
  Widget build(BuildContext context) {
    return MaterialApp(home: const AnimatedToSamplePage());
  }
}

class AnimatedToSamplePage extends StatefulWidget {
  const AnimatedToSamplePage({super.key});

  @override
  State<AnimatedToSamplePage> createState() => _AnimatedToSamplePageState();
}

class _AnimatedToSamplePageState extends State<AnimatedToSamplePage>
    with TickerProviderStateMixin {
  var _isExpanded = true;

  /// Some item objects. In this demo, simply a list of [String].
  final _items = List.generate(
    5,
    (index) => index.toString(),
  );

  bool _enabled = true;

  @override
  Widget build(BuildContext context) {
    final children = _items
        .map(
          (item) => _Card(
            item: item,
            index: _items.indexOf(item),
            opacity: 1.0,
            isExpanded: _isExpanded,
            vsync: this,
            enabled: _enabled,
          ),
        )
        .toList();

    return Scaffold(
      appBar: AppBar(title: const Text('Animated Sample Page')),
      // TODO(chooyan-eng): note that [AnimatedTo] doesn't work on scrollable widgets
      body: NotificationListener<ScrollNotification>(
        // workaround to fix scrolling issue by disabling animation when scrolling
        onNotification: (notification) {
          if (notification is ScrollStartNotification) {
            WidgetsBinding.instance.addPostFrameCallback((_) {
              setState(() => _enabled = false);
            });
          }
          if (notification is ScrollEndNotification) {
            WidgetsBinding.instance.addPostFrameCallback((_) {
              setState(() => _enabled = true);
            });
          }
          return true;
        },
        child: SingleChildScrollView(
          child: Center(
            child: Padding(
              padding: const EdgeInsets.all(4),
              child: switch (_isExpanded) {
                // toggle [Wrap] and [Column] with animation
                true => Wrap(
                    spacing: 4,
                    runSpacing: 4,
                    children: children,
                  ),
                false => Column(
                    mainAxisSize: MainAxisSize.min,
                    children: children,
                  ),
              },
            ),
          ),
        ),
      ),
      floatingActionButton: FloatingActionButton.extended(
        onPressed: () {
          setState(() => _isExpanded = !_isExpanded);
        },
        icon: Icon(_isExpanded ? Icons.collections : Icons.expand),
        label: Text(_isExpanded ? 'Collapse' : 'Expand'),
      ),
    );
  }
}

/// A circle or a rectangle shaped widget to be animated with [AnimatedTo].
class _Card extends StatelessWidget {
  const _Card({
    required this.item,
    required this.index,
    required this.opacity,
    required this.isExpanded,
    required this.vsync,
    required this.enabled,
  });

  final String item;
  final int index;
  final double opacity;
  final bool isExpanded;
  final TickerProvider vsync;
  final bool enabled;

  @override
  Widget build(BuildContext context) {
    final size = 60.0;
    return AnimatedTo(
      // try either of [appearingFrom] or [slidingFrom]
      appearingFrom: const Offset(100, 0),
      // slidingFrom: const Offset(0, 100),

      // [TickerProviderStateMixin] is currently required to use [AnimatedTo]
      vsync: vsync,
      // [GlobalObjectKey] is required to identify the widget
      key: GlobalObjectKey(item),
      duration: Duration(milliseconds: 300 + (10 * index)),
      curve: Curves.easeOutQuad,
      enabled: enabled,
      onEnd: (cause) {
        switch (cause) {
          case AnimationEndCause.interrupted:
            // called when the animation is interrupted by another animation
            break;
          case AnimationEndCause.completed:
            // called when the animation is completed
            break;
        }
      },
      // [AnimatedTo] can be combined with some Animated widgets
      child: Container(
        // duration: Duration(milliseconds: 300 + (10 * index)),
        // curve: Curves.easeIn,
        decoration: BoxDecoration(
          color: Colors.primaries[int.parse(item) % Colors.primaries.length]
              .withAlpha(isExpanded ? 255 : 128),
          borderRadius: isExpanded
              ? BorderRadius.circular(100)
              : BorderRadius.circular(4),
        ),
        margin: const EdgeInsets.all(2),
        width: size,
        height: size,
        child: Center(
          child: Text(
            item,
            style: TextStyle(
              color: Colors.white.withAlpha(isExpanded ? 255 : 128),
              fontSize: 24,
              fontWeight: FontWeight.bold,
            ),
          ),
        ),
      ),
    );
  }
}
83
likes
0
points
519
downloads

Publisher

unverified uploader

Weekly Downloads

AnimateTo animates its child to a new position requiring no calculation.

Repository (GitHub)
View/report issues

Topics

#animation

License

unknown (license)

Dependencies

flutter

More

Packages that depend on animated_to