blob: bf1b6c43b39be42ad598e3aa11831a3c9202eff4 [file] [log] [blame]
package com.airbnb.lottie;
import android.graphics.PointF;
import android.support.annotation.FloatRange;
import android.support.annotation.Nullable;
import android.support.v4.view.animation.PathInterpolatorCompat;
import android.view.animation.Interpolator;
import android.view.animation.LinearInterpolator;
import org.json.JSONArray;
import org.json.JSONObject;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
public class Keyframe<T> {
private static final Interpolator LINEAR_INTERPOLATOR = new LinearInterpolator();
static <T> List<Keyframe<T>> parseKeyframes(JSONArray json, LottieComposition composition,
float scale, AnimatableValue<T, ?> animatableValue) {
if (json.length() == 0) {
return Collections.emptyList();
}
List<Keyframe<T>> keyframes = new ArrayList<>();
for (int i = 0; i < json.length(); i++) {
keyframes.add(new Keyframe<>(json.optJSONObject(i), composition, scale, animatableValue));
}
setEndFrames(keyframes);
return keyframes;
}
/**
* The json doesn't include end frames. The data can be taken from the start frame of the next
* keyframe though.
*/
static void setEndFrames(List<? extends Keyframe<?>> keyframes) {
for (int i = 0; i < keyframes.size() - 1; i++) {
// In the json, the keyframes only contain their starting frame.
keyframes.get(i).endFrame = keyframes.get(i + 1).startFrame;
}
Keyframe<?> lastKeyframe = keyframes.get(keyframes.size() - 1);
if (lastKeyframe.startValue == null) {
// The only purpose the last keyframe has is to provide the end frame of the previous
// keyframe.
//noinspection SuspiciousMethodCalls
keyframes.remove(lastKeyframe);
}
}
private final LottieComposition composition;
T startValue;
T endValue;
Interpolator interpolator;
@SuppressWarnings("WeakerAccess") float startFrame;
@SuppressWarnings("WeakerAccess") @Nullable Float endFrame = null;
public Keyframe(LottieComposition composition, float startFrame, float endFrame) {
this.composition = composition;
this.startFrame = startFrame;
this.endFrame = endFrame;
}
Keyframe(JSONObject json, LottieComposition composition, float scale,
AnimatableValue<T, ?> animatableValue) {
this.composition = composition;
PointF cp1 = null;
PointF cp2 = null;
if (json.has("t")) {
startFrame = (float) json.optDouble("t", 0);
Object startValueJson = json.opt("s");
if (startValueJson != null) {
startValue = animatableValue.valueFromObject(startValueJson, scale);
}
Object endValueJson = json.opt("e");
if (endValueJson != null) {
endValue = animatableValue.valueFromObject(endValueJson, scale);
}
JSONObject cp1Json = json.optJSONObject("o");
JSONObject cp2Json = json.optJSONObject("i");
if (cp1Json != null && cp2Json != null) {
cp1 = JsonUtils.pointFromJsonObject(cp1Json, scale);
cp2 = JsonUtils.pointFromJsonObject(cp2Json, scale);
}
boolean hold = json.optInt("h", 0) == 1;
if (hold) {
endValue = startValue;
// TODO: create a HoldInterpolator so progress changes don't invalidate.
interpolator = LINEAR_INTERPOLATOR;
} else if (cp1 != null) {
interpolator = PathInterpolatorCompat.create(
cp1.x / scale, cp1.y / scale, cp2.x / scale, cp2.y / scale);
} else {
interpolator = LINEAR_INTERPOLATOR;
}
} else {
startValue = animatableValue.valueFromObject(json, scale);
endValue = startValue;
}
}
@FloatRange(from = 0f, to = 1f)
float getStartProgress() {
return startFrame / composition.getDurationFrames();
}
@FloatRange(from = 0f, to = 1f)
float getEndProgress() {
//noinspection Range
return endFrame == null ? 1f : endFrame / composition.getDurationFrames();
}
boolean isStatic() {
return interpolator == null;
}
boolean containsProgress(@FloatRange(from = 0f, to = 1f) float progress) {
return progress >= getStartProgress() && progress <= getEndProgress();
}
@Override public String toString() {
return "Keyframe{" + "startValue=" + startValue +
", endValue=" + endValue +
", startFrame=" + startFrame +
", endFrame=" + endFrame +
", interpolator=" + interpolator +
'}';
}
}