Hit animation bug fixes.
- In "Hidden" mod, don't draw expanding animation for circles. (related to #121) - With "Show Perfect Hits" disabled, still show the hit animations instead of having the objects just disappear. (also mentioned in #121) - Removed the two easing functions in Utils (and replaced references with calls to AnimationEquations). Signed-off-by: Jeffrey Han <itdelatrisu@gmail.com>
This commit is contained in:
parent
8efcc1f2ee
commit
6e7de654b0
|
@ -28,6 +28,7 @@ import itdelatrisu.opsu.downloads.Updater;
|
||||||
import itdelatrisu.opsu.objects.curves.Curve;
|
import itdelatrisu.opsu.objects.curves.Curve;
|
||||||
import itdelatrisu.opsu.replay.Replay;
|
import itdelatrisu.opsu.replay.Replay;
|
||||||
import itdelatrisu.opsu.replay.ReplayFrame;
|
import itdelatrisu.opsu.replay.ReplayFrame;
|
||||||
|
import itdelatrisu.opsu.ui.animations.AnimationEquation;
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
|
@ -255,6 +256,9 @@ public class GameData {
|
||||||
/** Whether or not to expand when animating. */
|
/** Whether or not to expand when animating. */
|
||||||
public boolean expand;
|
public boolean expand;
|
||||||
|
|
||||||
|
/** Whether or not to hide the hit result. */
|
||||||
|
public boolean hideResult;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor.
|
* Constructor.
|
||||||
* @param time the result's starting track position
|
* @param time the result's starting track position
|
||||||
|
@ -264,9 +268,10 @@ public class GameData {
|
||||||
* @param color the color of the hit object
|
* @param color the color of the hit object
|
||||||
* @param curve the slider curve (or null if not applicable)
|
* @param curve the slider curve (or null if not applicable)
|
||||||
* @param expand whether or not the hit result animation should expand (if applicable)
|
* @param expand whether or not the hit result animation should expand (if applicable)
|
||||||
|
* @param hideResult whether or not to hide the hit result (but still show the other animations)
|
||||||
*/
|
*/
|
||||||
public HitObjectResult(int time, int result, float x, float y, Color color,
|
public HitObjectResult(int time, int result, float x, float y, Color color,
|
||||||
HitObjectType hitResultType, Curve curve, boolean expand) {
|
HitObjectType hitResultType, Curve curve, boolean expand, boolean hideResult) {
|
||||||
this.time = time;
|
this.time = time;
|
||||||
this.result = result;
|
this.result = result;
|
||||||
this.x = x;
|
this.x = x;
|
||||||
|
@ -275,6 +280,7 @@ public class GameData {
|
||||||
this.hitResultType = hitResultType;
|
this.hitResultType = hitResultType;
|
||||||
this.curve = curve;
|
this.curve = curve;
|
||||||
this.expand = expand;
|
this.expand = expand;
|
||||||
|
this.hideResult = hideResult;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -872,7 +878,7 @@ public class GameData {
|
||||||
}
|
}
|
||||||
|
|
||||||
// hit lighting
|
// hit lighting
|
||||||
else if (Options.isHitLightingEnabled() && hitResult.result != HIT_MISS &&
|
else if (Options.isHitLightingEnabled() && !hitResult.hideResult && hitResult.result != HIT_MISS &&
|
||||||
hitResult.result != HIT_SLIDER30 && hitResult.result != HIT_SLIDER10) {
|
hitResult.result != HIT_SLIDER30 && hitResult.result != HIT_SLIDER10) {
|
||||||
// TODO: add particle system
|
// TODO: add particle system
|
||||||
Image lighting = GameImage.LIGHTING.getImage();
|
Image lighting = GameImage.LIGHTING.getImage();
|
||||||
|
@ -885,14 +891,10 @@ public class GameData {
|
||||||
hitResult.hitResultType == HitObjectType.CIRCLE ||
|
hitResult.hitResultType == HitObjectType.CIRCLE ||
|
||||||
hitResult.hitResultType == HitObjectType.SLIDER_FIRST ||
|
hitResult.hitResultType == HitObjectType.SLIDER_FIRST ||
|
||||||
hitResult.hitResultType == HitObjectType.SLIDER_LAST)) {
|
hitResult.hitResultType == HitObjectType.SLIDER_LAST)) {
|
||||||
float scale = (!hitResult.expand) ? 1f : Utils.easeOut(
|
float progress = AnimationEquation.OUT_CUBIC.calc(
|
||||||
Utils.clamp(trackPosition - hitResult.time, 0, HITCIRCLE_FADE_TIME),
|
(float) Utils.clamp(trackPosition - hitResult.time, 0, HITCIRCLE_FADE_TIME) / HITCIRCLE_FADE_TIME);
|
||||||
1f, HITCIRCLE_ANIM_SCALE - 1f, HITCIRCLE_FADE_TIME
|
float scale = (!hitResult.expand) ? 1f : 1f + (HITCIRCLE_ANIM_SCALE - 1f) * progress;
|
||||||
);
|
float alpha = 1f - progress;
|
||||||
float alpha = Utils.easeOut(
|
|
||||||
Utils.clamp(trackPosition - hitResult.time, 0, HITCIRCLE_FADE_TIME),
|
|
||||||
1f, -1f, HITCIRCLE_FADE_TIME
|
|
||||||
);
|
|
||||||
|
|
||||||
// slider curve
|
// slider curve
|
||||||
if (hitResult.curve != null) {
|
if (hitResult.curve != null) {
|
||||||
|
@ -906,6 +908,8 @@ public class GameData {
|
||||||
}
|
}
|
||||||
|
|
||||||
// hit circles
|
// hit circles
|
||||||
|
if (!(hitResult.hitResultType == HitObjectType.CIRCLE && GameMod.HIDDEN.isActive())) {
|
||||||
|
// "hidden" mod: expanding animation for only circles not drawn
|
||||||
Image scaledHitCircle = GameImage.HITCIRCLE.getImage().getScaledCopy(scale);
|
Image scaledHitCircle = GameImage.HITCIRCLE.getImage().getScaledCopy(scale);
|
||||||
Image scaledHitCircleOverlay = GameImage.HITCIRCLE_OVERLAY.getImage().getScaledCopy(scale);
|
Image scaledHitCircleOverlay = GameImage.HITCIRCLE_OVERLAY.getImage().getScaledCopy(scale);
|
||||||
scaledHitCircle.setAlpha(alpha);
|
scaledHitCircle.setAlpha(alpha);
|
||||||
|
@ -913,19 +917,19 @@ public class GameData {
|
||||||
scaledHitCircle.drawCentered(hitResult.x, hitResult.y, hitResult.color);
|
scaledHitCircle.drawCentered(hitResult.x, hitResult.y, hitResult.color);
|
||||||
scaledHitCircleOverlay.drawCentered(hitResult.x, hitResult.y);
|
scaledHitCircleOverlay.drawCentered(hitResult.x, hitResult.y);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// hit result
|
// hit result
|
||||||
if (hitResult.hitResultType == HitObjectType.CIRCLE ||
|
if (!hitResult.hideResult && (
|
||||||
|
hitResult.hitResultType == HitObjectType.CIRCLE ||
|
||||||
hitResult.hitResultType == HitObjectType.SPINNER ||
|
hitResult.hitResultType == HitObjectType.SPINNER ||
|
||||||
hitResult.curve != null) {
|
hitResult.curve != null)) {
|
||||||
float scale = Utils.easeBounce(
|
float scaleProgress = AnimationEquation.IN_OUT_BOUNCE.calc(
|
||||||
Utils.clamp(trackPosition - hitResult.time, 0, HITCIRCLE_TEXT_BOUNCE_TIME),
|
(float) Utils.clamp(trackPosition - hitResult.time, 0, HITCIRCLE_TEXT_BOUNCE_TIME) / HITCIRCLE_TEXT_BOUNCE_TIME);
|
||||||
1f, HITCIRCLE_TEXT_ANIM_SCALE - 1f, HITCIRCLE_TEXT_BOUNCE_TIME
|
float scale = 1f + (HITCIRCLE_TEXT_ANIM_SCALE - 1f) * scaleProgress;
|
||||||
);
|
float fadeProgress = AnimationEquation.OUT_CUBIC.calc(
|
||||||
float alpha = Utils.easeOut(
|
(float) Utils.clamp((trackPosition - hitResult.time) - HITCIRCLE_FADE_TIME, 0, HITCIRCLE_TEXT_FADE_TIME) / HITCIRCLE_TEXT_FADE_TIME);
|
||||||
Utils.clamp((trackPosition - hitResult.time) - HITCIRCLE_FADE_TIME, 0, HITCIRCLE_TEXT_FADE_TIME),
|
float alpha = 1f - fadeProgress;
|
||||||
1f, -1f, HITCIRCLE_TEXT_FADE_TIME
|
|
||||||
);
|
|
||||||
Image scaledHitResult = hitResults[hitResult.result].getScaledCopy(scale);
|
Image scaledHitResult = hitResults[hitResult.result].getScaledCopy(scale);
|
||||||
scaledHitResult.setAlpha(alpha);
|
scaledHitResult.setAlpha(alpha);
|
||||||
scaledHitResult.drawCentered(hitResult.x, hitResult.y);
|
scaledHitResult.drawCentered(hitResult.x, hitResult.y);
|
||||||
|
@ -1207,7 +1211,7 @@ public class GameData {
|
||||||
if (!Options.isPerfectHitBurstEnabled())
|
if (!Options.isPerfectHitBurstEnabled())
|
||||||
; // hide perfect hit results
|
; // hide perfect hit results
|
||||||
else
|
else
|
||||||
hitResultList.add(new HitObjectResult(time, result, x, y, null, HitObjectType.SLIDERTICK, null, false));
|
hitResultList.add(new HitObjectResult(time, result, x, y, null, HitObjectType.SLIDERTICK, null, false, false));
|
||||||
}
|
}
|
||||||
fullObjectCount++;
|
fullObjectCount++;
|
||||||
}
|
}
|
||||||
|
@ -1363,20 +1367,18 @@ public class GameData {
|
||||||
int hitResult = handleHitResult(time, result, x, y, color, end, hitObject,
|
int hitResult = handleHitResult(time, result, x, y, color, end, hitObject,
|
||||||
hitResultType, repeat, (curve != null && !sliderHeldToEnd));
|
hitResultType, repeat, (curve != null && !sliderHeldToEnd));
|
||||||
|
|
||||||
if ((hitResult == HIT_300 || hitResult == HIT_300G || hitResult == HIT_300K) && !Options.isPerfectHitBurstEnabled())
|
if (hitResult == HIT_MISS && (GameMod.RELAX.isActive() || GameMod.AUTOPILOT.isActive()))
|
||||||
; // hide perfect hit results
|
return; // "relax" and "autopilot" mods: hide misses
|
||||||
else if (hitResult == HIT_MISS && (GameMod.RELAX.isActive() || GameMod.AUTOPILOT.isActive()))
|
|
||||||
; // "relax" and "autopilot" mods: hide misses
|
boolean hideResult = (hitResult == HIT_300 || hitResult == HIT_300G || hitResult == HIT_300K) && !Options.isPerfectHitBurstEnabled();
|
||||||
else {
|
hitResultList.add(new HitObjectResult(time, hitResult, x, y, color, hitResultType, curve, expand, hideResult));
|
||||||
hitResultList.add(new HitObjectResult(time, hitResult, x, y, color, hitResultType, curve, expand));
|
|
||||||
|
|
||||||
// sliders: add the other curve endpoint for the hit animation
|
// sliders: add the other curve endpoint for the hit animation
|
||||||
if (curve != null) {
|
if (curve != null) {
|
||||||
boolean isFirst = (hitResultType == HitObjectType.SLIDER_FIRST);
|
boolean isFirst = (hitResultType == HitObjectType.SLIDER_FIRST);
|
||||||
float[] p = curve.pointAt((isFirst) ? 1f : 0f);
|
float[] p = curve.pointAt((isFirst) ? 1f : 0f);
|
||||||
HitObjectType type = (isFirst) ? HitObjectType.SLIDER_LAST : HitObjectType.SLIDER_FIRST;
|
HitObjectType type = (isFirst) ? HitObjectType.SLIDER_LAST : HitObjectType.SLIDER_FIRST;
|
||||||
hitResultList.add(new HitObjectResult(time, hitResult, p[0], p[1], color, type, null, expand));
|
hitResultList.add(new HitObjectResult(time, hitResult, p[0], p[1], color, type, null, expand, hideResult));
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -654,32 +654,6 @@ public class Utils {
|
||||||
return String.format("%02d:%02d:%02d", seconds / 3600, (seconds / 60) % 60, seconds % 60);
|
return String.format("%02d:%02d:%02d", seconds / 3600, (seconds / 60) % 60, seconds % 60);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Cubic ease out function.
|
|
||||||
* @param t the current time
|
|
||||||
* @param a the starting position
|
|
||||||
* @param b the finishing position
|
|
||||||
* @param d the duration
|
|
||||||
* @return the eased float
|
|
||||||
*/
|
|
||||||
public static float easeOut(float t, float a, float b, float d) {
|
|
||||||
return b * ((t = t / d - 1f) * t * t + 1f) + a;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Fake bounce ease function.
|
|
||||||
* @param t the current time
|
|
||||||
* @param a the starting position
|
|
||||||
* @param b the finishing position
|
|
||||||
* @param d the duration
|
|
||||||
* @return the eased float
|
|
||||||
*/
|
|
||||||
public static float easeBounce(float t, float a, float b, float d) {
|
|
||||||
if (t < d / 2)
|
|
||||||
return easeOut(t, a, b, d);
|
|
||||||
return easeOut(d - t, a, b, d);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns whether or not the application is running within a JAR.
|
* Returns whether or not the application is running within a JAR.
|
||||||
* @return true if JAR, false if file
|
* @return true if JAR, false if file
|
||||||
|
|
|
@ -18,7 +18,6 @@
|
||||||
|
|
||||||
package itdelatrisu.opsu.ui.animations;
|
package itdelatrisu.opsu.ui.animations;
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* These equations are copyright (c) 2001 Robert Penner, all rights reserved,
|
* These equations are copyright (c) 2001 Robert Penner, all rights reserved,
|
||||||
* and are open source under the BSD License.
|
* and are open source under the BSD License.
|
||||||
|
|
Loading…
Reference in New Issue
Block a user