2023-04-11 13:25:45 +02:00
|
|
|
package net.knarcraft.dropper.config;
|
|
|
|
|
|
|
|
import net.knarcraft.dropper.Dropper;
|
|
|
|
import org.bukkit.Bukkit;
|
|
|
|
import org.bukkit.Material;
|
|
|
|
import org.bukkit.NamespacedKey;
|
|
|
|
import org.bukkit.Tag;
|
|
|
|
import org.bukkit.configuration.file.FileConfiguration;
|
|
|
|
|
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.HashSet;
|
|
|
|
import java.util.List;
|
|
|
|
import java.util.Set;
|
|
|
|
import java.util.logging.Level;
|
|
|
|
import java.util.regex.Matcher;
|
|
|
|
import java.util.regex.Pattern;
|
|
|
|
|
2023-04-11 20:04:04 +02:00
|
|
|
/**
|
|
|
|
* The configuration keeping track of the player's current configuration
|
|
|
|
*/
|
2023-04-11 13:25:45 +02:00
|
|
|
public class DropperConfiguration {
|
|
|
|
|
2023-04-11 20:04:04 +02:00
|
|
|
private FileConfiguration configuration;
|
|
|
|
private final static String rootNode = "dropper.";
|
2023-04-11 13:25:45 +02:00
|
|
|
|
|
|
|
private double verticalVelocity;
|
|
|
|
private float horizontalVelocity;
|
|
|
|
private int randomlyInvertedTimer;
|
|
|
|
private boolean mustDoGroupedInSequence;
|
|
|
|
private boolean ignoreRecordsUntilGroupBeatenOnce;
|
|
|
|
private boolean mustDoNormalModeFirst;
|
|
|
|
private boolean makePlayersInvisible;
|
|
|
|
private boolean disableHitCollision;
|
|
|
|
private double liquidHitBoxDepth;
|
|
|
|
private double solidHitBoxDistance;
|
2023-04-11 20:04:04 +02:00
|
|
|
private boolean blockSneaking;
|
|
|
|
private boolean blockSprinting;
|
2023-04-11 13:25:45 +02:00
|
|
|
private Set<Material> blockWhitelist;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Instantiates a new dropper configuration
|
|
|
|
*
|
|
|
|
* @param configuration <p>The YAML configuration to use internally</p>
|
|
|
|
*/
|
|
|
|
public DropperConfiguration(FileConfiguration configuration) {
|
|
|
|
this.configuration = configuration;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the default vertical velocity
|
|
|
|
*
|
|
|
|
* @return <p>The default vertical velocity</p>
|
|
|
|
*/
|
|
|
|
public double getVerticalVelocity() {
|
|
|
|
return this.verticalVelocity;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the default horizontal velocity
|
|
|
|
*
|
|
|
|
* @return <p>The default horizontal velocity</p>
|
|
|
|
*/
|
|
|
|
public float getHorizontalVelocity() {
|
|
|
|
return this.horizontalVelocity;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the number of seconds before the randomly inverted game-mode toggles
|
|
|
|
*
|
|
|
|
* @return <p>Number of seconds before the inversion toggles</p>
|
|
|
|
*/
|
|
|
|
public int getRandomlyInvertedTimer() {
|
|
|
|
return this.randomlyInvertedTimer;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets whether grouped arenas must be done in the set sequence
|
|
|
|
*
|
|
|
|
* @return <p>Whether grouped arenas must be done in sequence</p>
|
|
|
|
*/
|
|
|
|
public boolean mustDoGroupedInSequence() {
|
|
|
|
return this.mustDoGroupedInSequence;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets whether the normal/default mode must be beaten before playing another game-mode
|
|
|
|
*
|
|
|
|
* @return <p>Whether the normal game-mode must be beaten first</p>
|
|
|
|
*/
|
|
|
|
public boolean mustDoNormalModeFirst() {
|
|
|
|
return this.mustDoNormalModeFirst;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the types of block which should not trigger a loss
|
|
|
|
*
|
|
|
|
* @return <p>The materials that should not trigger a loss</p>
|
|
|
|
*/
|
|
|
|
public Set<Material> getBlockWhitelist() {
|
|
|
|
return new HashSet<>(this.blockWhitelist);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets whether records should be discarded, unless the player has already beaten all arenas in the group
|
|
|
|
*
|
|
|
|
* @return <p>Whether to ignore records on the first play-through</p>
|
|
|
|
*/
|
|
|
|
public boolean ignoreRecordsUntilGroupBeatenOnce() {
|
|
|
|
return this.ignoreRecordsUntilGroupBeatenOnce;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets whether players should be made invisible while in an arena
|
|
|
|
*
|
|
|
|
* @return <p>Whether players should be made invisible</p>
|
|
|
|
*/
|
|
|
|
public boolean makePlayersInvisible() {
|
|
|
|
return this.makePlayersInvisible;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets whether entity hit-collision of players in an arena should be disabled
|
|
|
|
*
|
|
|
|
* @return <p>Whether to disable hit collision</p>
|
|
|
|
*/
|
|
|
|
public boolean disableHitCollision() {
|
|
|
|
return this.disableHitCollision;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the negative depth a player must reach in a liquid block for fail/win detection to trigger
|
|
|
|
*
|
|
|
|
* <p>This decides how far inside a non-solid block the player must go before detection triggers. The closer to -1
|
|
|
|
* it is, the more accurate it will seem to the player, but the likelihood of not detecting the hit increases.</p>
|
|
|
|
*
|
|
|
|
* @return <p>The liquid hit box depth to use</p>
|
|
|
|
*/
|
|
|
|
public double getLiquidHitBoxDepth() {
|
|
|
|
return this.liquidHitBoxDepth;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the positive distance a player must at most be from a block for fail/win detection to trigger
|
|
|
|
*
|
|
|
|
* <p>This decides the distance the player must be from a block below them before a hit triggers. If too low, the
|
|
|
|
* likelihood of detecting the hit decreases, but it won't look like the player hit the block without being near.</p>
|
|
|
|
*
|
|
|
|
* @return <p>The solid hit box distance to use</p>
|
|
|
|
*/
|
|
|
|
public double getSolidHitBoxDistance() {
|
|
|
|
return this.solidHitBoxDistance;
|
|
|
|
}
|
|
|
|
|
2023-04-11 20:04:04 +02:00
|
|
|
/**
|
|
|
|
* Gets whether players trying to sneak while in a dropper arena to increase their downwards speed should be blocked
|
|
|
|
*
|
|
|
|
* @return <p>Whether to block sneak to speed up</p>
|
|
|
|
*/
|
|
|
|
public boolean blockSneaking() {
|
|
|
|
return blockSneaking;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets whether players trying to sprint to improve their horizontal speed while in a dropper arena should be blocked
|
|
|
|
*
|
|
|
|
* @return <p>Whether to block sprint to speed up</p>
|
|
|
|
*/
|
|
|
|
public boolean blockSprinting() {
|
|
|
|
return this.blockSprinting;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Loads all configuration values from disk
|
|
|
|
*
|
|
|
|
* @param configuration <p>The configuration to load</p>
|
|
|
|
*/
|
|
|
|
public void load(FileConfiguration configuration) {
|
|
|
|
this.configuration = configuration;
|
|
|
|
this.load();
|
|
|
|
}
|
|
|
|
|
2023-04-11 13:25:45 +02:00
|
|
|
/**
|
|
|
|
* Loads all configuration values from disk
|
|
|
|
*/
|
|
|
|
public void load() {
|
2023-04-11 20:04:04 +02:00
|
|
|
this.verticalVelocity = configuration.getDouble(rootNode + "verticalVelocity", 1.0);
|
|
|
|
this.horizontalVelocity = (float) configuration.getDouble(rootNode + "horizontalVelocity", 1.0);
|
|
|
|
this.randomlyInvertedTimer = configuration.getInt(rootNode + "randomlyInvertedTimer", 7);
|
|
|
|
this.mustDoGroupedInSequence = configuration.getBoolean(rootNode + "mustDoGroupedInSequence", true);
|
|
|
|
this.ignoreRecordsUntilGroupBeatenOnce = configuration.getBoolean(rootNode + "ignoreRecordsUntilGroupBeatenOnce", false);
|
|
|
|
this.mustDoNormalModeFirst = configuration.getBoolean(rootNode + "mustDoNormalModeFirst", true);
|
|
|
|
this.makePlayersInvisible = configuration.getBoolean(rootNode + "makePlayersInvisible", false);
|
|
|
|
this.disableHitCollision = configuration.getBoolean(rootNode + "disableHitCollision", true);
|
|
|
|
this.liquidHitBoxDepth = configuration.getDouble(rootNode + "liquidHitBoxDepth", -0.8);
|
|
|
|
this.solidHitBoxDistance = configuration.getDouble(rootNode + "solidHitBoxDistance", 0.2);
|
|
|
|
this.blockSprinting = configuration.getBoolean(rootNode + "blockSprinting", true);
|
|
|
|
this.blockSneaking = configuration.getBoolean(rootNode + "blockSneaking", true);
|
2023-04-11 13:55:46 +02:00
|
|
|
sanitizeValues();
|
2023-04-11 13:25:45 +02:00
|
|
|
|
|
|
|
loadBlockWhitelist();
|
|
|
|
}
|
|
|
|
|
2023-04-11 13:55:46 +02:00
|
|
|
/**
|
|
|
|
* Sanitizes configuration values to ensure they are within expected bounds
|
|
|
|
*/
|
|
|
|
private void sanitizeValues() {
|
|
|
|
if (this.liquidHitBoxDepth <= -1 || this.liquidHitBoxDepth > 0) {
|
|
|
|
this.liquidHitBoxDepth = -0.8;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.solidHitBoxDistance <= 0 || this.solidHitBoxDistance > 1) {
|
|
|
|
this.solidHitBoxDistance = 0.2;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.horizontalVelocity > 1 || this.horizontalVelocity <= 0) {
|
|
|
|
this.horizontalVelocity = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.verticalVelocity <= 0 || this.verticalVelocity > 75) {
|
|
|
|
this.verticalVelocity = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.randomlyInvertedTimer <= 0 || this.randomlyInvertedTimer > 3600) {
|
|
|
|
this.randomlyInvertedTimer = 7;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-11 13:25:45 +02:00
|
|
|
/**
|
|
|
|
* Loads the materials specified in the block whitelist
|
|
|
|
*/
|
|
|
|
private void loadBlockWhitelist() {
|
|
|
|
this.blockWhitelist = new HashSet<>();
|
2023-04-11 20:04:04 +02:00
|
|
|
List<?> blockWhitelist = configuration.getList(rootNode + "blockWhitelist", new ArrayList<>());
|
2023-04-11 13:25:45 +02:00
|
|
|
for (Object value : blockWhitelist) {
|
|
|
|
if (!(value instanceof String string)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to parse a material tag first
|
|
|
|
if (parseMaterialTag(string)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to parse a material name
|
|
|
|
Material matched = Material.matchMaterial(string);
|
|
|
|
if (matched != null) {
|
|
|
|
this.blockWhitelist.add(matched);
|
|
|
|
} else {
|
|
|
|
Dropper.log(Level.WARNING, "Unable to parse: " + string);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tries to parse the material tag in the specified material name
|
|
|
|
*
|
|
|
|
* @param materialName <p>The material name that might be a material tag</p>
|
|
|
|
* @return <p>True if a tag was found</p>
|
|
|
|
*/
|
|
|
|
private boolean parseMaterialTag(String materialName) {
|
|
|
|
Pattern pattern = Pattern.compile("^\\+([a-zA-Z_]+)");
|
|
|
|
Matcher matcher = pattern.matcher(materialName);
|
|
|
|
if (matcher.find()) {
|
|
|
|
// The material is a material tag
|
|
|
|
Tag<Material> tag = Bukkit.getTag(Tag.REGISTRY_BLOCKS, NamespacedKey.minecraft(
|
|
|
|
matcher.group(1).toLowerCase()), Material.class);
|
|
|
|
if (tag != null) {
|
|
|
|
this.blockWhitelist.addAll(tag.getValues());
|
|
|
|
} else {
|
|
|
|
Dropper.log(Level.WARNING, "Unable to parse: " + materialName);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-04-11 20:04:04 +02:00
|
|
|
@Override
|
|
|
|
public String toString() {
|
|
|
|
StringBuilder builder = new StringBuilder(
|
|
|
|
"Current configuration:" +
|
|
|
|
"\n" + "Vertical velocity: " + verticalVelocity +
|
|
|
|
"\n" + "Horizontal velocity: " + horizontalVelocity +
|
|
|
|
"\n" + "Randomly inverted timer: " + randomlyInvertedTimer +
|
|
|
|
"\n" + "Must do groups in sequence: " + mustDoGroupedInSequence +
|
|
|
|
"\n" + "Ignore records until group beaten once: " + ignoreRecordsUntilGroupBeatenOnce +
|
|
|
|
"\n" + "Must do normal mode first: " + mustDoNormalModeFirst +
|
|
|
|
"\n" + "Make players invisible: " + makePlayersInvisible +
|
|
|
|
"\n" + "Disable hit collision: " + disableHitCollision +
|
|
|
|
"\n" + "Liquid hit box depth: " + liquidHitBoxDepth +
|
|
|
|
"\n" + "Solid hit box distance: " + solidHitBoxDistance +
|
|
|
|
"\n" + "Block whitelist: ");
|
|
|
|
for (Material material : blockWhitelist) {
|
|
|
|
builder.append("\n - ").append(material.name());
|
|
|
|
}
|
|
|
|
return builder.toString();
|
|
|
|
}
|
|
|
|
|
2023-04-11 13:25:45 +02:00
|
|
|
}
|