client: update splashscreen
This commit is contained in:
@@ -80,13 +80,13 @@ import org.slf4j.LoggerFactory;
|
|||||||
@Slf4j
|
@Slf4j
|
||||||
public class RuneLite
|
public class RuneLite
|
||||||
{
|
{
|
||||||
public static final String PLUS_VERSION = "2.1.1.0";
|
|
||||||
public static final File RUNELITE_DIR = new File(System.getProperty("user.home"), ".runelite");
|
public static final File RUNELITE_DIR = new File(System.getProperty("user.home"), ".runelite");
|
||||||
public static final File PROFILES_DIR = new File(RUNELITE_DIR, "profiles");
|
public static final File PROFILES_DIR = new File(RUNELITE_DIR, "profiles");
|
||||||
public static final File PLUGIN_DIR = new File(RUNELITE_DIR, "plugins");
|
public static final File PLUGIN_DIR = new File(RUNELITE_DIR, "plugins");
|
||||||
public static final File SCREENSHOT_DIR = new File(RUNELITE_DIR, "screenshots");
|
public static final File SCREENSHOT_DIR = new File(RUNELITE_DIR, "screenshots");
|
||||||
public static final File LOGS_DIR = new File(RUNELITE_DIR, "logs");
|
public static final File LOGS_DIR = new File(RUNELITE_DIR, "logs");
|
||||||
private static final RuneLiteSplashScreen splashScreen = new RuneLiteSplashScreen();
|
private static final File LOG_FILE = new File(LOGS_DIR, "client.log");
|
||||||
|
private static final RuneLiteProperties PROPERTIES = new RuneLiteProperties();
|
||||||
public static boolean allowPrivateServer = false;
|
public static boolean allowPrivateServer = false;
|
||||||
|
|
||||||
@Getter
|
@Getter
|
||||||
@@ -260,6 +260,13 @@ public class RuneLite
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!options.has("no-splash"))
|
||||||
|
{
|
||||||
|
RuneLiteSplashScreen.init();
|
||||||
|
}
|
||||||
|
|
||||||
|
RuneLiteSplashScreen.stage(0, "Initializing client");
|
||||||
|
|
||||||
PROFILES_DIR.mkdirs();
|
PROFILES_DIR.mkdirs();
|
||||||
|
|
||||||
if (options.has("debug"))
|
if (options.has("debug"))
|
||||||
@@ -277,14 +284,8 @@ public class RuneLite
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!options.has("no-splash"))
|
|
||||||
{
|
|
||||||
splashScreen.open(4);
|
|
||||||
}
|
|
||||||
|
|
||||||
// The submessage is shown in case the connection is slow
|
RuneLiteSplashScreen.stage(.2, "Starting RuneLitePlus injector");
|
||||||
splashScreen.setMessage("Starting RuneLite Injector");
|
|
||||||
splashScreen.setSubMessage(" ");
|
|
||||||
|
|
||||||
final long start = System.currentTimeMillis();
|
final long start = System.currentTimeMillis();
|
||||||
|
|
||||||
@@ -293,7 +294,6 @@ public class RuneLite
|
|||||||
true));
|
true));
|
||||||
|
|
||||||
injector.getInstance(RuneLite.class).start();
|
injector.getInstance(RuneLite.class).start();
|
||||||
splashScreen.setProgress(1, 5);
|
|
||||||
final long end = System.currentTimeMillis();
|
final long end = System.currentTimeMillis();
|
||||||
final RuntimeMXBean rb = ManagementFactory.getRuntimeMXBean();
|
final RuntimeMXBean rb = ManagementFactory.getRuntimeMXBean();
|
||||||
final long uptime = rb.getUptime();
|
final long uptime = rb.getUptime();
|
||||||
@@ -312,14 +312,13 @@ public class RuneLite
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Load user configuration
|
// Load user configuration
|
||||||
splashScreen.setMessage("Loading configuration");
|
|
||||||
|
RuneLiteSplashScreen.stage(.57, "Loading user config");
|
||||||
configManager.load();
|
configManager.load();
|
||||||
|
|
||||||
// Load the session, including saved configuration
|
// Load the session, including saved configuration
|
||||||
sessionManager.loadSession();
|
sessionManager.loadSession();
|
||||||
splashScreen.setProgress(2, 5);
|
RuneLiteSplashScreen.stage(.58, "Loading session data");
|
||||||
|
|
||||||
splashScreen.setMessage("Loading plugins");
|
|
||||||
|
|
||||||
// Begin watching for new plugins
|
// Begin watching for new plugins
|
||||||
pluginManager.watch();
|
pluginManager.watch();
|
||||||
@@ -330,20 +329,15 @@ public class RuneLite
|
|||||||
// Load the plugins, but does not start them yet.
|
// Load the plugins, but does not start them yet.
|
||||||
// This will initialize configuration
|
// This will initialize configuration
|
||||||
pluginManager.loadCorePlugins();
|
pluginManager.loadCorePlugins();
|
||||||
|
RuneLiteSplashScreen.stage(.70, "Finalizing configuration");
|
||||||
|
|
||||||
// Plugins have provided their config, so set default config
|
// Plugins have provided their config, so set default config
|
||||||
// to main settings
|
// to main settings
|
||||||
pluginManager.loadDefaultPluginConfiguration();
|
pluginManager.loadDefaultPluginConfiguration();
|
||||||
splashScreen.setProgress(3, 5);
|
|
||||||
|
|
||||||
splashScreen.setMessage("Starting Session");
|
|
||||||
// Start client session
|
// Start client session
|
||||||
|
RuneLiteSplashScreen.stage(.80, "Starting core interface");
|
||||||
clientSessionManager.start();
|
clientSessionManager.start();
|
||||||
splashScreen.setProgress(4, 5);
|
|
||||||
|
|
||||||
// Load the session, including saved configuration
|
|
||||||
splashScreen.setMessage("Loading interface");
|
|
||||||
splashScreen.setProgress(5, 5);
|
|
||||||
|
|
||||||
// Initialize UI
|
// Initialize UI
|
||||||
clientUI.init(this);
|
clientUI.init(this);
|
||||||
@@ -371,11 +365,6 @@ public class RuneLite
|
|||||||
overlayManager.add(arrowMinimapOverlay.get());
|
overlayManager.add(arrowMinimapOverlay.get());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Close the splash screen
|
|
||||||
splashScreen.close();
|
|
||||||
|
|
||||||
clientUI.show();
|
|
||||||
|
|
||||||
// Start plugins
|
// Start plugins
|
||||||
pluginManager.startCorePlugins();
|
pluginManager.startCorePlugins();
|
||||||
|
|
||||||
@@ -386,6 +375,11 @@ public class RuneLite
|
|||||||
{
|
{
|
||||||
scheduler.registerObject(modelOutlineRenderer.get());
|
scheduler.registerObject(modelOutlineRenderer.get());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Close the splash screen
|
||||||
|
RuneLiteSplashScreen.close();
|
||||||
|
|
||||||
|
clientUI.show();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void shutdown()
|
public void shutdown()
|
||||||
|
|||||||
@@ -54,6 +54,7 @@ import java.util.concurrent.ExecutorService;
|
|||||||
import java.util.concurrent.Executors;
|
import java.util.concurrent.Executors;
|
||||||
import java.util.concurrent.Future;
|
import java.util.concurrent.Future;
|
||||||
import java.util.concurrent.ScheduledExecutorService;
|
import java.util.concurrent.ScheduledExecutorService;
|
||||||
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
import javax.inject.Inject;
|
import javax.inject.Inject;
|
||||||
import javax.inject.Named;
|
import javax.inject.Named;
|
||||||
@@ -74,6 +75,7 @@ import net.runelite.client.events.SessionOpen;
|
|||||||
import net.runelite.client.task.Schedule;
|
import net.runelite.client.task.Schedule;
|
||||||
import net.runelite.client.task.ScheduledMethod;
|
import net.runelite.client.task.ScheduledMethod;
|
||||||
import net.runelite.client.task.Scheduler;
|
import net.runelite.client.task.Scheduler;
|
||||||
|
import net.runelite.client.ui.RuneLiteSplashScreen;
|
||||||
import net.runelite.client.util.GameEventManager;
|
import net.runelite.client.util.GameEventManager;
|
||||||
|
|
||||||
@Singleton
|
@Singleton
|
||||||
@@ -216,6 +218,8 @@ public class PluginManager
|
|||||||
public void startCorePlugins()
|
public void startCorePlugins()
|
||||||
{
|
{
|
||||||
List<Plugin> scannedPlugins = new ArrayList<>(plugins);
|
List<Plugin> scannedPlugins = new ArrayList<>(plugins);
|
||||||
|
int loaded = 0;
|
||||||
|
|
||||||
for (Plugin plugin : scannedPlugins)
|
for (Plugin plugin : scannedPlugins)
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
@@ -227,12 +231,17 @@ public class PluginManager
|
|||||||
log.warn("Unable to start plugin {}. {}", plugin.getClass().getSimpleName(), ex);
|
log.warn("Unable to start plugin {}. {}", plugin.getClass().getSimpleName(), ex);
|
||||||
plugins.remove(plugin);
|
plugins.remove(plugin);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
loaded++;
|
||||||
|
|
||||||
|
RuneLiteSplashScreen.stage(.80, 1, "Starting plugins", loaded, scannedPlugins.size());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
List<Plugin> scanAndInstantiate(ClassLoader classLoader, String packageName) throws IOException
|
List<Plugin> scanAndInstantiate(ClassLoader classLoader, String packageName) throws IOException
|
||||||
{
|
{
|
||||||
|
RuneLiteSplashScreen.stage(.59, "Loading plugins");
|
||||||
MutableGraph<Class<? extends Plugin>> graph = GraphBuilder
|
MutableGraph<Class<? extends Plugin>> graph = GraphBuilder
|
||||||
.directed()
|
.directed()
|
||||||
.build();
|
.build();
|
||||||
@@ -295,6 +304,7 @@ public class PluginManager
|
|||||||
|
|
||||||
List<List<Class<? extends Plugin>>> sortedPlugins = topologicalGroupSort(graph);
|
List<List<Class<? extends Plugin>>> sortedPlugins = topologicalGroupSort(graph);
|
||||||
sortedPlugins = Lists.reverse(sortedPlugins);
|
sortedPlugins = Lists.reverse(sortedPlugins);
|
||||||
|
AtomicInteger loaded = new AtomicInteger();
|
||||||
|
|
||||||
final long start = System.currentTimeMillis();
|
final long start = System.currentTimeMillis();
|
||||||
|
|
||||||
@@ -312,13 +322,17 @@ public class PluginManager
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
plugin = instantiate(scannedPlugins, (Class<Plugin>) pluginClazz);
|
plugin = instantiate(scannedPlugins, (Class<Plugin>) pluginClazz);
|
||||||
|
scannedPlugins.add(plugin);
|
||||||
}
|
}
|
||||||
catch (PluginInstantiationException e)
|
catch (PluginInstantiationException e)
|
||||||
{
|
{
|
||||||
log.warn("Error instantiating plugin!", e);
|
log.warn("Error instantiating plugin!", e);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
scannedPlugins.add(plugin);
|
|
||||||
|
loaded.getAndIncrement();
|
||||||
|
|
||||||
|
RuneLiteSplashScreen.stage(.60, .70, "Loading plugins", loaded.get(), scannedPlugins.size());
|
||||||
})));
|
})));
|
||||||
curGroup.forEach(future ->
|
curGroup.forEach(future ->
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -29,13 +29,11 @@ import com.google.inject.Inject;
|
|||||||
import java.awt.BorderLayout;
|
import java.awt.BorderLayout;
|
||||||
import java.awt.Color;
|
import java.awt.Color;
|
||||||
import java.awt.Cursor;
|
import java.awt.Cursor;
|
||||||
import java.awt.Desktop;
|
|
||||||
import java.awt.Font;
|
import java.awt.Font;
|
||||||
import java.awt.GridLayout;
|
import java.awt.GridLayout;
|
||||||
import java.awt.event.MouseAdapter;
|
import java.awt.event.MouseAdapter;
|
||||||
import java.awt.event.MouseEvent;
|
import java.awt.event.MouseEvent;
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.io.IOException;
|
|
||||||
import java.util.concurrent.ScheduledExecutorService;
|
import java.util.concurrent.ScheduledExecutorService;
|
||||||
import javax.annotation.Nullable;
|
import javax.annotation.Nullable;
|
||||||
import javax.inject.Singleton;
|
import javax.inject.Singleton;
|
||||||
@@ -184,16 +182,7 @@ public class InfoPanel extends PluginPanel
|
|||||||
*/
|
*/
|
||||||
private JPanel buildLinkPanel(ImageIcon icon, String topText, String bottomText, File dir)
|
private JPanel buildLinkPanel(ImageIcon icon, String topText, String bottomText, File dir)
|
||||||
{
|
{
|
||||||
return buildLinkPanel(icon, topText, bottomText, () ->
|
return buildLinkPanel(icon, topText, bottomText, () -> LinkBrowser.openLocalFile(dir));
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
Desktop.getDesktop().open(dir);
|
|
||||||
}
|
|
||||||
catch (IOException ex)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -531,7 +531,7 @@ public class ClientUI
|
|||||||
if (client != null && !(client instanceof Client))
|
if (client != null && !(client instanceof Client))
|
||||||
{
|
{
|
||||||
SwingUtilities.invokeLater(() -> JOptionPane.showMessageDialog(frame,
|
SwingUtilities.invokeLater(() -> JOptionPane.showMessageDialog(frame,
|
||||||
"RuneLite has not yet been updated to work with the latest\n"
|
"RuneLitePlus has not yet been updated to work with the latest\n"
|
||||||
+ "game update, it will work with reduced functionality until then.",
|
+ "game update, it will work with reduced functionality until then.",
|
||||||
"RuneLite is outdated", INFORMATION_MESSAGE));
|
"RuneLite is outdated", INFORMATION_MESSAGE));
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2016-2017, Jeremy Plsek <github.com/jplsek>
|
* Copyright (c) 2019, TheStonedTurtle <https://github.com/TheStonedTurtle>
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*
|
*
|
||||||
* Redistribution and use in source and binary forms, with or without
|
* Redistribution and use in source and binary forms, with or without
|
||||||
@@ -24,247 +24,125 @@
|
|||||||
*/
|
*/
|
||||||
package net.runelite.client.ui;
|
package net.runelite.client.ui;
|
||||||
|
|
||||||
import java.awt.Color;
|
import java.awt.BorderLayout;
|
||||||
import java.awt.GridBagConstraints;
|
import java.awt.Dimension;
|
||||||
import java.awt.GridBagLayout;
|
import java.lang.reflect.InvocationTargetException;
|
||||||
import java.awt.Image;
|
|
||||||
import java.awt.image.BufferedImage;
|
|
||||||
import java.io.IOException;
|
|
||||||
import java.io.InputStream;
|
|
||||||
import java.util.Properties;
|
|
||||||
import javax.imageio.ImageIO;
|
|
||||||
import javax.inject.Singleton;
|
|
||||||
import javax.swing.ImageIcon;
|
|
||||||
import javax.swing.JFrame;
|
import javax.swing.JFrame;
|
||||||
import javax.swing.JLabel;
|
|
||||||
import javax.swing.JPanel;
|
import javax.swing.JPanel;
|
||||||
import javax.swing.JProgressBar;
|
import javax.swing.JProgressBar;
|
||||||
import javax.swing.SwingUtilities;
|
import javax.swing.SwingUtilities;
|
||||||
import javax.swing.plaf.basic.BasicProgressBarUI;
|
import lombok.Getter;
|
||||||
import lombok.extern.slf4j.Slf4j;
|
import lombok.extern.slf4j.Slf4j;
|
||||||
import net.runelite.client.util.SwingUtil;
|
import net.runelite.client.ui.components.InfoPanel;
|
||||||
import org.pushingpixels.substance.internal.SubstanceSynapse;
|
import net.runelite.client.ui.components.MessagePanel;
|
||||||
|
import net.runelite.client.util.ImageUtil;
|
||||||
|
|
||||||
/**
|
|
||||||
* This is a custom Splash Screen and does not use Java's SplashScreen class. This has helper methods to update the
|
|
||||||
* status while loading RuneLite. All public methods run non-blocking passed to the swing thread.
|
|
||||||
*/
|
|
||||||
@Slf4j
|
@Slf4j
|
||||||
@Singleton
|
public class RuneLiteSplashScreen extends JFrame
|
||||||
public class RuneLiteSplashScreen
|
|
||||||
{
|
{
|
||||||
private static final String RUNELITE_VERSION = "runelite.version";
|
private static RuneLiteSplashScreen INSTANCE;
|
||||||
private static final String RUNELITE_PLUS_VERSION = "runelite.plus.version";
|
public static final Dimension FRAME_SIZE = new Dimension(600, 350);
|
||||||
private static final String RUNELITE_PLUS_DATE = "runelite.plus.builddate";
|
|
||||||
|
|
||||||
private JFrame frame;
|
@Getter
|
||||||
private final JPanel panel = new JPanel();
|
private final MessagePanel messagePanel = new MessagePanel();
|
||||||
private JLabel messageLabel;
|
|
||||||
private JLabel subMessageLabel;
|
|
||||||
private final JProgressBar progressBar = new JProgressBar();
|
|
||||||
|
|
||||||
private final Properties properties = new Properties();
|
private RuneLiteSplashScreen()
|
||||||
|
|
||||||
public RuneLiteSplashScreen()
|
|
||||||
{
|
{
|
||||||
try (InputStream in = getClass().getResourceAsStream("/runelite.plus.properties"))
|
this.setTitle("RuneLitePlus");
|
||||||
|
this.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
|
||||||
|
this.setSize(FRAME_SIZE);
|
||||||
|
this.setLayout(new BorderLayout());
|
||||||
|
this.setUndecorated(true);
|
||||||
|
this.setIconImage(ImageUtil.getResourceStreamFromClass(RuneLiteSplashScreen.class, "/runeliteplus.png"));
|
||||||
|
|
||||||
|
final JPanel panel = new JPanel();
|
||||||
|
panel.setLayout(new BorderLayout());
|
||||||
|
panel.setPreferredSize(RuneLiteSplashScreen.FRAME_SIZE);
|
||||||
|
|
||||||
|
panel.add(new InfoPanel(), BorderLayout.EAST);
|
||||||
|
panel.add(messagePanel, BorderLayout.WEST);
|
||||||
|
|
||||||
|
this.setContentPane(panel);
|
||||||
|
pack();
|
||||||
|
|
||||||
|
this.setLocationRelativeTo(null);
|
||||||
|
this.setVisible(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setBarText(final String text)
|
||||||
|
{
|
||||||
|
final JProgressBar bar = messagePanel.getBar();
|
||||||
|
bar.setString(text);
|
||||||
|
bar.setStringPainted(text != null);
|
||||||
|
bar.revalidate();
|
||||||
|
bar.repaint();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setMessage(final String msg, final double value)
|
||||||
|
{
|
||||||
|
messagePanel.getBarLabel().setText(msg);
|
||||||
|
messagePanel.getBar().setMaximum(1000);
|
||||||
|
messagePanel.getBar().setValue((int) (value * 1000));
|
||||||
|
setBarText(null);
|
||||||
|
|
||||||
|
this.getContentPane().revalidate();
|
||||||
|
this.getContentPane().repaint();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void init()
|
||||||
|
{
|
||||||
|
try
|
||||||
{
|
{
|
||||||
properties.load(in);
|
SwingUtilities.invokeAndWait(() ->
|
||||||
|
{
|
||||||
|
if (INSTANCE != null)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
INSTANCE = new RuneLiteSplashScreen();
|
||||||
|
}
|
||||||
|
catch (Exception e)
|
||||||
|
{
|
||||||
|
log.warn("Unable to start splash screen", e);
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
catch (IOException ex)
|
catch (InterruptedException | InvocationTargetException bs)
|
||||||
{
|
{
|
||||||
log.warn("unable to load propertries", ex);
|
throw new RuntimeException(bs);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
public static void close()
|
||||||
* This is not done in the constructor in order to avoid processing in case the user chooses to not load
|
|
||||||
* the splash screen.
|
|
||||||
*
|
|
||||||
* @param estimatedSteps steps until completion, used for the progress bar
|
|
||||||
*/
|
|
||||||
private void initLayout(final int estimatedSteps)
|
|
||||||
{
|
{
|
||||||
SwingUtil.setupRuneLiteLookAndFeel();
|
SwingUtilities.invokeLater(() ->
|
||||||
|
|
||||||
// init fields with updated swing look and feel
|
|
||||||
frame = new JFrame("RuneLitePlus Loading");
|
|
||||||
messageLabel = new JLabel("Loading...");
|
|
||||||
subMessageLabel = new JLabel();
|
|
||||||
progressBar.setUI(new BasicProgressBarUI());
|
|
||||||
progressBar.setMinimum(0);
|
|
||||||
progressBar.setMaximum(estimatedSteps);
|
|
||||||
|
|
||||||
// frame setup
|
|
||||||
frame.setSize(220, 290);
|
|
||||||
frame.setLocationRelativeTo(null);
|
|
||||||
frame.setUndecorated(true);
|
|
||||||
|
|
||||||
// main panel setup
|
|
||||||
// To reduce substance's colorization (tinting)
|
|
||||||
panel.putClientProperty(SubstanceSynapse.COLORIZATION_FACTOR, 1.0);
|
|
||||||
panel.setBackground(ColorScheme.DARKER_GRAY_COLOR);
|
|
||||||
final GridBagLayout layout = new GridBagLayout();
|
|
||||||
layout.columnWeights = new double[]{1};
|
|
||||||
layout.rowWeights = new double[]{1, 0, 0, 1, 0, 0, 1};
|
|
||||||
panel.setLayout(layout);
|
|
||||||
|
|
||||||
// logo
|
|
||||||
synchronized (ImageIO.class)
|
|
||||||
{
|
{
|
||||||
try
|
if (INSTANCE == null)
|
||||||
{
|
{
|
||||||
final BufferedImage logo = ImageIO.read(RuneLiteSplashScreen.class.getResourceAsStream("/runeliteplus.png"));
|
return;
|
||||||
frame.setIconImage(logo);
|
}
|
||||||
|
|
||||||
final BufferedImage logoTransparent = ImageIO.read(RuneLiteSplashScreen.class.getResourceAsStream("/runeliteplus_transparent.png"));
|
INSTANCE.setVisible(false);
|
||||||
final GridBagConstraints logoConstraints = new GridBagConstraints();
|
INSTANCE.dispose();
|
||||||
logoConstraints.anchor = GridBagConstraints.SOUTH;
|
INSTANCE = null;
|
||||||
panel.add(new JLabel(new ImageIcon(logoTransparent.getScaledInstance(96, 96, Image.SCALE_SMOOTH))), logoConstraints);
|
});
|
||||||
}
|
}
|
||||||
catch (IOException e)
|
|
||||||
{
|
public static void stage(double startProgress, double endProgress,
|
||||||
log.warn("Error loading logo", e);
|
String progressText, int done, int total)
|
||||||
}
|
{
|
||||||
|
String progress = done + " / " + total;
|
||||||
|
stage(startProgress + ((endProgress - startProgress) * done / total), progressText + " " + progress);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void stage(double overallProgress, String progressText)
|
||||||
|
{
|
||||||
|
if (INSTANCE != null)
|
||||||
|
{
|
||||||
|
INSTANCE.setMessage(progressText, overallProgress);
|
||||||
}
|
}
|
||||||
|
|
||||||
// runelite title
|
|
||||||
final JLabel title = new JLabel("RuneLitePlus");
|
|
||||||
final GridBagConstraints titleConstraints = new GridBagConstraints();
|
|
||||||
titleConstraints.gridy = 1;
|
|
||||||
panel.add(title, titleConstraints);
|
|
||||||
|
|
||||||
// version
|
|
||||||
final JLabel version = new JLabel("RuneLite Version : " + properties.getProperty(RUNELITE_VERSION));
|
|
||||||
version.setForeground(Color.GREEN);
|
|
||||||
version.setFont(FontManager.getRunescapeSmallFont());
|
|
||||||
version.setForeground(version.getForeground().darker());
|
|
||||||
final GridBagConstraints versionConstraints = new GridBagConstraints();
|
|
||||||
versionConstraints.gridy = 2;
|
|
||||||
panel.add(version, versionConstraints);
|
|
||||||
|
|
||||||
// version
|
|
||||||
final JLabel litVersion = new JLabel("Plus Version : " + properties.getProperty(RUNELITE_PLUS_VERSION));
|
|
||||||
litVersion.setForeground(Color.GREEN);
|
|
||||||
litVersion.setFont(FontManager.getRunescapeSmallFont());
|
|
||||||
litVersion.setForeground(litVersion.getForeground().darker());
|
|
||||||
final GridBagConstraints litVersionConstraints = new GridBagConstraints();
|
|
||||||
litVersionConstraints.gridy = 3;
|
|
||||||
panel.add(litVersion, litVersionConstraints);
|
|
||||||
|
|
||||||
// build date
|
|
||||||
final JLabel litBuildDate = new JLabel("Build date : " + properties.getProperty(RUNELITE_PLUS_DATE));
|
|
||||||
litBuildDate.setForeground(Color.GREEN);
|
|
||||||
litBuildDate.setFont(FontManager.getRunescapeSmallFont());
|
|
||||||
litBuildDate.setForeground(litBuildDate.getForeground().darker());
|
|
||||||
final GridBagConstraints litBuildDateConstraints = new GridBagConstraints();
|
|
||||||
litBuildDateConstraints.gridy = 4;
|
|
||||||
panel.add(litBuildDate, litBuildDateConstraints);
|
|
||||||
|
|
||||||
|
|
||||||
// progressbar
|
|
||||||
final GridBagConstraints progressConstraints = new GridBagConstraints();
|
|
||||||
progressConstraints.fill = GridBagConstraints.HORIZONTAL;
|
|
||||||
progressConstraints.anchor = GridBagConstraints.SOUTH;
|
|
||||||
progressConstraints.gridy = 5;
|
|
||||||
panel.add(progressBar, progressConstraints);
|
|
||||||
|
|
||||||
// main message
|
|
||||||
messageLabel.setFont(FontManager.getRunescapeSmallFont());
|
|
||||||
final GridBagConstraints messageConstraints = new GridBagConstraints();
|
|
||||||
messageConstraints.gridy = 6;
|
|
||||||
panel.add(messageLabel, messageConstraints);
|
|
||||||
|
|
||||||
// alternate message
|
|
||||||
final GridBagConstraints subMessageConstraints = new GridBagConstraints();
|
|
||||||
subMessageLabel.setForeground(subMessageLabel.getForeground().darker());
|
|
||||||
subMessageLabel.setFont(FontManager.getRunescapeSmallFont());
|
|
||||||
subMessageConstraints.gridy = 7;
|
|
||||||
panel.add(subMessageLabel, subMessageConstraints);
|
|
||||||
|
|
||||||
frame.setContentPane(panel);
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
private boolean notActive()
|
|
||||||
{
|
|
||||||
return frame == null || !frame.isDisplayable();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Close/dispose of the splash screen
|
|
||||||
*/
|
|
||||||
public void close()
|
|
||||||
{
|
|
||||||
SwingUtilities.invokeLater(() ->
|
|
||||||
{
|
|
||||||
if (notActive())
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
frame.dispose();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Set the splash screen to be visible.
|
|
||||||
*
|
|
||||||
* @param estimatedSteps steps until completion, used for the progress bar
|
|
||||||
*/
|
|
||||||
public void open(final int estimatedSteps)
|
|
||||||
{
|
|
||||||
SwingUtilities.invokeLater(() ->
|
|
||||||
{
|
|
||||||
initLayout(estimatedSteps);
|
|
||||||
frame.setVisible(true);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setMessage(final String message)
|
|
||||||
{
|
|
||||||
SwingUtilities.invokeLater(() ->
|
|
||||||
{
|
|
||||||
if (notActive())
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
messageLabel.setText(message);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setSubMessage(final String subMessage)
|
|
||||||
{
|
|
||||||
SwingUtilities.invokeLater(() ->
|
|
||||||
{
|
|
||||||
if (notActive())
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
subMessageLabel.setText(subMessage);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setProgress(int currentProgress, int progressGoal)
|
|
||||||
{
|
|
||||||
SwingUtilities.invokeLater(() ->
|
|
||||||
{
|
|
||||||
if (notActive())
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (progressGoal != progressBar.getMaximum())
|
|
||||||
{
|
|
||||||
panel.remove(progressBar);
|
|
||||||
panel.validate();
|
|
||||||
final GridBagConstraints progressConstraints = new GridBagConstraints();
|
|
||||||
progressConstraints.fill = GridBagConstraints.HORIZONTAL;
|
|
||||||
progressConstraints.anchor = GridBagConstraints.SOUTH;
|
|
||||||
progressConstraints.gridy = 5;
|
|
||||||
panel.add(progressBar, progressConstraints);
|
|
||||||
panel.validate();
|
|
||||||
}
|
|
||||||
progressBar.setMaximum(progressGoal);
|
|
||||||
progressBar.setValue(currentProgress);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,187 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2019, TheStonedTurtle <https://github.com/TheStonedTurtle>
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright notice, this
|
||||||
|
* list of conditions and the following disclaimer.
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer in the documentation
|
||||||
|
* and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
||||||
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
||||||
|
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
package net.runelite.client.ui.components;
|
||||||
|
|
||||||
|
import java.awt.Color;
|
||||||
|
import java.awt.Cursor;
|
||||||
|
import java.awt.Dimension;
|
||||||
|
import java.awt.GridBagConstraints;
|
||||||
|
import java.awt.GridBagLayout;
|
||||||
|
import java.awt.Image;
|
||||||
|
import java.awt.event.MouseAdapter;
|
||||||
|
import java.awt.event.MouseEvent;
|
||||||
|
import java.awt.image.BufferedImage;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.util.Properties;
|
||||||
|
import javax.swing.ImageIcon;
|
||||||
|
import javax.swing.JLabel;
|
||||||
|
import javax.swing.JPanel;
|
||||||
|
import javax.swing.border.CompoundBorder;
|
||||||
|
import javax.swing.border.EmptyBorder;
|
||||||
|
import javax.swing.border.MatteBorder;
|
||||||
|
import lombok.extern.slf4j.Slf4j;
|
||||||
|
import net.runelite.client.RuneLiteProperties;
|
||||||
|
import net.runelite.client.ui.ColorScheme;
|
||||||
|
import net.runelite.client.ui.FontManager;
|
||||||
|
import net.runelite.client.ui.RuneLiteSplashScreen;
|
||||||
|
import net.runelite.client.util.ImageUtil;
|
||||||
|
import net.runelite.client.util.LinkBrowser;
|
||||||
|
|
||||||
|
@Slf4j
|
||||||
|
public class InfoPanel extends JPanel
|
||||||
|
{
|
||||||
|
private static final String RUNELITE_VERSION = "runelite.version";
|
||||||
|
private static final String RUNELITE_PLUS_VERSION = "runelite.plus.version";
|
||||||
|
private static final String RUNELITE_PLUS_DATE = "runelite.plus.builddate";
|
||||||
|
|
||||||
|
private static final BufferedImage TRANSPARENT_LOGO = ImageUtil.getResourceStreamFromClass(InfoPanel.class, "/runeliteplus_transparent.png");
|
||||||
|
static final Dimension PANEL_SIZE = new Dimension(200, RuneLiteSplashScreen.FRAME_SIZE.height);
|
||||||
|
private static final Dimension VERSION_SIZE = new Dimension(PANEL_SIZE.width, 25);
|
||||||
|
private static final File RUNELITE_DIR = new File(System.getProperty("user.home"), ".runelite");
|
||||||
|
private static final File LOGS_DIR = new File(RUNELITE_DIR, "logs");
|
||||||
|
|
||||||
|
private final Properties properties = new Properties();
|
||||||
|
|
||||||
|
public InfoPanel()
|
||||||
|
{
|
||||||
|
try (InputStream in = getClass().getResourceAsStream("/runelite.plus.properties"))
|
||||||
|
{
|
||||||
|
properties.load(in);
|
||||||
|
}
|
||||||
|
catch (IOException ex)
|
||||||
|
{
|
||||||
|
log.warn("unable to load propertries", ex);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.setLayout(new GridBagLayout());
|
||||||
|
this.setPreferredSize(PANEL_SIZE);
|
||||||
|
this.setBackground(new Color(38, 38, 38));
|
||||||
|
|
||||||
|
final GridBagConstraints c = new GridBagConstraints();
|
||||||
|
c.fill = GridBagConstraints.HORIZONTAL;
|
||||||
|
c.weightx = 1;
|
||||||
|
c.gridx = 0;
|
||||||
|
c.gridy = 0;
|
||||||
|
c.ipady = 5;
|
||||||
|
|
||||||
|
// Logo
|
||||||
|
final ImageIcon transparentLogo = new ImageIcon();
|
||||||
|
if (TRANSPARENT_LOGO != null)
|
||||||
|
{
|
||||||
|
transparentLogo.setImage(TRANSPARENT_LOGO.getScaledInstance(128, 128, Image.SCALE_SMOOTH));
|
||||||
|
}
|
||||||
|
final JLabel logo = new JLabel(transparentLogo);
|
||||||
|
|
||||||
|
c.anchor = GridBagConstraints.NORTH;
|
||||||
|
c.weighty = 1;
|
||||||
|
this.add(logo, c);
|
||||||
|
c.gridy++;
|
||||||
|
c.anchor = GridBagConstraints.SOUTH;
|
||||||
|
c.weighty = 0;
|
||||||
|
|
||||||
|
// Version
|
||||||
|
this.add(createPanelTextButton("RuneLite Version: " + properties.getProperty(RUNELITE_VERSION)), c);
|
||||||
|
c.gridy++;
|
||||||
|
|
||||||
|
// Plus version
|
||||||
|
this.add(createPanelTextButton("Plus Version: " + properties.getProperty(RUNELITE_PLUS_VERSION)), c);
|
||||||
|
c.gridy++;
|
||||||
|
|
||||||
|
// Build date
|
||||||
|
this.add(createPanelTextButton("Build date: " + properties.getProperty(RUNELITE_PLUS_DATE)), c);
|
||||||
|
c.gridy++;
|
||||||
|
|
||||||
|
final JLabel logsFolder = createPanelButton("Open logs folder", null, () -> LinkBrowser.openLocalFile(LOGS_DIR));
|
||||||
|
this.add(logsFolder, c);
|
||||||
|
c.gridy++;
|
||||||
|
|
||||||
|
final JLabel discord = createPanelButton("Get help on Discord", "Instant invite link to join the RuneLitePlus discord", () -> LinkBrowser.browse(RuneLiteProperties.getDiscordInvite()));
|
||||||
|
this.add(discord, c);
|
||||||
|
c.gridy++;
|
||||||
|
|
||||||
|
final JLabel troubleshooting = createPanelButton("Troubleshooting steps", "Opens a link to the troubleshooting wiki", () -> LinkBrowser.browse(RuneLiteProperties.getTroubleshootingLink()));
|
||||||
|
this.add(troubleshooting, c);
|
||||||
|
c.gridy++;
|
||||||
|
|
||||||
|
final JLabel exit = createPanelButton("Exit", "Closes the application immediately", () -> System.exit(0));
|
||||||
|
this.add(exit, c);
|
||||||
|
c.gridy++;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static JLabel createPanelTextButton(final String title)
|
||||||
|
{
|
||||||
|
final JLabel textButton = new JLabel(title);
|
||||||
|
textButton.setFont(FontManager.getRunescapeSmallFont());
|
||||||
|
textButton.setHorizontalAlignment(JLabel.CENTER);
|
||||||
|
textButton.setForeground(ColorScheme.BRAND_ORANGE);
|
||||||
|
textButton.setBackground(null);
|
||||||
|
textButton.setPreferredSize(VERSION_SIZE);
|
||||||
|
textButton.setMinimumSize(VERSION_SIZE);
|
||||||
|
textButton.setBorder(new MatteBorder(1, 0, 0, 0, Color.LIGHT_GRAY));
|
||||||
|
|
||||||
|
return textButton;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static JLabel createPanelButton(final String name, final String tooltip, final Runnable runnable)
|
||||||
|
{
|
||||||
|
final JLabel btn = new JLabel(name, JLabel.CENTER);
|
||||||
|
btn.setToolTipText(tooltip);
|
||||||
|
btn.setOpaque(true);
|
||||||
|
btn.setBackground(null);
|
||||||
|
btn.setForeground(Color.WHITE);
|
||||||
|
btn.setFont(FontManager.getRunescapeFont());
|
||||||
|
btn.setBorder(new CompoundBorder(
|
||||||
|
new MatteBorder(1, 0, 0, 0, Color.LIGHT_GRAY),
|
||||||
|
new EmptyBorder(3, 0, 3, 0))
|
||||||
|
);
|
||||||
|
btn.setCursor(Cursor.getPredefinedCursor(Cursor.HAND_CURSOR));
|
||||||
|
btn.addMouseListener(new MouseAdapter()
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public void mouseClicked(MouseEvent e)
|
||||||
|
{
|
||||||
|
runnable.run();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void mouseEntered(MouseEvent e)
|
||||||
|
{
|
||||||
|
btn.setBackground(new Color(60, 60, 60));
|
||||||
|
btn.repaint();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void mouseExited(MouseEvent e)
|
||||||
|
{
|
||||||
|
btn.setBackground(null);
|
||||||
|
btn.repaint();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return btn;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,155 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2019, TheStonedTurtle <https://github.com/TheStonedTurtle>
|
||||||
|
* All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without
|
||||||
|
* modification, are permitted provided that the following conditions are met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright notice, this
|
||||||
|
* list of conditions and the following disclaimer.
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer in the documentation
|
||||||
|
* and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
||||||
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
||||||
|
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||||||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
package net.runelite.client.ui.components;
|
||||||
|
|
||||||
|
import java.awt.Color;
|
||||||
|
import java.awt.Dimension;
|
||||||
|
import java.awt.Font;
|
||||||
|
import java.awt.GridBagConstraints;
|
||||||
|
import java.awt.GridBagLayout;
|
||||||
|
import javax.swing.JLabel;
|
||||||
|
import javax.swing.JPanel;
|
||||||
|
import javax.swing.JProgressBar;
|
||||||
|
import javax.swing.JScrollPane;
|
||||||
|
import javax.swing.JViewport;
|
||||||
|
import javax.swing.border.EmptyBorder;
|
||||||
|
import javax.swing.border.MatteBorder;
|
||||||
|
import javax.swing.plaf.basic.BasicProgressBarUI;
|
||||||
|
import lombok.AccessLevel;
|
||||||
|
import lombok.Getter;
|
||||||
|
import net.runelite.client.ui.ColorScheme;
|
||||||
|
import net.runelite.client.ui.FontManager;
|
||||||
|
import net.runelite.client.ui.RuneLiteSplashScreen;
|
||||||
|
|
||||||
|
@Getter
|
||||||
|
public class MessagePanel extends JPanel
|
||||||
|
{
|
||||||
|
private static final Dimension PANEL_SIZE = new Dimension(RuneLiteSplashScreen.FRAME_SIZE.width - InfoPanel.PANEL_SIZE.width, RuneLiteSplashScreen.FRAME_SIZE.height);
|
||||||
|
private static final Dimension BAR_SIZE = new Dimension(PANEL_SIZE.width, 30);
|
||||||
|
private static final int MESSAGE_AREA_PADDING = 15;
|
||||||
|
|
||||||
|
private final JLabel titleLabel = new JLabel("Welcome to RuneLitePlus");
|
||||||
|
private final JLabel messageArea;
|
||||||
|
private final JLabel barLabel = new JLabel("Doing something important");
|
||||||
|
private final JProgressBar bar = new JProgressBar(0, 100);
|
||||||
|
|
||||||
|
@Getter(AccessLevel.NONE)
|
||||||
|
private final JScrollPane scrollPane;
|
||||||
|
|
||||||
|
public MessagePanel()
|
||||||
|
{
|
||||||
|
this.setPreferredSize(PANEL_SIZE);
|
||||||
|
this.setLayout(new GridBagLayout());
|
||||||
|
this.setBackground(ColorScheme.DARKER_GRAY_COLOR);
|
||||||
|
|
||||||
|
final GridBagConstraints c = new GridBagConstraints();
|
||||||
|
c.fill = GridBagConstraints.HORIZONTAL;
|
||||||
|
c.anchor = GridBagConstraints.NORTH;
|
||||||
|
c.weightx = 1;
|
||||||
|
c.gridx = 0;
|
||||||
|
c.gridy = 0;
|
||||||
|
c.ipady = 25;
|
||||||
|
|
||||||
|
// main message
|
||||||
|
titleLabel.setFont(new Font(FontManager.getRunescapeFont().getName(), FontManager.getRunescapeFont().getStyle(), 32));
|
||||||
|
titleLabel.setHorizontalAlignment(JLabel.CENTER);
|
||||||
|
titleLabel.setForeground(Color.WHITE);
|
||||||
|
this.add(titleLabel, c);
|
||||||
|
c.gridy++;
|
||||||
|
|
||||||
|
// alternate message action
|
||||||
|
messageArea = new JLabel("<html><div style='text-align:center;'>Fork of RuneLite that provides more functionality and less restrictions while staying open source.</div></html>")
|
||||||
|
{
|
||||||
|
@Override
|
||||||
|
public Dimension getPreferredSize()
|
||||||
|
{
|
||||||
|
final Dimension results = super.getPreferredSize();
|
||||||
|
results.width = PANEL_SIZE.width - MESSAGE_AREA_PADDING;
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
messageArea.setFont(new Font(FontManager.getRunescapeFont().getName(), FontManager.getRunescapeSmallFont().getStyle(), 16));
|
||||||
|
messageArea.setForeground(Color.WHITE);
|
||||||
|
messageArea.setBorder(new EmptyBorder(0, MESSAGE_AREA_PADDING, 0, MESSAGE_AREA_PADDING));
|
||||||
|
|
||||||
|
scrollPane = new JScrollPane(messageArea, JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED, JScrollPane.HORIZONTAL_SCROLLBAR_NEVER);
|
||||||
|
scrollPane.setBorder(new EmptyBorder(0, 0, 0, 0));
|
||||||
|
scrollPane.getVerticalScrollBar().setUI(new CustomScrollBarUI());
|
||||||
|
final JViewport viewport = scrollPane.getViewport();
|
||||||
|
viewport.setForeground(Color.WHITE);
|
||||||
|
viewport.setBackground(ColorScheme.DARKER_GRAY_COLOR);
|
||||||
|
viewport.setOpaque(true);
|
||||||
|
|
||||||
|
c.weighty = 1;
|
||||||
|
c.fill = 1;
|
||||||
|
this.add(scrollPane, c);
|
||||||
|
c.gridy++;
|
||||||
|
|
||||||
|
c.weighty = 0;
|
||||||
|
c.weightx = 1;
|
||||||
|
c.ipady = 5;
|
||||||
|
|
||||||
|
barLabel.setFont(FontManager.getRunescapeFont());
|
||||||
|
barLabel.setHorizontalAlignment(JLabel.CENTER);
|
||||||
|
barLabel.setForeground(Color.WHITE);
|
||||||
|
barLabel.setBorder(new EmptyBorder(5, 0, 5, 0));
|
||||||
|
this.add(barLabel, c);
|
||||||
|
c.gridy++;
|
||||||
|
|
||||||
|
bar.setBackground(ColorScheme.BRAND_ORANGE_TRANSPARENT.darker());
|
||||||
|
bar.setForeground(ColorScheme.BRAND_ORANGE);
|
||||||
|
bar.setMinimumSize(BAR_SIZE);
|
||||||
|
bar.setMaximumSize(BAR_SIZE);
|
||||||
|
bar.setBorder(new MatteBorder(0, 0, 0, 0, Color.LIGHT_GRAY));
|
||||||
|
bar.setUI(new BasicProgressBarUI()
|
||||||
|
{
|
||||||
|
protected Color getSelectionBackground()
|
||||||
|
{
|
||||||
|
return ColorScheme.DARKER_GRAY_COLOR;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected Color getSelectionForeground()
|
||||||
|
{
|
||||||
|
return ColorScheme.DARKER_GRAY_COLOR;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
bar.setFont(FontManager.getRunescapeFont());
|
||||||
|
bar.setVisible(true);
|
||||||
|
this.add(bar, c);
|
||||||
|
c.gridy++;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setMessageContent(String content)
|
||||||
|
{
|
||||||
|
if (!content.startsWith("<html"))
|
||||||
|
{
|
||||||
|
content = "<html><div style='text-align:center;'>" + content + "</div></html>";
|
||||||
|
}
|
||||||
|
|
||||||
|
messageArea.setText(content);
|
||||||
|
messageArea.revalidate();
|
||||||
|
messageArea.repaint();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -26,6 +26,7 @@ package net.runelite.client.util;
|
|||||||
|
|
||||||
import com.google.common.base.Strings;
|
import com.google.common.base.Strings;
|
||||||
import java.awt.Desktop;
|
import java.awt.Desktop;
|
||||||
|
import java.io.File;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.net.URI;
|
import java.net.URI;
|
||||||
import java.net.URISyntaxException;
|
import java.net.URISyntaxException;
|
||||||
@@ -128,6 +129,55 @@ public class LinkBrowser
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tries to open the specified {@code File} with the systems default text editor. If operation fails
|
||||||
|
* an error message is displayed with the option to copy the absolute file path to clipboard.
|
||||||
|
* @param file the File instance of the log file
|
||||||
|
* @return did the file open successfully?
|
||||||
|
*/
|
||||||
|
public static boolean openLocalFile(final File file)
|
||||||
|
{
|
||||||
|
if (file == null || !file.exists())
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (attemptOpenLocalFile(file))
|
||||||
|
{
|
||||||
|
log.debug("Opened log file through Desktop#edit to {}", file);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
showMessageBox("Unable to open log file. Press 'OK' and the file path will be copied to your clipboard", file.getAbsolutePath());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static boolean attemptOpenLocalFile(final File file)
|
||||||
|
{
|
||||||
|
if (!Desktop.isDesktopSupported())
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
final Desktop desktop = Desktop.getDesktop();
|
||||||
|
|
||||||
|
if (!desktop.isSupported(Desktop.Action.OPEN))
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
desktop.open(file);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
catch (IOException ex)
|
||||||
|
{
|
||||||
|
log.warn("Failed to open Desktop#edit {}", file, ex);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Open swing message box with specified message and copy data to clipboard
|
* Open swing message box with specified message and copy data to clipboard
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -25,7 +25,9 @@
|
|||||||
package net.runelite.client.util;
|
package net.runelite.client.util;
|
||||||
|
|
||||||
import java.awt.AWTException;
|
import java.awt.AWTException;
|
||||||
|
import java.awt.BorderLayout;
|
||||||
import java.awt.Color;
|
import java.awt.Color;
|
||||||
|
import java.awt.Component;
|
||||||
import java.awt.Dimension;
|
import java.awt.Dimension;
|
||||||
import java.awt.Font;
|
import java.awt.Font;
|
||||||
import java.awt.Frame;
|
import java.awt.Frame;
|
||||||
@@ -42,18 +44,26 @@ import java.util.concurrent.Callable;
|
|||||||
import java.util.function.BiConsumer;
|
import java.util.function.BiConsumer;
|
||||||
import javax.annotation.Nonnull;
|
import javax.annotation.Nonnull;
|
||||||
import javax.annotation.Nullable;
|
import javax.annotation.Nullable;
|
||||||
|
import javax.swing.ButtonModel;
|
||||||
import javax.swing.ImageIcon;
|
import javax.swing.ImageIcon;
|
||||||
import javax.swing.JButton;
|
import javax.swing.JButton;
|
||||||
|
import javax.swing.JComponent;
|
||||||
|
import javax.swing.JDialog;
|
||||||
import javax.swing.JFrame;
|
import javax.swing.JFrame;
|
||||||
|
import javax.swing.JLabel;
|
||||||
import javax.swing.JMenuItem;
|
import javax.swing.JMenuItem;
|
||||||
import javax.swing.JOptionPane;
|
import javax.swing.JOptionPane;
|
||||||
|
import javax.swing.JPanel;
|
||||||
import javax.swing.JPopupMenu;
|
import javax.swing.JPopupMenu;
|
||||||
import javax.swing.LookAndFeel;
|
import javax.swing.LookAndFeel;
|
||||||
import javax.swing.ToolTipManager;
|
import javax.swing.ToolTipManager;
|
||||||
import javax.swing.UIManager;
|
import javax.swing.UIManager;
|
||||||
import javax.swing.UnsupportedLookAndFeelException;
|
import javax.swing.UnsupportedLookAndFeelException;
|
||||||
import static javax.swing.WindowConstants.DO_NOTHING_ON_CLOSE;
|
import static javax.swing.WindowConstants.DO_NOTHING_ON_CLOSE;
|
||||||
|
import javax.swing.border.Border;
|
||||||
|
import javax.swing.border.CompoundBorder;
|
||||||
import javax.swing.border.EmptyBorder;
|
import javax.swing.border.EmptyBorder;
|
||||||
|
import javax.swing.border.MatteBorder;
|
||||||
import javax.swing.plaf.FontUIResource;
|
import javax.swing.plaf.FontUIResource;
|
||||||
import javax.swing.plaf.basic.BasicProgressBarUI;
|
import javax.swing.plaf.basic.BasicProgressBarUI;
|
||||||
import lombok.extern.slf4j.Slf4j;
|
import lombok.extern.slf4j.Slf4j;
|
||||||
@@ -293,6 +303,136 @@ public class SwingUtil
|
|||||||
return button;
|
return button;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a custom {@link JButton} with a flat design for use inside {@link JOptionPane}.
|
||||||
|
* The button will display the passed {@code text} and set the value of the pane to {@code buttonOption} on click
|
||||||
|
*
|
||||||
|
* @param text text to be displayed inside the button
|
||||||
|
* @param buttonOption the code to be set via {@link JOptionPane#setValue(Object)}
|
||||||
|
* @return newly created {@link JButton}
|
||||||
|
*/
|
||||||
|
public static JButton createFlatButton(final String text, final int buttonOption)
|
||||||
|
{
|
||||||
|
final Border BUTTON_BORDER = new EmptyBorder(5, 17, 5, 17);
|
||||||
|
final Border BORDERED_BUTTON_BORDER = new CompoundBorder(
|
||||||
|
new MatteBorder(1, 1, 1, 1, Color.BLACK),
|
||||||
|
new EmptyBorder(4, 16, 4, 16)
|
||||||
|
);
|
||||||
|
|
||||||
|
final JButton button = new JButton(text);
|
||||||
|
button.setForeground(Color.WHITE);
|
||||||
|
button.setBackground(Color.BLACK);
|
||||||
|
button.setFont(FontManager.getRunescapeFont());
|
||||||
|
button.setBorder(BUTTON_BORDER);
|
||||||
|
|
||||||
|
button.setBorderPainted(false);
|
||||||
|
button.setFocusPainted(false);
|
||||||
|
button.setContentAreaFilled(false);
|
||||||
|
button.setOpaque(true);
|
||||||
|
|
||||||
|
// Selecting the button option requires us to determine which parent element is the JOptionPane
|
||||||
|
button.addActionListener(e -> {
|
||||||
|
JComponent component = (JComponent) e.getSource();
|
||||||
|
while (component != null)
|
||||||
|
{
|
||||||
|
if (component instanceof JOptionPane)
|
||||||
|
{
|
||||||
|
((JOptionPane) component).setValue(buttonOption);
|
||||||
|
component = null;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
component = component.getParent() == null ? null : (JComponent) component.getParent();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Use change listener instead of mouse listener for buttons
|
||||||
|
button.getModel().addChangeListener(e ->
|
||||||
|
{
|
||||||
|
final ButtonModel model = (ButtonModel) e.getSource();
|
||||||
|
button.setBackground(model.isRollover() ? ColorScheme.DARKER_GRAY_HOVER_COLOR : Color.BLACK);
|
||||||
|
button.setBorderPainted(model.isPressed());
|
||||||
|
button.setBorder(model.isPressed() ? BORDERED_BUTTON_BORDER : BUTTON_BORDER);
|
||||||
|
});
|
||||||
|
|
||||||
|
return button;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Opens a {@link JDialog} with a stylized {@link JOptionPane} ignoring UIManager defaults.
|
||||||
|
* The buttons should be created via the {@link #createFlatButton(String, int)} function to look correctly
|
||||||
|
*
|
||||||
|
* @param component The frame the dialog should be attached to. nullable
|
||||||
|
* @param content The string content to be added to the content pane
|
||||||
|
* @param optionType The JOptionPane option type of dialog pane to create
|
||||||
|
* @param buttons Buttons to display, created via {@link #createFlatButton(String, int)}
|
||||||
|
* @return The Integer value representing the button selected
|
||||||
|
*/
|
||||||
|
public static int showRuneLiteOptionPane(final JComponent component, final String content, final int optionType, final JButton[] buttons)
|
||||||
|
{
|
||||||
|
final JLabel contentLabel = new JLabel(content);
|
||||||
|
contentLabel.setFont(FontManager.getRunescapeFont());
|
||||||
|
contentLabel.setForeground(Color.WHITE);
|
||||||
|
contentLabel.setBackground(ColorScheme.DARKER_GRAY_COLOR);
|
||||||
|
|
||||||
|
final JPanel p = new JPanel(new BorderLayout());
|
||||||
|
p.setBackground(ColorScheme.DARKER_GRAY_COLOR);
|
||||||
|
p.setForeground(Color.WHITE);
|
||||||
|
p.add(contentLabel, BorderLayout.NORTH);
|
||||||
|
|
||||||
|
final JOptionPane pane = new JOptionPane(p,
|
||||||
|
JOptionPane.ERROR_MESSAGE,
|
||||||
|
optionType,
|
||||||
|
null,
|
||||||
|
buttons,
|
||||||
|
buttons[1]);
|
||||||
|
pane.setBackground(ColorScheme.DARKER_GRAY_COLOR);
|
||||||
|
pane.setForeground(Color.WHITE);
|
||||||
|
stylizeJPanels(pane);
|
||||||
|
|
||||||
|
final Frame frame = component == null ? JOptionPane.getRootFrame() : JOptionPane.getFrameForComponent(component);
|
||||||
|
final JDialog dialog = new JDialog(frame, "RuneLitePlus Error", true);
|
||||||
|
dialog.setContentPane(pane);
|
||||||
|
dialog.setDefaultCloseOperation(JDialog.DISPOSE_ON_CLOSE);
|
||||||
|
dialog.setAlwaysOnTop(true);
|
||||||
|
dialog.setAutoRequestFocus(true);
|
||||||
|
dialog.setLocationRelativeTo(null);
|
||||||
|
dialog.setIconImage(ImageUtil.getResourceStreamFromClass(SwingUtil.class, "/runeliteplus_transparent.png"));
|
||||||
|
|
||||||
|
// Listen for value changes and close dialog when necessary
|
||||||
|
pane.addPropertyChangeListener(e -> {
|
||||||
|
String prop = e.getPropertyName();
|
||||||
|
|
||||||
|
if (dialog.isVisible()
|
||||||
|
&& (e.getSource() == pane)
|
||||||
|
&& (prop.equals(JOptionPane.VALUE_PROPERTY)))
|
||||||
|
{
|
||||||
|
dialog.setVisible(false);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
dialog.pack();
|
||||||
|
// Try to center dialog based on its size
|
||||||
|
dialog.setLocation(dialog.getX() - dialog.getSize().width / 2, dialog.getY() - dialog.getSize().height / 2);
|
||||||
|
dialog.setVisible(true);
|
||||||
|
|
||||||
|
return (Integer) pane.getValue();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void stylizeJPanels(final JComponent component)
|
||||||
|
{
|
||||||
|
for (final Component c : component.getComponents())
|
||||||
|
{
|
||||||
|
if (c instanceof JPanel)
|
||||||
|
{
|
||||||
|
c.setBackground(ColorScheme.DARKER_GRAY_COLOR);
|
||||||
|
c.setForeground(Color.WHITE);
|
||||||
|
stylizeJPanels((JComponent) c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets up the RuneLite look and feel. Checks to see if the look and feel
|
* Sets up the RuneLite look and feel. Checks to see if the look and feel
|
||||||
* was already set up before running in case the splash screen has already
|
* was already set up before running in case the splash screen has already
|
||||||
|
|||||||
Reference in New Issue
Block a user