mirror of
https://github.com/TotalFreedomMC/TF-EssentialsX.git
synced 2025-02-13 12:38:58 +00:00
![snowleo](/assets/img/avatar_default.png)
Found using PMD and FindBugs. git-svn-id: https://svn.java.net/svn/essentials~svn/trunk@1553 e251c2fe-e539-e718-e476-b85c1f46cddb
449 lines
11 KiB
Java
449 lines
11 KiB
Java
package com.earth2me.essentials;
|
|
|
|
import java.io.BufferedReader;
|
|
import java.io.File;
|
|
import java.io.FileInputStream;
|
|
import java.io.FileReader;
|
|
import java.io.IOException;
|
|
import java.io.InputStream;
|
|
import java.net.MalformedURLException;
|
|
import java.net.URL;
|
|
import java.text.DecimalFormat;
|
|
import java.text.DecimalFormatSymbols;
|
|
import java.text.MessageFormat;
|
|
import java.util.Calendar;
|
|
import java.util.Enumeration;
|
|
import java.util.GregorianCalendar;
|
|
import java.util.Locale;
|
|
import java.util.MissingResourceException;
|
|
import java.util.ResourceBundle;
|
|
import java.util.logging.Level;
|
|
import java.util.logging.Logger;
|
|
import java.util.regex.Matcher;
|
|
import java.util.regex.Pattern;
|
|
import org.bukkit.Location;
|
|
import org.bukkit.Material;
|
|
import org.bukkit.World;
|
|
import org.bukkit.block.Block;
|
|
|
|
|
|
public class Util
|
|
{
|
|
private Util()
|
|
{
|
|
}
|
|
private final static Logger logger = Logger.getLogger("Minecraft");
|
|
|
|
public static String sanitizeFileName(String name)
|
|
{
|
|
return name.toLowerCase().replaceAll("[^a-z0-9]", "_");
|
|
}
|
|
|
|
public static String formatDateDiff(long date)
|
|
{
|
|
Calendar c = new GregorianCalendar();
|
|
c.setTimeInMillis(date);
|
|
Calendar now = new GregorianCalendar();
|
|
return Util.formatDateDiff(now, c);
|
|
}
|
|
|
|
public static String formatDateDiff(Calendar fromDate, Calendar toDate)
|
|
{
|
|
boolean future = false;
|
|
if (toDate.equals(fromDate))
|
|
{
|
|
return Util.i18n("now");
|
|
}
|
|
if (toDate.after(fromDate))
|
|
{
|
|
future = true;
|
|
}
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
int[] types = new int[]
|
|
{
|
|
Calendar.YEAR,
|
|
Calendar.MONTH,
|
|
Calendar.DAY_OF_MONTH,
|
|
Calendar.HOUR_OF_DAY,
|
|
Calendar.MINUTE,
|
|
Calendar.SECOND
|
|
};
|
|
String[] names = new String[]
|
|
{
|
|
Util.i18n("year"),
|
|
Util.i18n("years"),
|
|
Util.i18n("month"),
|
|
Util.i18n("months"),
|
|
Util.i18n("day"),
|
|
Util.i18n("days"),
|
|
Util.i18n("hour"),
|
|
Util.i18n("hours"),
|
|
Util.i18n("minute"),
|
|
Util.i18n("minutes"),
|
|
Util.i18n("second"),
|
|
Util.i18n("seconds")
|
|
};
|
|
for (int i = 0; i < types.length; i++)
|
|
{
|
|
int diff = dateDiff(types[i], fromDate, toDate, future);
|
|
if (diff > 0)
|
|
{
|
|
sb.append(" ").append(diff).append(" ").append(names[i * 2 + (diff > 1 ? 1 : 0)]);
|
|
}
|
|
}
|
|
if (sb.length() == 0)
|
|
{
|
|
return "now";
|
|
}
|
|
return sb.toString();
|
|
}
|
|
|
|
private static int dateDiff(int type, Calendar fromDate, Calendar toDate, boolean future)
|
|
{
|
|
int diff = 0;
|
|
long savedDate = fromDate.getTimeInMillis();
|
|
while ((future && !fromDate.after(toDate)) || (!future && !fromDate.before(toDate)))
|
|
{
|
|
savedDate = fromDate.getTimeInMillis();
|
|
fromDate.add(type, future ? 1 : -1);
|
|
diff++;
|
|
}
|
|
diff--;
|
|
fromDate.setTimeInMillis(savedDate);
|
|
return diff;
|
|
}
|
|
|
|
public static long parseDateDiff(String time, boolean future) throws Exception
|
|
{
|
|
Pattern timePattern = Pattern.compile(
|
|
"(?:([0-9]+)\\s*y[a-z]*[,\\s]*)?"
|
|
+ "(?:([0-9]+)\\s*mo[a-z]*[,\\s]*)?"
|
|
+ "(?:([0-9]+)\\s*w[a-z]*[,\\s]*)?"
|
|
+ "(?:([0-9]+)\\s*d[a-z]*[,\\s]*)?"
|
|
+ "(?:([0-9]+)\\s*h[a-z]*[,\\s]*)?"
|
|
+ "(?:([0-9]+)\\s*m[a-z]*[,\\s]*)?"
|
|
+ "(?:([0-9]+)\\s*(?:s[a-z]*)?)?", Pattern.CASE_INSENSITIVE);
|
|
Matcher m = timePattern.matcher(time);
|
|
int years = 0;
|
|
int months = 0;
|
|
int weeks = 0;
|
|
int days = 0;
|
|
int hours = 0;
|
|
int minutes = 0;
|
|
int seconds = 0;
|
|
boolean found = false;
|
|
while (m.find())
|
|
{
|
|
if (m.group() == null || m.group().isEmpty())
|
|
{
|
|
continue;
|
|
}
|
|
for (int i = 0; i < m.groupCount(); i++)
|
|
{
|
|
if (m.group(i) != null && !m.group(i).isEmpty())
|
|
{
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
if (found)
|
|
{
|
|
if (m.group(1) != null && !m.group(1).isEmpty())
|
|
{
|
|
years = Integer.parseInt(m.group(1));
|
|
}
|
|
if (m.group(2) != null && !m.group(2).isEmpty())
|
|
{
|
|
months = Integer.parseInt(m.group(2));
|
|
}
|
|
if (m.group(3) != null && !m.group(3).isEmpty())
|
|
{
|
|
weeks = Integer.parseInt(m.group(3));
|
|
}
|
|
if (m.group(4) != null && !m.group(4).isEmpty())
|
|
{
|
|
days = Integer.parseInt(m.group(4));
|
|
}
|
|
if (m.group(5) != null && !m.group(5).isEmpty())
|
|
{
|
|
hours = Integer.parseInt(m.group(5));
|
|
}
|
|
if (m.group(6) != null && !m.group(6).isEmpty())
|
|
{
|
|
minutes = Integer.parseInt(m.group(6));
|
|
}
|
|
if (m.group(7) != null && !m.group(7).isEmpty())
|
|
{
|
|
seconds = Integer.parseInt(m.group(7));
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
if (!found)
|
|
{
|
|
throw new Exception(Util.i18n("illegalDate"));
|
|
}
|
|
Calendar c = new GregorianCalendar();
|
|
if (years > 0)
|
|
{
|
|
c.add(Calendar.YEAR, years * (future ? 1 : -1));
|
|
}
|
|
if (months > 0)
|
|
{
|
|
c.add(Calendar.MONTH, months * (future ? 1 : -1));
|
|
}
|
|
if (weeks > 0)
|
|
{
|
|
c.add(Calendar.WEEK_OF_YEAR, weeks * (future ? 1 : -1));
|
|
}
|
|
if (days > 0)
|
|
{
|
|
c.add(Calendar.DAY_OF_MONTH, days * (future ? 1 : -1));
|
|
}
|
|
if (hours > 0)
|
|
{
|
|
c.add(Calendar.HOUR_OF_DAY, hours * (future ? 1 : -1));
|
|
}
|
|
if (minutes > 0)
|
|
{
|
|
c.add(Calendar.MINUTE, minutes * (future ? 1 : -1));
|
|
}
|
|
if (seconds > 0)
|
|
{
|
|
c.add(Calendar.SECOND, seconds * (future ? 1 : -1));
|
|
}
|
|
return c.getTimeInMillis();
|
|
}
|
|
|
|
public static Location getSafeDestination(Location loc) throws Exception
|
|
{
|
|
if (loc == null)
|
|
{
|
|
throw new Exception(Util.i18n("destinationNotSet"));
|
|
}
|
|
World world = loc.getWorld();
|
|
double x = Math.floor(loc.getX()) + 0.5;
|
|
double y = Math.floor(loc.getY());
|
|
double z = Math.floor(loc.getZ()) + 0.5;
|
|
|
|
while (isBlockAboveAir(world, x, y, z))
|
|
{
|
|
y -= 1.0D;
|
|
if (y < 0.0D)
|
|
{
|
|
throw new Exception(Util.i18n("holeInFloor"));
|
|
}
|
|
}
|
|
|
|
while (isBlockUnsafe(world, x, y, z))
|
|
{
|
|
y += 1.0D;
|
|
if (y >= 110.0D)
|
|
{
|
|
x += 1.0D;
|
|
break;
|
|
}
|
|
}
|
|
while (isBlockUnsafe(world, x, y, z))
|
|
{
|
|
y -= 1.0D;
|
|
if (y <= 1.0D)
|
|
{
|
|
y = 110.0D;
|
|
x += 1.0D;
|
|
}
|
|
}
|
|
return new Location(world, x, y, z, loc.getYaw(), loc.getPitch());
|
|
}
|
|
|
|
private static boolean isBlockAboveAir(World world, double x, double y, double z)
|
|
{
|
|
return world.getBlockAt((int)Math.floor(x), (int)Math.floor(y - 1.0D), (int)Math.floor(z)).getType() == Material.AIR;
|
|
}
|
|
|
|
public static boolean isBlockUnsafe(World world, double x, double y, double z)
|
|
{
|
|
Block below = world.getBlockAt((int)Math.floor(x), (int)Math.floor(y - 1.0D), (int)Math.floor(z));
|
|
if (below.getType() == Material.LAVA || below.getType() == Material.STATIONARY_LAVA)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if (below.getType() == Material.FIRE)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
if ((world.getBlockAt((int)Math.floor(x), (int)Math.floor(y), (int)Math.floor(z)).getType() != Material.AIR)
|
|
|| (world.getBlockAt((int)Math.floor(x), (int)Math.floor(y + 1.0D), (int)Math.floor(z)).getType() != Material.AIR))
|
|
{
|
|
return true;
|
|
}
|
|
return isBlockAboveAir(world, x, y, z);
|
|
}
|
|
private static DecimalFormat df = new DecimalFormat("#0.00", DecimalFormatSymbols.getInstance(Locale.US));
|
|
|
|
public static String formatCurrency(double value)
|
|
{
|
|
String str = Essentials.getStatic().getSettings().getCurrencySymbol() + df.format(value);
|
|
if (str.endsWith(".00"))
|
|
{
|
|
str = str.substring(0, str.length() - 3);
|
|
}
|
|
return str;
|
|
}
|
|
|
|
public static double roundDouble(double d)
|
|
{
|
|
return Math.round(d * 100.0) / 100.0;
|
|
}
|
|
|
|
public static Locale getCurrentLocale()
|
|
{
|
|
return currentLocale;
|
|
}
|
|
|
|
|
|
private static class ConfigClassLoader extends ClassLoader
|
|
{
|
|
private final File dataFolder;
|
|
private final ClassLoader cl;
|
|
|
|
public ConfigClassLoader(File dataFolder, ClassLoader cl)
|
|
{
|
|
this.dataFolder = dataFolder;
|
|
this.cl = cl;
|
|
}
|
|
|
|
@Override
|
|
public URL getResource(String string)
|
|
{
|
|
File file = new File(dataFolder, string);
|
|
if (file.exists())
|
|
{
|
|
try
|
|
{
|
|
return file.toURI().toURL();
|
|
}
|
|
catch (MalformedURLException ex)
|
|
{
|
|
return cl.getResource(string);
|
|
}
|
|
}
|
|
return cl.getResource(string);
|
|
}
|
|
|
|
@Override
|
|
public synchronized void clearAssertionStatus()
|
|
{
|
|
cl.clearAssertionStatus();
|
|
}
|
|
|
|
@Override
|
|
public InputStream getResourceAsStream(String string)
|
|
{
|
|
File file = new File(dataFolder, string);
|
|
if (file.exists())
|
|
{
|
|
try
|
|
{
|
|
BufferedReader br = new BufferedReader(new FileReader(file));
|
|
String version = br.readLine();
|
|
br.close();
|
|
if (version == null || !version.equals("#version: " + Essentials.getStatic().getDescription().getVersion()))
|
|
{
|
|
logger.log(Level.WARNING, String.format("Translation file %s is not updated for Essentials version. Will use default.", file));
|
|
return cl.getResourceAsStream(string);
|
|
}
|
|
return new FileInputStream(file);
|
|
}
|
|
catch (IOException ex)
|
|
{
|
|
return cl.getResourceAsStream(string);
|
|
}
|
|
}
|
|
return cl.getResourceAsStream(string);
|
|
}
|
|
|
|
@Override
|
|
public Enumeration<URL> getResources(String string) throws IOException
|
|
{
|
|
return cl.getResources(string);
|
|
}
|
|
|
|
@Override
|
|
public Class<?> loadClass(String string) throws ClassNotFoundException
|
|
{
|
|
return cl.loadClass(string);
|
|
}
|
|
|
|
@Override
|
|
public synchronized void setClassAssertionStatus(String string, boolean bln)
|
|
{
|
|
cl.setClassAssertionStatus(string, bln);
|
|
}
|
|
|
|
@Override
|
|
public synchronized void setDefaultAssertionStatus(boolean bln)
|
|
{
|
|
cl.setDefaultAssertionStatus(bln);
|
|
}
|
|
|
|
@Override
|
|
public synchronized void setPackageAssertionStatus(String string, boolean bln)
|
|
{
|
|
cl.setPackageAssertionStatus(string, bln);
|
|
}
|
|
}
|
|
private static final Locale defaultLocale = Locale.getDefault();
|
|
private static Locale currentLocale = defaultLocale;
|
|
private static ResourceBundle bundle = ResourceBundle.getBundle("messages", defaultLocale);
|
|
private static ResourceBundle defaultBundle = ResourceBundle.getBundle("messages", Locale.US);
|
|
|
|
public static String i18n(String string)
|
|
{
|
|
try
|
|
{
|
|
return bundle.getString(string);
|
|
}
|
|
catch (MissingResourceException ex)
|
|
{
|
|
logger.log(Level.WARNING, String.format("Missing translation key \"%s\" in translation file %s", ex.getKey(), bundle.getLocale().toString()), ex);
|
|
return defaultBundle.getString(string);
|
|
}
|
|
}
|
|
|
|
public static String format(String string, Object... objects)
|
|
{
|
|
MessageFormat mf = new MessageFormat(i18n(string));
|
|
return mf.format(objects);
|
|
}
|
|
|
|
public static void updateLocale(String loc, File dataFolder)
|
|
{
|
|
if (loc == null || loc.isEmpty())
|
|
{
|
|
return;
|
|
}
|
|
String[] parts = loc.split("[_\\.]");
|
|
if (parts.length == 1)
|
|
{
|
|
currentLocale = new Locale(parts[0]);
|
|
}
|
|
if (parts.length == 2)
|
|
{
|
|
currentLocale = new Locale(parts[0], parts[1]);
|
|
}
|
|
if (parts.length == 3)
|
|
{
|
|
currentLocale = new Locale(parts[0], parts[1], parts[2]);
|
|
}
|
|
logger.log(Level.INFO, String.format("Using locale %s", currentLocale.toString()));
|
|
bundle = ResourceBundle.getBundle("messages", currentLocale, new ConfigClassLoader(dataFolder, Util.class.getClassLoader()));
|
|
if (!bundle.keySet().containsAll(defaultBundle.keySet()))
|
|
{
|
|
logger.log(Level.WARNING, String.format("Translation file %s does not contain all translation keys.", currentLocale.toString()));
|
|
}
|
|
}
|
|
}
|