Files
flame/lib/effects/combined_effect.dart
2020-10-06 15:24:11 +02:00

103 lines
2.9 KiB
Dart

import 'dart:math';
import 'package:meta/meta.dart';
import './effects.dart';
import '../components/position_component.dart';
class CombinedEffect extends PositionComponentEffect {
final List<PositionComponentEffect> effects;
final double offset;
CombinedEffect({
@required this.effects,
this.offset = 0.0,
bool isInfinite = false,
bool isAlternating = false,
Function onComplete,
}) : super(isInfinite, isAlternating, onComplete: onComplete) {
assert(
effects.every((effect) => effect.component == null),
"No effects can be added to components from the start",
);
final types = effects.map((e) => e.runtimeType);
assert(
types.toSet().length == types.length,
"All effect types have to be different so that they don't clash",
);
}
@override
void initialize(PositionComponent _comp) {
super.initialize(_comp);
effects.forEach((effect) {
effect.initialize(_comp);
final isSameSize = effect.endSize == _comp.size;
final isSamePosition = effect.endPosition == _comp.position;
final isSameAngle = effect.endAngle == _comp.angle;
endSize = isSameSize ? endSize : effect.endSize;
endPosition = isSamePosition ? endPosition : effect.endPosition;
endAngle = isSameAngle ? endAngle : effect.endAngle;
travelTime = max(travelTime ?? 0,
effect.totalTravelTime + offset * effects.indexOf(effect));
});
}
@override
void update(double dt) {
if (hasFinished()) {
return;
}
super.update(dt);
effects.forEach((effect) => _updateEffect(effect, dt));
if (effects.every((effect) => effect.hasFinished())) {
if (isAlternating && curveDirection.isNegative) {
effects.forEach((effect) => effect.isAlternating = true);
} else if (isInfinite) {
reset();
} else if (isAlternating && isMin()) {
dispose();
}
}
}
@override
void reset() {
super.reset();
effects.forEach((effect) => effect.reset());
if (component != null) {
initialize(component);
}
}
@override
void dispose() {
super.dispose();
effects.forEach((effect) => effect.dispose());
}
void _updateEffect(final effect, double dt) {
final isReverse = curveDirection.isNegative;
final initialOffset = effects.indexOf(effect) * offset;
final effectOffset = isReverse
? travelTime - effect.travelTime - initialOffset
: initialOffset;
final passedOffset = isReverse ? travelTime - currentTime : currentTime;
if (!effect.hasFinished() && effectOffset < passedOffset) {
final time =
effectOffset < passedOffset - dt ? dt : passedOffset - effectOffset;
effect.update(time);
}
if (isMax()) {
_maybeReverse(effect);
}
}
void _maybeReverse(PositionComponentEffect effect) {
if (isAlternating && !effect.isAlternating && effect.isMax()) {
// Make the effect go in reverse
effect.isAlternating = true;
}
}
}