2015-03-05 19:27:45 +01:00
|
|
|
/*
|
|
|
|
* opsu! - an open-source osu! client
|
|
|
|
* Copyright (C) 2014, 2015 Jeffrey Han
|
|
|
|
*
|
|
|
|
* opsu! is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* opsu! is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with opsu!. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2015-05-29 07:55:57 +02:00
|
|
|
package itdelatrisu.opsu.ui;
|
2015-03-05 19:27:45 +01:00
|
|
|
|
2015-05-29 07:55:57 +02:00
|
|
|
import itdelatrisu.opsu.GameImage;
|
|
|
|
import itdelatrisu.opsu.Utils;
|
2015-03-07 21:38:59 +01:00
|
|
|
import itdelatrisu.opsu.audio.SoundController;
|
2015-05-29 09:07:58 +02:00
|
|
|
import itdelatrisu.opsu.beatmap.BeatmapParser;
|
2015-08-08 19:04:15 +02:00
|
|
|
import itdelatrisu.opsu.ui.animations.AnimatedValue;
|
2015-08-06 05:28:14 +02:00
|
|
|
import itdelatrisu.opsu.ui.animations.AnimationEquation;
|
2015-03-05 19:27:45 +01:00
|
|
|
|
|
|
|
import org.newdawn.slick.Color;
|
|
|
|
import org.newdawn.slick.Graphics;
|
|
|
|
import org.newdawn.slick.Image;
|
|
|
|
|
2017-03-26 22:57:10 +02:00
|
|
|
import static yugecin.opsudance.options.Options.*;
|
2017-12-21 22:49:31 +01:00
|
|
|
import static yugecin.opsudance.core.InstanceContainer.*;
|
2017-03-26 22:57:10 +02:00
|
|
|
|
2015-03-05 19:27:45 +01:00
|
|
|
/**
|
2015-05-29 10:48:03 +02:00
|
|
|
* Draws common UI components.
|
2015-03-05 19:27:45 +01:00
|
|
|
*/
|
|
|
|
public class UI {
|
2015-05-29 10:48:03 +02:00
|
|
|
|
2015-03-05 19:27:45 +01:00
|
|
|
/** Time to show volume image, in milliseconds. */
|
|
|
|
private static final int VOLUME_DISPLAY_TIME = 1500;
|
|
|
|
|
|
|
|
/** Volume display elapsed time. */
|
|
|
|
private static int volumeDisplay = -1;
|
|
|
|
|
2015-03-13 07:52:18 +01:00
|
|
|
/** The current tooltip. */
|
|
|
|
private static String tooltip;
|
|
|
|
|
|
|
|
/** Whether or not to check the current tooltip for line breaks. */
|
|
|
|
private static boolean tooltipNewlines;
|
|
|
|
|
2015-08-08 19:04:15 +02:00
|
|
|
/** The alpha level of the current tooltip (if any). */
|
|
|
|
private static AnimatedValue tooltipAlpha = new AnimatedValue(200, 0f, 1f, AnimationEquation.LINEAR);
|
2015-03-13 07:52:18 +01:00
|
|
|
|
2015-03-05 19:27:45 +01:00
|
|
|
// This class should not be instantiated.
|
|
|
|
private UI() {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Updates all UI components by a delta interval.
|
|
|
|
* @param delta the delta interval since the last call.
|
|
|
|
*/
|
|
|
|
public static void update(int delta) {
|
|
|
|
updateVolumeDisplay(delta);
|
2015-08-08 19:04:15 +02:00
|
|
|
tooltipAlpha.update(-delta);
|
2015-03-05 19:27:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-09-11 17:43:19 +02:00
|
|
|
* Draws the global UI components: cursor, FPS, volume bar, tooltips, bar notifications.
|
2015-03-05 19:27:45 +01:00
|
|
|
* @param g the graphics context
|
|
|
|
*/
|
|
|
|
public static void draw(Graphics g) {
|
|
|
|
drawVolume(g);
|
2015-03-09 23:32:43 +01:00
|
|
|
}
|
|
|
|
|
2015-03-05 19:27:45 +01:00
|
|
|
/**
|
|
|
|
* Resets the necessary UI components upon entering a state.
|
|
|
|
*/
|
|
|
|
public static void enter() {
|
2015-03-13 07:52:18 +01:00
|
|
|
resetTooltip();
|
2015-03-05 19:27:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Draws a tab image and text centered at a location.
|
|
|
|
* @param x the center x coordinate
|
|
|
|
* @param y the center y coordinate
|
|
|
|
* @param text the text to draw inside the tab
|
|
|
|
* @param selected whether the tab is selected (white) or not (red)
|
|
|
|
* @param isHover whether to include a hover effect (unselected only)
|
|
|
|
*/
|
|
|
|
public static void drawTab(float x, float y, String text, boolean selected, boolean isHover) {
|
|
|
|
Image tabImage = GameImage.MENU_TAB.getImage();
|
2015-08-21 03:40:07 +02:00
|
|
|
float tabTextX = x - (Fonts.MEDIUM.getWidth(text) / 2);
|
2015-05-15 07:33:53 +02:00
|
|
|
float tabTextY = y - (tabImage.getHeight() / 2);
|
2015-03-05 19:27:45 +01:00
|
|
|
Color filter, textColor;
|
|
|
|
if (selected) {
|
|
|
|
filter = Color.white;
|
|
|
|
textColor = Color.black;
|
|
|
|
} else {
|
2015-08-21 03:02:23 +02:00
|
|
|
filter = (isHover) ? Colors.RED_HOVER : Color.red;
|
2015-03-05 19:27:45 +01:00
|
|
|
textColor = Color.white;
|
|
|
|
}
|
|
|
|
tabImage.drawCentered(x, y, filter);
|
2015-08-21 03:40:07 +02:00
|
|
|
Fonts.MEDIUM.drawString(tabTextX, tabTextY, text, textColor);
|
2015-03-05 19:27:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Draws the volume bar on the middle right-hand side of the game container.
|
|
|
|
* Only draws if the volume has recently been changed using with {@link #changeVolume(int)}.
|
|
|
|
* @param g the graphics context
|
|
|
|
*/
|
|
|
|
public static void drawVolume(Graphics g) {
|
|
|
|
if (volumeDisplay == -1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
Image img = GameImage.VOLUME.getImage();
|
|
|
|
|
|
|
|
// move image in/out
|
|
|
|
float xOffset = 0;
|
|
|
|
float ratio = (float) volumeDisplay / VOLUME_DISPLAY_TIME;
|
|
|
|
if (ratio <= 0.1f)
|
|
|
|
xOffset = img.getWidth() * (1 - (ratio * 10f));
|
|
|
|
else if (ratio >= 0.9f)
|
|
|
|
xOffset = img.getWidth() * (1 - ((1 - ratio) * 10f));
|
|
|
|
|
2018-07-08 00:30:47 +02:00
|
|
|
img.drawCentered(width - img.getWidth() / 2f + xOffset, height2);
|
2015-03-05 19:27:45 +01:00
|
|
|
float barHeight = img.getHeight() * 0.9f;
|
2017-03-26 22:57:10 +02:00
|
|
|
float volume = OPTION_MASTER_VOLUME.val / 100f;
|
2015-03-05 19:27:45 +01:00
|
|
|
g.setColor(Color.white);
|
|
|
|
g.fillRoundRect(
|
2018-07-08 00:30:47 +02:00
|
|
|
width - (img.getWidth() * 0.368f) + xOffset,
|
|
|
|
height2 - (img.getHeight() * 0.47f) + (barHeight * (1 - volume)),
|
2015-03-05 19:27:45 +01:00
|
|
|
img.getWidth() * 0.15f, barHeight * volume, 3
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Updates volume display by a delta interval.
|
|
|
|
* @param delta the delta interval since the last call
|
|
|
|
*/
|
|
|
|
private static void updateVolumeDisplay(int delta) {
|
|
|
|
if (volumeDisplay == -1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
volumeDisplay += delta;
|
|
|
|
if (volumeDisplay > VOLUME_DISPLAY_TIME)
|
|
|
|
volumeDisplay = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Changes the master volume by a unit (positive or negative).
|
|
|
|
* @param units the number of units
|
|
|
|
*/
|
|
|
|
public static void changeVolume(int units) {
|
|
|
|
final float UNIT_OFFSET = 0.05f;
|
2017-03-26 22:57:10 +02:00
|
|
|
float volume = Utils.clamp(OPTION_MASTER_VOLUME.val / 100f + (UNIT_OFFSET * units), 0f, 1f);
|
|
|
|
OPTION_MASTER_VOLUME.setValue((int) (volume * 100f));
|
2015-03-05 19:27:45 +01:00
|
|
|
if (volumeDisplay == -1)
|
|
|
|
volumeDisplay = 0;
|
|
|
|
else if (volumeDisplay >= VOLUME_DISPLAY_TIME / 10)
|
|
|
|
volumeDisplay = VOLUME_DISPLAY_TIME / 10;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-09-10 05:51:16 +02:00
|
|
|
* Draws loading progress (OSZ unpacking, beatmap parsing, replay importing, sound loading)
|
2015-03-05 19:27:45 +01:00
|
|
|
* at the bottom of the screen.
|
2015-09-10 05:51:16 +02:00
|
|
|
* @param g the graphics context
|
2015-03-05 19:27:45 +01:00
|
|
|
*/
|
|
|
|
public static void drawLoadingProgress(Graphics g) {
|
|
|
|
String text, file;
|
|
|
|
int progress;
|
|
|
|
|
|
|
|
// determine current action
|
2017-12-21 22:49:31 +01:00
|
|
|
if ((file = oszunpacker.getCurrentFileName()) != null) {
|
2015-03-05 19:27:45 +01:00
|
|
|
text = "Unpacking new beatmaps...";
|
2017-12-21 22:49:31 +01:00
|
|
|
progress = oszunpacker.getUnpackerProgress();
|
|
|
|
} else if ((file = beatmapParser.getCurrentFileName()) != null) {
|
|
|
|
text = (beatmapParser.getStatus() == BeatmapParser.Status.INSERTING) ?
|
2015-03-08 05:24:19 +01:00
|
|
|
"Updating database..." : "Loading beatmaps...";
|
2017-12-21 22:49:31 +01:00
|
|
|
progress = beatmapParser.getParserProgress();
|
|
|
|
} else if ((file = replayImporter.getCurrentFileName()) != null) {
|
2015-06-30 02:22:38 +02:00
|
|
|
text = "Importing replays...";
|
2017-12-21 22:49:31 +01:00
|
|
|
progress = replayImporter.getLoadingProgress();
|
2015-03-05 19:27:45 +01:00
|
|
|
} else if ((file = SoundController.getCurrentFileName()) != null) {
|
|
|
|
text = "Loading sounds...";
|
|
|
|
progress = SoundController.getLoadingProgress();
|
|
|
|
} else
|
|
|
|
return;
|
|
|
|
|
|
|
|
// draw loading info
|
2018-07-08 00:30:47 +02:00
|
|
|
float marginX = width * 0.02f, marginY = height * 0.02f;
|
|
|
|
float lineY = height - marginY;
|
2015-08-21 03:40:07 +02:00
|
|
|
int lineOffsetY = Fonts.MEDIUM.getLineHeight();
|
2017-03-26 22:57:10 +02:00
|
|
|
if (OPTION_LOAD_VERBOSE.state) {
|
2015-03-05 19:27:45 +01:00
|
|
|
// verbose: display percentages and file names
|
2015-08-21 03:40:07 +02:00
|
|
|
Fonts.MEDIUM.drawString(
|
2015-03-05 19:27:45 +01:00
|
|
|
marginX, lineY - (lineOffsetY * 2),
|
|
|
|
String.format("%s (%d%%)", text, progress), Color.white);
|
2015-08-21 03:40:07 +02:00
|
|
|
Fonts.MEDIUM.drawString(marginX, lineY - lineOffsetY, file, Color.white);
|
2015-03-05 19:27:45 +01:00
|
|
|
} else {
|
|
|
|
// draw loading bar
|
2015-08-21 03:40:07 +02:00
|
|
|
Fonts.MEDIUM.drawString(marginX, lineY - (lineOffsetY * 2), text, Color.white);
|
2015-03-05 19:27:45 +01:00
|
|
|
g.setColor(Color.white);
|
|
|
|
g.fillRoundRect(marginX, lineY - (lineOffsetY / 2f),
|
2018-07-08 00:30:47 +02:00
|
|
|
(width - (marginX * 2f)) * progress / 100f, lineOffsetY / 4f, 4
|
2015-03-05 19:27:45 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Draws a scroll bar.
|
|
|
|
* @param g the graphics context
|
2015-07-02 01:45:13 +02:00
|
|
|
* @param position the position in the virtual area
|
|
|
|
* @param totalLength the total length of the virtual area
|
|
|
|
* @param lengthShown the length of the virtual area shown
|
|
|
|
* @param unitBaseX the base x coordinate
|
|
|
|
* @param unitBaseY the base y coordinate
|
2015-03-05 19:27:45 +01:00
|
|
|
* @param unitWidth the width of a unit
|
2015-07-02 01:45:13 +02:00
|
|
|
* @param scrollAreaHeight the height of the scroll area
|
2015-03-05 19:27:45 +01:00
|
|
|
* @param bgColor the scroll bar area background color (null if none)
|
|
|
|
* @param scrollbarColor the scroll bar color
|
|
|
|
* @param right whether or not to place the scroll bar on the right side of the unit
|
|
|
|
*/
|
|
|
|
public static void drawScrollbar(
|
2015-07-02 01:45:13 +02:00
|
|
|
Graphics g, float position, float totalLength, float lengthShown,
|
|
|
|
float unitBaseX, float unitBaseY, float unitWidth, float scrollAreaHeight,
|
2015-03-05 19:27:45 +01:00
|
|
|
Color bgColor, Color scrollbarColor, boolean right
|
|
|
|
) {
|
2018-07-08 00:30:47 +02:00
|
|
|
float scrollbarWidth = width * 0.00347f;
|
2015-07-02 01:45:13 +02:00
|
|
|
float scrollbarHeight = scrollAreaHeight * lengthShown / totalLength;
|
|
|
|
float offsetY = (scrollAreaHeight - scrollbarHeight) * (position / (totalLength - lengthShown));
|
2015-03-05 19:27:45 +01:00
|
|
|
float scrollbarX = unitBaseX + unitWidth - ((right) ? scrollbarWidth : 0);
|
|
|
|
if (bgColor != null) {
|
|
|
|
g.setColor(bgColor);
|
|
|
|
g.fillRect(scrollbarX, unitBaseY, scrollbarWidth, scrollAreaHeight);
|
|
|
|
}
|
|
|
|
g.setColor(scrollbarColor);
|
|
|
|
g.fillRect(scrollbarX, unitBaseY + offsetY, scrollbarWidth, scrollbarHeight);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-13 07:52:18 +01:00
|
|
|
* Sets or updates a tooltip for drawing.
|
|
|
|
* Must be called with {@link #drawTooltip(Graphics)}.
|
|
|
|
* @param delta the delta interval since the last call
|
|
|
|
* @param s the tooltip text
|
2015-03-05 19:27:45 +01:00
|
|
|
* @param newlines whether to check for line breaks ('\n')
|
|
|
|
*/
|
2015-03-13 07:52:18 +01:00
|
|
|
public static void updateTooltip(int delta, String s, boolean newlines) {
|
|
|
|
if (s != null) {
|
|
|
|
tooltip = s;
|
|
|
|
tooltipNewlines = newlines;
|
2015-08-08 19:04:15 +02:00
|
|
|
tooltipAlpha.update(delta * 2);
|
2015-03-13 07:52:18 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Draws a tooltip, if any, near the current mouse coordinates,
|
|
|
|
* bounded by the container dimensions.
|
|
|
|
* @param g the graphics context
|
|
|
|
*/
|
|
|
|
public static void drawTooltip(Graphics g) {
|
2015-08-08 19:04:15 +02:00
|
|
|
if (tooltipAlpha.getTime() == 0 || tooltip == null)
|
2015-03-13 07:52:18 +01:00
|
|
|
return;
|
|
|
|
|
2018-07-08 00:30:47 +02:00
|
|
|
int margin = width / 100, textMarginX = 2;
|
2018-07-08 09:52:03 +02:00
|
|
|
int offset = GameImage.CURSOR_MIDDLE.getWidth() / 2;
|
2015-08-21 03:40:07 +02:00
|
|
|
int lineHeight = Fonts.SMALL.getLineHeight();
|
2015-03-05 19:27:45 +01:00
|
|
|
int textWidth = textMarginX * 2, textHeight = lineHeight;
|
2015-03-13 07:52:18 +01:00
|
|
|
if (tooltipNewlines) {
|
|
|
|
String[] lines = tooltip.split("\\n");
|
2015-08-21 03:40:07 +02:00
|
|
|
int maxWidth = Fonts.SMALL.getWidth(lines[0]);
|
2015-03-05 19:27:45 +01:00
|
|
|
for (int i = 1; i < lines.length; i++) {
|
2015-08-21 03:40:07 +02:00
|
|
|
int w = Fonts.SMALL.getWidth(lines[i]);
|
2015-03-05 19:27:45 +01:00
|
|
|
if (w > maxWidth)
|
|
|
|
maxWidth = w;
|
|
|
|
}
|
|
|
|
textWidth += maxWidth;
|
|
|
|
textHeight += lineHeight * (lines.length - 1);
|
|
|
|
} else
|
2015-08-21 03:40:07 +02:00
|
|
|
textWidth += Fonts.SMALL.getWidth(tooltip);
|
2015-03-05 19:27:45 +01:00
|
|
|
|
|
|
|
// get drawing coordinates
|
2018-07-08 00:47:35 +02:00
|
|
|
int x = mouseX + offset;
|
|
|
|
int y = mouseY + offset;
|
2018-07-08 00:30:47 +02:00
|
|
|
if (x + textWidth > width - margin)
|
|
|
|
x = width - margin - textWidth;
|
2015-03-05 19:27:45 +01:00
|
|
|
else if (x < margin)
|
|
|
|
x = margin;
|
2018-07-08 00:30:47 +02:00
|
|
|
if (y + textHeight > height - margin)
|
|
|
|
y = height - margin - textHeight;
|
2015-03-05 19:27:45 +01:00
|
|
|
else if (y < margin)
|
|
|
|
y = margin;
|
|
|
|
|
|
|
|
// draw tooltip text inside a filled rectangle
|
2015-08-08 19:04:15 +02:00
|
|
|
float alpha = tooltipAlpha.getValue();
|
2015-08-21 03:02:23 +02:00
|
|
|
float oldAlpha = Colors.BLACK_ALPHA.a;
|
|
|
|
Colors.BLACK_ALPHA.a = alpha;
|
|
|
|
g.setColor(Colors.BLACK_ALPHA);
|
|
|
|
Colors.BLACK_ALPHA.a = oldAlpha;
|
2015-03-05 19:27:45 +01:00
|
|
|
g.fillRect(x, y, textWidth, textHeight);
|
2015-08-21 03:02:23 +02:00
|
|
|
oldAlpha = Colors.DARK_GRAY.a;
|
|
|
|
Colors.DARK_GRAY.a = alpha;
|
|
|
|
g.setColor(Colors.DARK_GRAY);
|
2015-03-05 19:27:45 +01:00
|
|
|
g.setLineWidth(1);
|
|
|
|
g.drawRect(x, y, textWidth, textHeight);
|
2015-08-21 03:02:23 +02:00
|
|
|
Colors.DARK_GRAY.a = oldAlpha;
|
|
|
|
oldAlpha = Colors.WHITE_ALPHA.a;
|
|
|
|
Colors.WHITE_ALPHA.a = alpha;
|
2015-08-21 03:40:07 +02:00
|
|
|
Fonts.SMALL.drawString(x + textMarginX, y, tooltip, Colors.WHITE_ALPHA);
|
2015-08-21 03:02:23 +02:00
|
|
|
Colors.WHITE_ALPHA.a = oldAlpha;
|
2015-03-13 07:52:18 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Resets the tooltip.
|
|
|
|
*/
|
|
|
|
public static void resetTooltip() {
|
2015-08-08 19:04:15 +02:00
|
|
|
tooltipAlpha.setTime(0);
|
2015-03-13 07:52:18 +01:00
|
|
|
tooltip = null;
|
2015-03-05 19:27:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|