Last updated:
0 purchases
animated styled widget
animated_styled_widget #
Powerful styling, serialization, animation, and custom components all in one place.
Current features of this package:
Let you use a serializable style to design highly customizable StyledContainer widget.
Implicit animations with the AnimatedStyledContainer widget.
Explicit animations (local/global, timed/scroll based) with the ExplicitAnimatedStyledContainer widget.
Styled Components: StyledButton, StyledToggleButtons, StyledSwitch, StyledRadio, StyledCheckbox, StyledSlider. Your app don't need to look like Material.
Many features of this package is inspired by CSS or follows CSS specifications. If you are familiar with CSS, this package should be even easier to use.
What is the Style class? #
The style class is a collection of UI data classes.
It currently supports the following properties:
Sizing and Aligning
Dimension width
Dimension height
Alignment alignment
Alignment childAlignment
EdgeInsets margin
EdgeInsets padding
copied to clipboard
Shape and Decoration
BoxDecorartion backgroundDecoration
BoxDecorartion foregroundDecoration
List<ShapeShadow> shadows
List<ShapeShadow> insetShadows
MorphableShapeBorder shapeBorder
copied to clipboard
Visibility
bool visible
double opacity
copied to clipboard
Transformation
SmoothMatrix4 transform
Alignment transformAlignment
copied to clipboard
Typography
DynamicTextStyle textStyle
TextAlign textAlign
copied to clipboard
Mouse cursor style
SystemMouseCursor mouseCursor
copied to clipboard
Shader and ImageFilter
Gradient shaderGradient
ImageFilter imageFilter
ImageFilter backdropFilter
copied to clipboard
The Dimension type is from the dimension package. It supports both absolute and relative units. You can also combine/nest min/max/clamp functions on Dimension. You can think of this as a supercharged combination of SizedBox and FractionallySizedBox.
ShapeShadow and MorphableShapeBorder are from the morphable_shape package. ShapeShadow supports inset shadows and gradient filling, in addition to what BoxShadow supports. MorpableShapeBorder supports many commonly used shapes, shape morphing, and many more. Check out fluttershape.com for an interactive demo.
DynamicTextStyle lets you define font size, letter spacing etc using absolute/relative values. 300% font size means 3 times the default font size.
SmoothMatrix4 is similar to Matrix4 but ensures that all the transformations can be smoothly animated. It also allows you to use Dimension as translation distances to adapt to different screen sizes.
The layout model and paint order is shown below:
An example of a responsive style:
Style style=Style(
alignment: Alignment.center,
width: 50.toVWLength,
height: 50.toPercentLength,
margin: EdgeInsets.symmetric(vertical: 10),
backgroundDecoration: BoxDecoration(
gradient:
LinearGradient(colors: [Colors.cyanAccent, Colors.purpleAccent])),
shapeBorder: RoundedRectangleShapeBorder(
borderRadius:
DynamicBorderRadius.all(DynamicRadius.circular(15.toPXLength))
);
copied to clipboard
After you have defined a style, use the StyledContainer widget:
var widget=StyledContainer(
style: style,
child: ...
);
copied to clipboard
Implicit Animation #
Almost every property in the Style class can be animated. See the
following GIF for a demonstration:
Just replace the StyledContainer with AnimatedStyledContainer and provide a duration and a curve. Notice the animation can not only be triggered by providing a new style/style map, but also by window resizing/screen rotation as long as you provide the appropriate styles.
Styled Components #
Styled components is a selection of UI components with simple logic like button, radio button, toggle button, switch, etc. The Material implementation of those components is great, but the customization ability is quited limited. There are many other packages out there for even more customization, but the customizability is typically limited to color, border radius, sizing, etc. This package gives you much more.
The Material components animates according to its internal MaterialState like hovered, pressed, selected, etc. So does the styled components. Lets look at the StyleButton's constructor as an example:
StyledButton(
{Key? key,
this.onPressed,
required this.style,
this.hoveredStyle,
this.pressedStyle,
this.disabledStyle,
this.curve = Curves.linear,
this.duration = const Duration(milliseconds: 100),
Widget? child});
StyledButton.builder(
{Key? key,
this.onPressed,
required this.style,
this.hoveredStyle,
this.pressedStyle,
this.disabledStyle,
this.curve = Curves.linear,
this.duration = const Duration(milliseconds: 100),
required this.builder});
copied to clipboard
A button can be in one of the four states: idle, hovered, pressed, disabled. You need to provide the four corresponding styles. Notice only the idle(default) style is required, and the other three will be set to this style if not provided. You can also specify the duration and animation curve for transitioning between different states. The child parameter is the child widget of this button. But you can also provide a builder which can access the internal state of the button:
builder: (context, state) {
Widget child;
switch (state) {
case StyledState.pressed:
child = Text("TAPPED", key: UniqueKey());
break;
default:
child = Text("TAP ME", key: UniqueKey());
break;
}
return AnimatedSwitcher(
duration: Duration(milliseconds: 200),
child: child,
);
}
copied to clipboard
Above are six different styled buttons that works great with/without a cursor.
StyledCheckbox is very similar to StyledButton except for that the pressedStyle parameter is replaced by selectedStyle, as checkbox (and other toggleable components) cares about whether itself is selected (not pressed).
And then there's the StyledRadio:
And StyledToggleButtons:
Notice all those components follow the exact parameter naming schemes as the built-in ones except for the styling part.
The StyledSwitch and StyledSlider are a little tricker, as now the styling involves two components. For StyledSwitch, you need to provide styling for both the track and the thumb. And the thumb will be aligned to either the left ot right side of the track (if the switch is horizontal).
The StyledSlider need styling for the track and thumb as well, but you can also provide styling for the active track and tooltip.
The interface for the slider is not final, I may add more styling paramters in the future.
I have also implemented the StyledNavigationBar and StylePopupMenuButton. You can find demos of all those components in the example app.
The next step may be adding themeing capabilities to the styled components so that people.
Explicit Animation #
Implicit animations are easy to use but can not achieve every effect we want. That's when the ExplicitAnimatedStyledContainer comes in:
Widget widget = ExplicitAnimatedStyledContainer(
style: style,
child: child,
localAnimations: localAnimations,
globalAnimationIds: globalAnimationIds,
id: id,
...
);
copied to clipboard
You still provide an initial style to the widget, but then you use local/global animations to animate the widget’s style. Let’s first talk about the localAnimations:
Map<AnimationTrigger, MultiAnimationSequence> localAnimations
copied to clipboard
It is a map between AnimationTrigger and MultiAnimationSequence. Currently supported AnimationTriggers are the following:
enum AnimationTrigger {
mouseEnter,
mouseExit,
tap,
visible,
scroll,
}
copied to clipboard
When a trigger event happens(e.g. you tapped this widget), the corresponding MultiAnimationSequence is fired. A MultiAnimationSequence contains a sequences map:
Map<AnimationProperty, AnimationSequence> sequences
copied to clipboard
where AnimationProperty is an enum class corresponding to every animatable property the Style class supports, and AnimationSequence is a list of generic values, durations, delays, and curves that tells us how a certain animation property is evolved. For example:
MultiAnimationSequence(sequences: {
AnimationProperty.width: AnimationSequence()
..add(
delay: Duration(seconds: 1),
duration: Duration(milliseconds: 200),
curve: Curves.linear,
value: 100.toPXLength)
..add(
duration: Duration(milliseconds: 200),
curve: Curves.easeIn,
value: 50.toVWLength)
});
copied to clipboard
will delay 1 second, then animate the width from its current value to 100 px in 200ms, then to 50% screen width in 200ms. You can animate other properties using the same syntax.
The above mouse hover effect is achieved by writing:
Widget widget = ExplicitAnimatedStyledContainer(
style: style,
child: child,
localAnimations: {
AnimationTrigger.mouseEnter: enterSequence,
AnimationTrigger.mouseExit: exitSequence,
}
);
copied to clipboard
You can have different durations and curves for mouse entering and exiting, and also for different style properties.
Now let's talk about other animation triggers. The AnimationTrigger.tap is easy to understand. The AnimationTrigger.visible is triggered when the widget becomes visible in the viewport (by using the visibility_detector package). The AnimationTrigger.scroll is triggered when the widget is inside a Scrollable (like a ListView). Then the widget will animate according to its position along the scroll direction:
The animation progress by default is calculated as shown in the figure above (if scrolled horizontally). But you can also make the animation start/end earlier or later using two percentage offsets.
Preset Animation #
Now those MultiAnimationSequence stuff looks powerful, but also complicated to code. I’ve prepared some predefined animations for general usages. They are categorized into entrance, attention seeker, and exit. For example, one common entrance animation called SlideInAnimation is defined as:
class SlideInAnimation extends PresetAnimation {
final AxisDirection direction;
final Dimension distance;
const SlideInAnimation(
{this.distance = const Length(100, unit: LengthUnit.vmax),
this.direction = AxisDirection.up,
Duration duration = const Duration(seconds: 1),
Duration delay = Duration.zero,
Curve curve = Curves.linear,
CustomAnimationControl control = CustomAnimationControl.play})
: super(duration: duration, delay: delay, curve: curve, control: control);
...
}
copied to clipboard
You can configure the slide distance and direction, as well as duration, delay, curve, and control (whether the animation should play once or infinitely). Other predefined animations are:
FadeInAnimation
ZoomInAnimation
FadeOutAnimation
SlideOutAnimation
ZoomOutAnimation
FlipAnimation
FlashAnimation
PulseAnimation
SwingAnimation
WobbleAnimation
RainbowAnimation
ElevateAnimation
...
copied to clipboard
You can use them like this:
Widget widget = ExplicitAnimatedStyledContainer(
style: style,
child: child,
localAnimations: {
AnimationTrigger.visible: FadeInAnimation().getAnimationSequences()
}
);
copied to clipboard
Then every time the widget moves into the screen it will fade in (opacity from 0 to 1). Another feature of MultiAnimationSequence is the ability to merge or extend other MultiAnimationSequence. So you can do something like this:
Widget widget = ExplicitAnimatedStyledContainer(
style: style,
child: child,
localAnimations: {
AnimationTrigger.visible: FadeInAnimation().getAnimationSequences()..merge(
SlideInAnimation().getAnimationSequences())
}
);
copied to clipboard
Then the widget will both fade and slide in once it shows up in the viewport. If you use extend, the animation will play one after another. Preset animations make animations much easier to use while still offer you great flexibility.
Global Explicit Animations #
If we want to stagger animations across different widgets, we can do that by providing global animations. A global animation contains a map between String and MultiAnimationSequence where the String is the identifier of a widget. You provide all the global animations you want to use in an animation pool. Then you can trigger a global animation like this:
var animationPool = {
"animation1": GlobalAnimation(sequences: {
"container1" : sequence1,
"container2": sequence2,
...})
}
...
ChangeNotifierProvider<GlobalAnimationNotifier>(
create: (_) =>
GlobalAnimationNotifier(animationPool: animationPool), child: child
)
...
Widget widget = ExplicitAnimatedStyledContainer(
id: "container1",
style: style,
child: child,
globalAnimationIds: {
AnimationTrigger.visible: "animation1"
}
);
copied to clipboard
The id does not need to be unique. You can have multiple widgets with the same id so they will all animate under the same global animation. Notice if a widget does not use global animations at all, there is no need for an id.
Other things to notice #
The explicit animations uses the provider and simple_animation package.
You can programmatically change the state of the animation by calling something like the following inside the child of a ExplicitAnimatedStyledContainer:
Provider.of<LocalAnimationNotifier>(context, listen: false)
.updateAnimationStatus(animationSequence, status);
Provider.of<GlobalAnimationNotifier>(context, listen: false)
.updateAnimationStatus(animationId, status);
copied to clipboard
which will update the animation and its status (like PLAY, STOP, LOOP, etc).
You can also provide callbacks to the AnimationTrigger events along with animations:
Widget widget = ExplicitAnimatedStyledContainer(
style: style,
child: child,
localAnimations: {
AnimationTrigger.visible: animationSequence
},
onVisible: onVisible,
);
copied to clipboard
Serialization #
The style class can be easily serialized/deserialized:
String styleJson=json.encode(style.toJson());
Style newStyle=Style.fromJson(json.decode(styleJson));
copied to clipboard
Classes involved in explicit animations also support serialization. You can basically create and store complex animated components in plain text and load them everywhere.
For personal and professional use. You cannot resell or redistribute these repositories in their original state.
There are no reviews.