Awesome Flutter Package for adding smooth text animations
Contents
Introduction
A package is a namespace that holds a collection of classes, interfaces, and sub-packages that are all of the same kind. Packages might be compared to different folders on our computers, where we might have videos in one, photographs in another, software in yet another, and so on. Dart organizes and shares a set of functionality through packages in Flutter. Flutter always supports shared packages, which are provided to the Flutter and Dart ecosystem by other developers. The packages enable us to create the app without having to start from scratch.
Animated Text Kit
A flutter package which contains a collection of some cool and awesome text animations. Recommended package for text animations in Codemagic’s Ebook, “Flutter libraries we love”. Try out our live example app.
Flutter Package of the Week
Installing
1. Depend on it
Add this to your package’s pubspec.yaml
file:
dependencies: animated_text_kit: ^4.2.1
2. Install it
You can install packages from the command line:
with pub
:
$ pub get
with Flutter
:
$ flutter pub get
3. Import it
Now in your Dart
code, you can use:
import 'package:animated_text_kit/animated_text_kit.dart';
Usage
AnimatedTextKit
is a Stateful Widget that produces text animations. Include it in your build
method like:
AnimatedTextKit( animatedTexts: [ TypewriterAnimatedText( 'Hello world!', textStyle: const TextStyle( fontSize: 32.0, fontWeight: FontWeight.bold, ), speed: const Duration(milliseconds: 2000), ), ], totalRepeatCount: 4, pause: const Duration(milliseconds: 1000), displayFullTextOnTap: true, stopPauseOnTap: true, )
It has many configurable properties, including:
pause
– the time of the pause between animation textsdisplayFullTextOnTap
– tapping the animation will rush it to completionisRepeatingAnimation
– controls whether the animation repeatsrepeatForever
– controls whether the animation repeats forevertotalRepeatCount
– number of times the animation should repeat (whenrepeatForever
isfalse
)
There are also custom callbacks:
onTap
– This is called when a user taps the animated textonNext(int index, bool isLast)
– This is called before the next text animation, after the previous one’s pauseonNextBeforePause(int index, bool isLast)
– This is called before the next text animation, before the previous one’s pauseonFinished
– This is called at the end, when the parameterisRepeatingAnimation
is set tofalse
Note: You might come up with an issue that the text
does not get updated with setState
as shown here. The solution to this, is a key that changes based on the text. For reference, watch this video.
New with Version 3
Version 3 refactored the code so that common animation controls were moved to AnimatedTextKit
and all animations, except for TextLiquidFill
, extend from AnimatedText
. This saved hundreds of lines of duplicate code, increased consistency across animations, and makes it easier to create new animations.
It also makes the animations more flexible because multiple animations may now be easily combined. For example:
AnimatedTextKit( animatedTexts: [ FadeAnimatedText( 'Fade First', textStyle: TextStyle(fontSize: 32.0, fontWeight: FontWeight.bold), ), ScaleAnimatedText( 'Then Scale', textStyle: TextStyle(fontSize: 70.0, fontFamily: 'Canterbury'), ), ], ),
Using the legacy FadeAnimatedTextKit
is equivalent to using AnimatedTextKit
with FadeAnimatedText
. An advantage of AnimatedTextKit
is that the animatedTexts
may be any subclass of AnimatedText
, while using FadeAnimatedTextKit
essentially restricts you to using just FadeAnimatedText
.
Legacy AnimatedTextKit classes
Have you noticed that animation classes come in pairs? For example, there is FadeAnimatedText
and FadeAnimatedTextKit
. The significant refactoring with Version 3 split the original FadeAnimatedTextKit
into FadeAnimatedText
and a re-usable AnimatedTextKit
, then FadeAnimatedTextKit
was adjusted for backwards compatibility.
When introducing a new AnimationText
subclass, you may wonder if you also need to also introduce an additional Kit
class. The answer is NO.
Going forward, we are championing the adoption of the Version 3 approach, and have deprecated the legacy Kit
classes. This will make creating new animations easier. We know it makes some legacy code more verbose, but the flexibility and simplicity is a conscious trade-off.
Animations
Many animations are provided, but you can also create your own animations.
Rotate
Row( mainAxisSize: MainAxisSize.min, children: <Widget>[ const SizedBox(width: 20.0, height: 100.0), const Text( 'Be', style: TextStyle(fontSize: 43.0), ), const SizedBox(width: 20.0, height: 100.0), DefaultTextStyle( style: const TextStyle( fontSize: 40.0, fontFamily: 'Horizon', ), child: AnimatedTextKit( animatedTexts: [ RotateAnimatedText('AWESOME'), RotateAnimatedText('OPTIMISTIC'), RotateAnimatedText('DIFFERENT'), ], onTap: () { print("Tap Event"); }, ), ), ], );
Note: You can override transition height by setting the value of parameter transitionHeight
for RotateAnimatedTextKit class.
Fade
return SizedBox( width: 250.0, child: DefaultTextStyle( style: const TextStyle( fontSize: 32.0, fontWeight: FontWeight.bold, ), child: AnimatedTextKit( animatedTexts: [ FadeAnimatedText('do IT!'), FadeAnimatedText('do it RIGHT!!'), FadeAnimatedText('do it RIGHT NOW!!!'), ], onTap: () { print("Tap Event"); }, ), ), );
Typer
return SizedBox( width: 250.0, child: DefaultTextStyle( style: const TextStyle( fontSize: 30.0, fontFamily: 'Bobbers', ), child: AnimatedTextKit( animatedTexts: [ TyperAnimatedText('It is not enough to do your best,'), TyperAnimatedText('you must know what to do,'), TyperAnimatedText('and then do your best'), TyperAnimatedText('- W.Edwards Deming'), ], onTap: () { print("Tap Event"); }, ), ), );
Typewriter
return SizedBox( width: 250.0, child: DefaultTextStyle( style: const TextStyle( fontSize: 30.0, fontFamily: 'Agne', ), child: AnimatedTextKit( animatedTexts: [ TypewriterAnimatedText('Discipline is the best tool'), TypewriterAnimatedText('Design first, then code'), TypewriterAnimatedText('Do not patch bugs out, rewrite them'), TypewriterAnimatedText('Do not test bugs out, design them out'), ], onTap: () { print("Tap Event"); }, ), ), );
Scale
return SizedBox( width: 250.0, child: DefaultTextStyle( style: const TextStyle( fontSize: 70.0, fontFamily: 'Canterbury', ), child: AnimatedTextKit( animatedTexts: [ ScaleAnimatedText('Think'), ScaleAnimatedText('Build'), ScaleAnimatedText('Ship'), ], onTap: () { print("Tap Event"); }, ), ), );
Colorize
const colorizeColors = [ Colors.purple, Colors.blue, Colors.yellow, Colors.red, ]; const colorizeTextStyle = TextStyle( fontSize: 50.0, fontFamily: 'Horizon', ); return SizedBox( width: 250.0, child: AnimatedTextKit( animatedTexts: [ ColorizeAnimatedText( 'Larry Page', textStyle: colorizeTextStyle, colors: colorizeColors, ), ColorizeAnimatedText( 'Bill Gates', textStyle: colorizeTextStyle, colors: colorizeColors, ), ColorizeAnimatedText( 'Steve Jobs', textStyle: colorizeTextStyle, colors: colorizeColors, ), ], isRepeatingAnimation: true, onTap: () { print("Tap Event"); }, ), );
Note: colors
list should contains at least two values.
TextLiquidFill
return SizedBox( width: 250.0, child: TextLiquidFill( text: 'LIQUIDY', waveColor: Colors.blueAccent, boxBackgroundColor: Colors.redAccent, textStyle: TextStyle( fontSize: 80.0, fontWeight: FontWeight.bold, ), boxHeight: 300.0, ), );
To get more information about how the animated text made from scratch by @HemilPanchiwala, visit the Medium blog.
Wavy
return DefaultTextStyle( style: const TextStyle( fontSize: 20.0, ), child: AnimatedTextKit( animatedTexts: [ WavyAnimatedText('Hello World'), WavyAnimatedText('Look at the waves'), ], isRepeatingAnimation: true, onTap: () { print("Tap Event"); }, ), );
Flicker
return SizedBox( width: 250.0, child: DefaultTextStyle( style: const TextStyle( fontSize: 35, color: Colors.white, shadows: [ Shadow( blurRadius: 7.0, color: Colors.white, offset: Offset(0, 0), ), ], ), child: AnimatedTextKit( repeatForever: true, animatedTexts: [ FlickerAnimatedText('Flicker Frenzy'), FlickerAnimatedText('Night Vibes On'), FlickerAnimatedText("C'est La Vie !"), ], onTap: () { print("Tap Event"); }, ), ), );
Create your own Animations
You can easily create your own animations by creating new classes that extend AnimatedText
, just like most animations in this package. You will need to implement:
- Class constructor – Initializes animation parameters.
initAnimation
– InitializesAnimation
instances and binds them to the givenAnimationController
.animatedBuilder
– Builder method to return aWidget
based onAnimation
values.completeText
– Returns theWidget
to display once the animation is complete. (The default implementation returns a styledText
widget.)
Then use AnimatedTextKit
to display the custom animated text class like:
AnimatedTextKit( animatedTexts: [ CustomAnimatedText( 'Insert Text Here', textStyle: const TextStyle( fontSize: 32.0, fontWeight: FontWeight.bold, ), ), ], ),
Example
- main.dart
import 'package:flutter/material.dart';
import 'package:animated_text_kit/animated_text_kit.dart';
void main() => runApp(MyApp());
class MyApp extends StatefulWidget {
/// This widget is the root of your application.
@override
_MyAppState createState() => _MyAppState();
}
class _MyAppState extends State<MyApp> {
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Animated Text Kit',
debugShowCheckedModeBanner: false,
theme: ThemeData.dark(),
home: MyHomePage(),
);
}
}
class MyHomePage extends StatefulWidget {
MyHomePage({Key? key}) : super(key: key);
@override
_MyHomePageState createState() => _MyHomePageState();
}
class _MyHomePageState extends State<MyHomePage> {
late List<AnimatedTextExample> _examples;
int _index = 0;
int _tapCount = 0;
@override
void initState() {
super.initState();
_examples = animatedTextExamples(onTap: () {
print('Tap Event');
setState(() {
_tapCount++;
});
});
}
@override
Widget build(BuildContext context) {
final animatedTextExample = _examples[_index];
return Scaffold(
appBar: AppBar(
title: Text(
animatedTextExample.label,
style: TextStyle(fontSize: 30.0, fontWeight: FontWeight.bold),
),
),
body: Column(
children: <Widget>[
Expanded(
child: Container(),
),
Container(
decoration: BoxDecoration(color: animatedTextExample.color),
height: 300.0,
width: 300.0,
child: Center(
key: ValueKey(animatedTextExample.label),
child: animatedTextExample.child,
),
),
Expanded(
child: Container(
alignment: Alignment.center,
child: Text('Taps: $_tapCount'),
),
),
],
),
floatingActionButton: FloatingActionButton(
onPressed: () {
setState(() {
_index = ++_index % _examples.length;
_tapCount = 0;
});
},
tooltip: 'Next',
child: const Icon(
Icons.play_circle_filled,
size: 50.0,
),
),
);
}
}
class AnimatedTextExample {
final String label;
final Color? color;
final Widget child;
const AnimatedTextExample({
required this.label,
required this.color,
required this.child,
});
}
// Colorize Text Style
const _colorizeTextStyle = TextStyle(
fontSize: 50.0,
fontFamily: 'Horizon',
);
// Colorize Colors
const _colorizeColors = [
Colors.purple,
Colors.blue,
Colors.yellow,
Colors.red,
];
List<AnimatedTextExample> animatedTextExamples({VoidCallback? onTap}) =>
<AnimatedTextExample>[
AnimatedTextExample(
label: 'Rotate',
color: Colors.orange[800],
child: ListView(
scrollDirection: Axis.horizontal,
children: <Widget>[
Row(
mainAxisSize: MainAxisSize.min,
children: <Widget>[
const SizedBox(
width: 20.0,
height: 100.0,
),
const Text(
'Be',
style: TextStyle(fontSize: 43.0),
),
const SizedBox(
width: 20.0,
height: 100.0,
),
DefaultTextStyle(
style: TextStyle(
fontSize: 40.0,
fontFamily: 'Horizon',
),
child: AnimatedTextKit(
animatedTexts: [
RotateAnimatedText('AWESOME'),
RotateAnimatedText('OPTIMISTIC'),
RotateAnimatedText(
'DIFFERENT',
textStyle: const TextStyle(
decoration: TextDecoration.underline,
),
),
],
onTap: onTap,
isRepeatingAnimation: true,
totalRepeatCount: 10,
),
),
],
),
],
),
),
AnimatedTextExample(
label: 'Fade',
color: Colors.brown[600],
child: DefaultTextStyle(
style: const TextStyle(
fontSize: 32.0,
fontWeight: FontWeight.bold,
),
child: AnimatedTextKit(
animatedTexts: [
FadeAnimatedText('do IT!'),
FadeAnimatedText('do it RIGHT!!'),
FadeAnimatedText('do it RIGHT NOW!!!'),
],
onTap: onTap,
),
),
),
AnimatedTextExample(
label: 'Typer',
color: Colors.lightGreen[800],
child: SizedBox(
width: 250.0,
child: DefaultTextStyle(
style: const TextStyle(
fontSize: 30.0,
fontFamily: 'Bobbers',
),
child: AnimatedTextKit(
animatedTexts: [
TyperAnimatedText('It is not enough to do your best,'),
TyperAnimatedText('you must know what to do,'),
TyperAnimatedText('and then do your best'),
TyperAnimatedText('- W.Edwards Deming'),
],
onTap: onTap,
),
),
),
),
AnimatedTextExample(
label: 'Typewriter',
color: Colors.teal[700],
child: SizedBox(
width: 250.0,
child: DefaultTextStyle(
style: const TextStyle(
fontSize: 30.0,
fontFamily: 'Agne',
),
child: AnimatedTextKit(
animatedTexts: [
TypewriterAnimatedText('Discipline is the best tool'),
TypewriterAnimatedText('Design first, then code', cursor: '|'),
TypewriterAnimatedText('Do not patch bugs out, rewrite them',
cursor: '<|>'),
TypewriterAnimatedText('Do not test bugs out, design them out',
cursor: '?'),
],
onTap: onTap,
),
),
),
),
AnimatedTextExample(
label: 'Scale',
color: Colors.blue[700],
child: DefaultTextStyle(
style: const TextStyle(
fontSize: 70.0,
fontFamily: 'Canterbury',
),
child: AnimatedTextKit(
animatedTexts: [
ScaleAnimatedText('Think'),
ScaleAnimatedText('Build'),
ScaleAnimatedText('Ship'),
],
onTap: onTap,
),
),
),
AnimatedTextExample(
label: 'Colorize',
color: Colors.blueGrey[50],
child: AnimatedTextKit(
animatedTexts: [
ColorizeAnimatedText(
'Larry Page',
textStyle: _colorizeTextStyle,
colors: _colorizeColors,
),
ColorizeAnimatedText(
'Bill Gates',
textStyle: _colorizeTextStyle,
colors: _colorizeColors,
),
ColorizeAnimatedText(
'Steve Jobs',
textStyle: _colorizeTextStyle,
colors: _colorizeColors,
),
],
onTap: onTap,
),
),
AnimatedTextExample(
label: 'TextLiquidFill',
color: Colors.white,
child: TextLiquidFill(
text: 'LIQUIDY',
waveColor: Colors.blueAccent,
boxBackgroundColor: Colors.redAccent,
textStyle: const TextStyle(
fontSize: 70,
fontWeight: FontWeight.bold,
),
boxHeight: 300,
),
),
AnimatedTextExample(
label: 'Wavy Text',
color: Colors.purple,
child: DefaultTextStyle(
style: const TextStyle(
fontSize: 20.0,
),
child: AnimatedTextKit(
animatedTexts: [
WavyAnimatedText(
'Hello World',
textStyle: const TextStyle(
fontSize: 24.0,
fontWeight: FontWeight.bold,
),
),
WavyAnimatedText('Look at the waves'),
WavyAnimatedText('They look so Amazing'),
],
onTap: onTap,
),
),
),
AnimatedTextExample(
label: 'Flicker',
color: Colors.pink[300],
child: DefaultTextStyle(
style: const TextStyle(
fontSize: 35,
color: Colors.white,
shadows: [
Shadow(
blurRadius: 7.0,
color: Colors.white,
offset: Offset(0, 0),
),
],
),
child: AnimatedTextKit(
repeatForever: true,
animatedTexts: [
FlickerAnimatedText('Flicker Frenzy'),
FlickerAnimatedText('Night Vibes On'),
FlickerAnimatedText("C'est La Vie !"),
],
onTap: onTap,
),
),
),
AnimatedTextExample(
label: 'Combination',
color: Colors.pink,
child: AnimatedTextKit(
onTap: onTap,
animatedTexts: [
WavyAnimatedText(
'On Your Marks',
textStyle: const TextStyle(
fontSize: 24.0,
),
),
FadeAnimatedText(
'Get Set',
textStyle: const TextStyle(
fontSize: 32.0,
fontWeight: FontWeight.bold,
),
),
ScaleAnimatedText(
'Ready',
textStyle: const TextStyle(
fontSize: 48.0,
fontWeight: FontWeight.bold,
),
),
RotateAnimatedText(
'Go!',
textStyle: const TextStyle(
fontSize: 64.0,
),
rotateOut: false,
duration: const Duration(milliseconds: 400),
)
],
),
),
];
Bugs or Requests
If you encounter any problems feel free to open an issue. If you feel the library is missing a feature, please raise a ticket on GitHub and I’ll look into it. Pull request are also welcome.
See Contributing.md.
GitHub
Source Code: Awesome Flutter Package for adding smooth text animations.