mirror of
https://github.com/BentoBoxWorld/Level.git
synced 2025-02-16 12:31:22 +01:00
Release 2.12.0 (#301)
* Version 2.7.1 * Version 2.7.2 * Use Java 9's takeWhile * Added placeholder %Level_[gamemode]_rank_value Fixes https://github.com/BentoBoxWorld/Level/issues/228 * No save on disable (#231) * Release 2.6.4 * Remove saving to database on disable. https://github.com/BentoBoxWorld/Level/issues/229 First, the top ten tables are never actually used or loaded. They are created in memory by loading the island levels. So there is no reason to keep saving them. Second, the island level data is saved every time it is changed, so there is no need to save all of the cache on exit. * Fixes tests * Rosestacker (#232) * Add support for RoseStacker 1.3.0 * Made plugin a Pladdon. * Version 2.8.0 * Added new placeholders %Level_%gamemode%_top_island_name_%rank% - lists the island name %Level_%gamemode%_top_island_members_%rank% - a comma separated list of team members https://github.com/BentoBoxWorld/Level/issues/224 https://github.com/BentoBoxWorld/Level/issues/211 https://github.com/BentoBoxWorld/Level/issues/132 https://github.com/BentoBoxWorld/Level/issues/107 https://github.com/BentoBoxWorld/Level/issues/105 * Update to BentoBox API 1.18 * Open up modules for testing access. * Back support for BentoBox 1.16.5. * Version 2.8.1 * Speeds up level calculation by doing more chunk scans async. If chests are scanned, then it will take longer because these have to be done sync. https://github.com/BentoBoxWorld/Level/issues/243 * add Vietnamese (#240) * Raw island level placeholder (#241) * Changed IslandLevelCalculator minHeight to world minHeight for negative blocks height support since 1.18. (#246) * Version 2.9.0 * Chinese Translation (#249) * Translate zh-CN.yml via GitLocalize * Translate zh-CN.yml via GitLocalize Co-authored-by: mt-gitlocalize <mt@gitlocalize.com> Co-authored-by: 织梦 <493733933@qq.com> * Translate id.yml via GitLocalize (#250) Co-authored-by: Nathan Adhitya <nathanadhitya@outlook.com> * Translate fr.yml via GitLocalize (#251) Co-authored-by: organizatsiya <organizatsiya.wildguns@gmail.com> * Korean translation (#252) * Translate ko.yml via GitLocalize * Translate ko.yml via GitLocalize Co-authored-by: chickiyeah <ruddls030@naver.com> Co-authored-by: mt-gitlocalize <mt@gitlocalize.com> * German Translation (#253) * Translate de.yml via GitLocalize * Update de.yml Co-authored-by: Rikamo045 <rik.amos.krajinovic@gmail.com> Co-authored-by: tastybento <tastybento@users.noreply.github.com> * Translate hu.yml via GitLocalize (#254) Co-authored-by: András Marczinkó <marczinkoandris@gmail.com> * Version 2.9.1 * Attempt to handle WildStacker spawners * Fix error lon loading id locale * Avoid async chunk snapshotting. Fixes https://github.com/BentoBoxWorld/Level/issues/256 * Update to BentoBox API 1.20. Replace plugin.yml with spigot-annotations. Implement customizable TopLevelPanel. * Fixes some small issues with TopLevelPanel Add Utils class that contains some useful things. * Implement customizable DetailsPanel. Remove old DetailsGUITab due to new implementation. * Fix failing test. * Remove blank file * Added repo for maven plugin snapshots * Implement feature that allows to sort items in detail panel. (#259) Apparently, because it is 2 years old request, it got in a state -> implement or drop. Fixes #192 * Implement calculated value for blocks. (#260) It is ~ value, as calculation formula cannot be applied per block. At least I think so. Part of #192 * Update es.yml (#261) * Implement customizable Values GUI. (#262) This GUI shows value to all items in game. It also shows max limit of blocks, if it is set. Fixes of #192 * Support for AdvancedChests was updated. (#266) * Implements visit/warp actions in top gui Add 2 new actions for island buttons in TOP GUI: - Visit -> allows to visit island, but it requires Visit Addon - Warp -> allows to warp to island, but it requires Warp Addon Requested via Discord. * Fixes a Level addon crash on startup. Level addon crashed at the startup if Visit or Warps addon were not installed. It happened because Level addon main class were implementing Listener interface. To avoid it and fix the crash, I moved migration listener to a separate class. Fixes #2012 * Translate pl.yml via GitLocalize (#269) Co-authored-by: wiktorm12 <wiktorm12@gmail.com> * Translate fr.yml via GitLocalize (#272) Co-authored-by: organizatsiya <organizatsiya.wildguns@gmail.com> * Update to Java 17 * Update Github workflow to Java 17 * Adds %Level_[gamemode]_island_level_max% placeholder This records the lifetime maximum level the island has ever had. Addresses #271 * Only shows Members or higher in the top members placeholder Fixes #267 * Add natural log to level-calc formula parsing Relates to #274 * feat: add island total points + placeholder (#264) * feat: add island total points + placeholder * Update IslandLevels.java * Fix JavaDoc * Translate zh-CN.yml via GitLocalize (#276) Co-authored-by: dawnTak <lanlongxiaode@outlook.com> * Translate nl.yml via GitLocalize (#277) Co-authored-by: DevSolaris <solaris.dev.2002@gmail.com> * Add ${argLine} to get jacoco coverage * Updated Jacoco POM * Add shulker to in chest count (#275) * Sonar Cloud code smell clean up (#278) * Refactor placeholders (#279) * Update ReadMe * Fix Jacoco * Remove unused imports * Remove placeholders from main class Created a separate class for cleaner code and added a test class. * Remove dependency * Add UltimateStacker hook for stacked blocks (#281) * Create plugin.yml (#282) * Create plugin.yml The annotations do not provide the option to define the version dynamically from maven. This should fix that. * Remove Spigot Plugin Annotations * Remove plugin-annotation repo * Updated dependencies * Version 2.10.1 * Add blocks that should be zero by default as they are available on the ocean floor. https://github.com/BentoBoxWorld/Level/issues/284 * Chinese (#288) * Translate zh-CN.yml via GitLocalize * Translate zh-CN.yml via GitLocalize --------- Co-authored-by: Jeansou <bettertreebot@gmail.com> Co-authored-by: dawnTak <lanlongxiaode@outlook.com> * Translate id.yml via GitLocalize (#287) Co-authored-by: Dusty <siapa-yg-mau-diblokir.kfrxp@simplelogin.com> * French (#286) * Translate fr.yml via GitLocalize * Translate fr.yml via GitLocalize * Translate fr.yml via GitLocalize --------- Co-authored-by: gitlocalize-app[bot] <55277160+gitlocalize-app[bot]@users.noreply.github.com> Co-authored-by: organizatsiya <organizatsiya.wildguns@gmail.com> Co-authored-by: Florian CUNY <poslovitch@bentobox.world> * Spanish (#285) * Translate es.yml via GitLocalize * Translate es.yml via GitLocalize --------- Co-authored-by: ChrissTM03 <criisbr193@gmail.com> Co-authored-by: Espan <zcraftyt@gmail.com> * Version 2.11.0 * Update Github Build script * Added distribution required for Github Action * Update Jacoco * Update pom.xml * Add config option to disable plugin hooks (#291) * Update UltimateStacker dependency * Add config option to disable plugin hooks * Use 2.0.0 BentoBox API * Version 2.12.0 * Adds an admin stats command. See #293 * Update tests * Update to BentoBox 2.0.0 API * Added test for Stats command * Try lower version of jacoco * Changed top ten internally to use islands instead of players as keys (#295) Added %[gamemode]_top_weighted_value_x% placeholder https://github.com/BentoBoxWorld/Level/issues/294 * Added more placeholders. #296 Refactored how the top ten maps are structured. In the future, it may be best to have the key be the island. * Translate uk.yml via GitLocalize (#297) Co-authored-by: GIGABAIT <freebitcoin93@gmail.com> * Move to 1.20.4 Refactored the calculator code for clarity. Added Jacoco line to prvent issues with the bigger Material class. * Added comments on the panel templates. * Add protection around unknown blockconfig.yml entries. GRASS>SHORT_GRASS * Uses latest Visit API to avoid chat spam. Fixes #299 (#300) * Added test class for EquationEvaluator * Fix merge error --------- Co-authored-by: Huynh Tien <huynhqtienvtag@gmail.com> Co-authored-by: Rubén <44579213+Rubenicos@users.noreply.github.com> Co-authored-by: Pierre Dedrie <Pirgosth74@gmail.com> Co-authored-by: gitlocalize-app[bot] <55277160+gitlocalize-app[bot]@users.noreply.github.com> Co-authored-by: mt-gitlocalize <mt@gitlocalize.com> Co-authored-by: 织梦 <493733933@qq.com> Co-authored-by: Nathan Adhitya <nathanadhitya@outlook.com> Co-authored-by: organizatsiya <organizatsiya.wildguns@gmail.com> Co-authored-by: chickiyeah <ruddls030@naver.com> Co-authored-by: Rikamo045 <rik.amos.krajinovic@gmail.com> Co-authored-by: András Marczinkó <marczinkoandris@gmail.com> Co-authored-by: BONNe <bonne@bonne.id.lv> Co-authored-by: KrazyxWolf <68208993+KrazyxWolf@users.noreply.github.com> Co-authored-by: DeadSilenceIV <Barreto-h2@hotmail.com> Co-authored-by: wiktorm12 <wiktorm12@gmail.com> Co-authored-by: evlad <emmanuelvlad@gmail.com> Co-authored-by: dawnTak <lanlongxiaode@outlook.com> Co-authored-by: DevSolaris <solaris.dev.2002@gmail.com> Co-authored-by: DevSolaris <105156235+DevSolaris@users.noreply.github.com> Co-authored-by: ceze88 <dev.ceze@gmail.com> Co-authored-by: Jeansou <bettertreebot@gmail.com> Co-authored-by: Dusty <siapa-yg-mau-diblokir.kfrxp@simplelogin.com> Co-authored-by: Florian CUNY <poslovitch@bentobox.world> Co-authored-by: ChrissTM03 <criisbr193@gmail.com> Co-authored-by: Espan <zcraftyt@gmail.com> Co-authored-by: PapiCapi <49530141+PapiCapi@users.noreply.github.com> Co-authored-by: GIGABAIT <freebitcoin93@gmail.com>
This commit is contained in:
parent
f87603de83
commit
3061e80097
9
.github/workflows/build.yml
vendored
9
.github/workflows/build.yml
vendored
@ -11,21 +11,22 @@ jobs:
|
|||||||
name: Build
|
name: Build
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v3
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0 # Shallow clones should be disabled for a better relevancy of analysis
|
fetch-depth: 0 # Shallow clones should be disabled for a better relevancy of analysis
|
||||||
- name: Set up JDK 17
|
- name: Set up JDK 17
|
||||||
uses: actions/setup-java@v1
|
uses: actions/setup-java@v3
|
||||||
with:
|
with:
|
||||||
|
distribution: 'adopt'
|
||||||
java-version: 17
|
java-version: 17
|
||||||
- name: Cache SonarCloud packages
|
- name: Cache SonarCloud packages
|
||||||
uses: actions/cache@v1
|
uses: actions/cache@v3
|
||||||
with:
|
with:
|
||||||
path: ~/.sonar/cache
|
path: ~/.sonar/cache
|
||||||
key: ${{ runner.os }}-sonar
|
key: ${{ runner.os }}-sonar
|
||||||
restore-keys: ${{ runner.os }}-sonar
|
restore-keys: ${{ runner.os }}-sonar
|
||||||
- name: Cache Maven packages
|
- name: Cache Maven packages
|
||||||
uses: actions/cache@v1
|
uses: actions/cache@v3
|
||||||
with:
|
with:
|
||||||
path: ~/.m2
|
path: ~/.m2
|
||||||
key: ${{ runner.os }}-m2-${{ hashFiles('**/pom.xml') }}
|
key: ${{ runner.os }}-m2-${{ hashFiles('**/pom.xml') }}
|
||||||
|
14
pom.xml
14
pom.xml
@ -58,12 +58,12 @@
|
|||||||
<!-- Non-minecraft related dependencies -->
|
<!-- Non-minecraft related dependencies -->
|
||||||
<powermock.version>2.0.9</powermock.version>
|
<powermock.version>2.0.9</powermock.version>
|
||||||
<!-- More visible way how to change dependency versions -->
|
<!-- More visible way how to change dependency versions -->
|
||||||
<spigot.version>1.19.4-R0.1-SNAPSHOT</spigot.version>
|
<spigot.version>1.20.4-R0.1-SNAPSHOT</spigot.version>
|
||||||
<bentobox.version>1.23.0</bentobox.version>
|
<bentobox.version>2.0.0-SNAPSHOT</bentobox.version>
|
||||||
<!-- Warps addon version -->
|
<!-- Warps addon version -->
|
||||||
<warps.version>1.12.0</warps.version>
|
<warps.version>1.12.0</warps.version>
|
||||||
<!-- Visit addon version -->
|
<!-- Visit addon version -->
|
||||||
<visit.version>1.4.0</visit.version>
|
<visit.version>1.6.0</visit.version>
|
||||||
<!-- Panel Utils version -->
|
<!-- Panel Utils version -->
|
||||||
<panelutils.version>1.1.0</panelutils.version>
|
<panelutils.version>1.1.0</panelutils.version>
|
||||||
<!-- Revision variable removes warning about dynamic version -->
|
<!-- Revision variable removes warning about dynamic version -->
|
||||||
@ -71,7 +71,7 @@
|
|||||||
<!-- Do not change unless you want different name for local builds. -->
|
<!-- Do not change unless you want different name for local builds. -->
|
||||||
<build.number>-LOCAL</build.number>
|
<build.number>-LOCAL</build.number>
|
||||||
<!-- This allows to change between versions. -->
|
<!-- This allows to change between versions. -->
|
||||||
<build.version>2.11.0</build.version>
|
<build.version>2.12.0</build.version>
|
||||||
<sonar.projectKey>BentoBoxWorld_Level</sonar.projectKey>
|
<sonar.projectKey>BentoBoxWorld_Level</sonar.projectKey>
|
||||||
<sonar.organization>bentobox-world</sonar.organization>
|
<sonar.organization>bentobox-world</sonar.organization>
|
||||||
<sonar.host.url>https://sonarcloud.io</sonar.host.url>
|
<sonar.host.url>https://sonarcloud.io</sonar.host.url>
|
||||||
@ -237,7 +237,7 @@
|
|||||||
<dependency>
|
<dependency>
|
||||||
<groupId>com.songoda</groupId>
|
<groupId>com.songoda</groupId>
|
||||||
<artifactId>UltimateStacker</artifactId>
|
<artifactId>UltimateStacker</artifactId>
|
||||||
<version>2.3.3</version>
|
<version>2.4.0</version>
|
||||||
<scope>provided</scope>
|
<scope>provided</scope>
|
||||||
</dependency>
|
</dependency>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
@ -403,13 +403,15 @@
|
|||||||
<plugin>
|
<plugin>
|
||||||
<groupId>org.jacoco</groupId>
|
<groupId>org.jacoco</groupId>
|
||||||
<artifactId>jacoco-maven-plugin</artifactId>
|
<artifactId>jacoco-maven-plugin</artifactId>
|
||||||
<version>0.8.7</version>
|
<version>0.8.10</version>
|
||||||
<configuration>
|
<configuration>
|
||||||
<append>true</append>
|
<append>true</append>
|
||||||
<excludes>
|
<excludes>
|
||||||
<!-- This is required to prevent Jacoco from adding
|
<!-- This is required to prevent Jacoco from adding
|
||||||
synthetic fields to a JavaBean class (causes errors in testing) -->
|
synthetic fields to a JavaBean class (causes errors in testing) -->
|
||||||
<exclude>**/*Names*</exclude>
|
<exclude>**/*Names*</exclude>
|
||||||
|
<!-- Prevents the Material is too large to mock error -->
|
||||||
|
<exclude>org/bukkit/Material*</exclude>
|
||||||
</excludes>
|
</excludes>
|
||||||
</configuration>
|
</configuration>
|
||||||
<executions>
|
<executions>
|
||||||
|
@ -24,6 +24,7 @@ import world.bentobox.level.calculators.Pipeliner;
|
|||||||
import world.bentobox.level.commands.AdminLevelCommand;
|
import world.bentobox.level.commands.AdminLevelCommand;
|
||||||
import world.bentobox.level.commands.AdminLevelStatusCommand;
|
import world.bentobox.level.commands.AdminLevelStatusCommand;
|
||||||
import world.bentobox.level.commands.AdminSetInitialLevelCommand;
|
import world.bentobox.level.commands.AdminSetInitialLevelCommand;
|
||||||
|
import world.bentobox.level.commands.AdminStatsCommand;
|
||||||
import world.bentobox.level.commands.AdminTopCommand;
|
import world.bentobox.level.commands.AdminTopCommand;
|
||||||
import world.bentobox.level.commands.IslandLevelCommand;
|
import world.bentobox.level.commands.IslandLevelCommand;
|
||||||
import world.bentobox.level.commands.IslandTopCommand;
|
import world.bentobox.level.commands.IslandTopCommand;
|
||||||
@ -39,7 +40,6 @@ import world.bentobox.level.requests.TopTenRequestHandler;
|
|||||||
import world.bentobox.visit.VisitAddon;
|
import world.bentobox.visit.VisitAddon;
|
||||||
import world.bentobox.warps.Warp;
|
import world.bentobox.warps.Warp;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author tastybento
|
* @author tastybento
|
||||||
*
|
*
|
||||||
@ -71,7 +71,6 @@ public class Level extends Addon {
|
|||||||
*/
|
*/
|
||||||
private VisitAddon visitHook;
|
private VisitAddon visitHook;
|
||||||
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void onLoad() {
|
public void onLoad() {
|
||||||
// Save the default config from config.yml
|
// Save the default config from config.yml
|
||||||
@ -112,8 +111,7 @@ public class Level extends Addon {
|
|||||||
// Register commands for GameModes
|
// Register commands for GameModes
|
||||||
registeredGameModes.clear();
|
registeredGameModes.clear();
|
||||||
getPlugin().getAddonsManager().getGameModeAddons().stream()
|
getPlugin().getAddonsManager().getGameModeAddons().stream()
|
||||||
.filter(gm -> !settings.getGameModes().contains(gm.getDescription().getName()))
|
.filter(gm -> !settings.getGameModes().contains(gm.getDescription().getName())).forEach(gm -> {
|
||||||
.forEach(gm -> {
|
|
||||||
log("Level hooking into " + gm.getDescription().getName());
|
log("Level hooking into " + gm.getDescription().getName());
|
||||||
registerCommands(gm);
|
registerCommands(gm);
|
||||||
new PlaceholderManager(this).registerPlaceholders(gm);
|
new PlaceholderManager(this).registerPlaceholders(gm);
|
||||||
@ -126,11 +124,15 @@ public class Level extends Addon {
|
|||||||
// Check if WildStackers is enabled on the server
|
// Check if WildStackers is enabled on the server
|
||||||
// I only added support for counting blocks into the island level
|
// I only added support for counting blocks into the island level
|
||||||
// Someone else can PR if they want spawners added to the Leveling system :)
|
// Someone else can PR if they want spawners added to the Leveling system :)
|
||||||
|
if (!settings.getDisabledPluginHooks().contains("WildStacker")) {
|
||||||
stackersEnabled = Bukkit.getPluginManager().isPluginEnabled("WildStacker");
|
stackersEnabled = Bukkit.getPluginManager().isPluginEnabled("WildStacker");
|
||||||
if (stackersEnabled) {
|
if (stackersEnabled) {
|
||||||
log("Hooked into WildStackers.");
|
log("Hooked into WildStackers.");
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Check if AdvancedChests is enabled on the server
|
// Check if AdvancedChests is enabled on the server
|
||||||
|
if (!settings.getDisabledPluginHooks().contains("AdvancedChests")) {
|
||||||
Plugin advChest = Bukkit.getPluginManager().getPlugin("AdvancedChests");
|
Plugin advChest = Bukkit.getPluginManager().getPlugin("AdvancedChests");
|
||||||
advChestEnabled = advChest != null;
|
advChestEnabled = advChest != null;
|
||||||
if (advChestEnabled) {
|
if (advChestEnabled) {
|
||||||
@ -138,58 +140,59 @@ public class Level extends Addon {
|
|||||||
if (compareVersions(advChest.getDescription().getVersion(), "23.0") > 0) {
|
if (compareVersions(advChest.getDescription().getVersion(), "23.0") > 0) {
|
||||||
log("Hooked into AdvancedChests.");
|
log("Hooked into AdvancedChests.");
|
||||||
} else {
|
} else {
|
||||||
logError("Could not hook into AdvancedChests " + advChest.getDescription().getVersion() + " - requires version 23.0 or later");
|
logError("Could not hook into AdvancedChests " + advChest.getDescription().getVersion()
|
||||||
|
+ " - requires version 23.0 or later");
|
||||||
advChestEnabled = false;
|
advChestEnabled = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Check if RoseStackers is enabled
|
// Check if RoseStackers is enabled
|
||||||
|
if (!settings.getDisabledPluginHooks().contains("RoseStacker")) {
|
||||||
roseStackersEnabled = Bukkit.getPluginManager().isPluginEnabled("RoseStacker");
|
roseStackersEnabled = Bukkit.getPluginManager().isPluginEnabled("RoseStacker");
|
||||||
if (roseStackersEnabled) {
|
if (roseStackersEnabled) {
|
||||||
log("Hooked into RoseStackers.");
|
log("Hooked into RoseStackers.");
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Check if UltimateStacker is enabled
|
// Check if UltimateStacker is enabled
|
||||||
|
if (!settings.getDisabledPluginHooks().contains("UltimateStacker")) {
|
||||||
ultimateStackerEnabled = Bukkit.getPluginManager().isPluginEnabled("UltimateStacker");
|
ultimateStackerEnabled = Bukkit.getPluginManager().isPluginEnabled("UltimateStacker");
|
||||||
if (ultimateStackerEnabled) {
|
if (ultimateStackerEnabled) {
|
||||||
log("Hooked into UltimateStacker.");
|
log("Hooked into UltimateStacker.");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void allLoaded()
|
public void allLoaded() {
|
||||||
{
|
|
||||||
super.allLoaded();
|
super.allLoaded();
|
||||||
|
|
||||||
if (this.isEnabled())
|
if (this.isEnabled()) {
|
||||||
{
|
|
||||||
this.hookExtensions();
|
this.hookExtensions();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method tries to hook into addons and plugins
|
* This method tries to hook into addons and plugins
|
||||||
*/
|
*/
|
||||||
private void hookExtensions()
|
private void hookExtensions() {
|
||||||
{
|
|
||||||
// Try to find Visit addon and if it does not exist, display a warning
|
// Try to find Visit addon and if it does not exist, display a warning
|
||||||
this.getAddonByName("Visit").ifPresentOrElse(addon ->
|
this.getAddonByName("Visit").ifPresentOrElse(addon -> {
|
||||||
{
|
|
||||||
this.visitHook = (VisitAddon) addon;
|
this.visitHook = (VisitAddon) addon;
|
||||||
this.log("Level Addon hooked into Visit addon.");
|
this.log("Level Addon hooked into Visit addon.");
|
||||||
}, () -> this.visitHook = null);
|
}, () -> this.visitHook = null);
|
||||||
|
|
||||||
// Try to find Warps addon and if it does not exist, display a warning
|
// Try to find Warps addon and if it does not exist, display a warning
|
||||||
this.getAddonByName("Warps").ifPresentOrElse(addon ->
|
this.getAddonByName("Warps").ifPresentOrElse(addon -> {
|
||||||
{
|
|
||||||
this.warpHook = (Warp) addon;
|
this.warpHook = (Warp) addon;
|
||||||
this.log("Level Addon hooked into Warps addon.");
|
this.log("Level Addon hooked into Warps addon.");
|
||||||
}, () -> this.warpHook = null);
|
}, () -> this.warpHook = null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Compares versions
|
* Compares versions
|
||||||
|
*
|
||||||
* @param version1 version 1
|
* @param version1 version 1
|
||||||
* @param version2 version 2
|
* @param version2 version 2
|
||||||
* @return {@code <0 if version 1 is older than version 2, =0 if the same, >0 if version 1 is newer than version 2}
|
* @return {@code <0 if version 1 is older than version 2, =0 if the same, >0 if version 1 is newer than version 2}
|
||||||
@ -221,6 +224,7 @@ public class Level extends Addon {
|
|||||||
if (getSettings().isZeroNewIslandLevels()) {
|
if (getSettings().isZeroNewIslandLevels()) {
|
||||||
new AdminSetInitialLevelCommand(this, adminCommand);
|
new AdminSetInitialLevelCommand(this, adminCommand);
|
||||||
}
|
}
|
||||||
|
new AdminStatsCommand(this, adminCommand);
|
||||||
});
|
});
|
||||||
gm.getPlayerCommand().ifPresent(playerCmd -> {
|
gm.getPlayerCommand().ifPresent(playerCmd -> {
|
||||||
new IslandLevelCommand(this, playerCmd);
|
new IslandLevelCommand(this, playerCmd);
|
||||||
@ -253,7 +257,6 @@ public class Level extends Addon {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return the blockConfig
|
* @return the blockConfig
|
||||||
*/
|
*/
|
||||||
@ -284,6 +287,7 @@ public class Level extends Addon {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Set the config settings - used for tests only
|
* Set the config settings - used for tests only
|
||||||
|
*
|
||||||
* @param configSettings - config settings
|
* @param configSettings - config settings
|
||||||
*/
|
*/
|
||||||
void setSettings(ConfigSettings configSettings) {
|
void setSettings(ConfigSettings configSettings) {
|
||||||
@ -307,6 +311,7 @@ public class Level extends Addon {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Get level from cache for a player.
|
* Get level from cache for a player.
|
||||||
|
*
|
||||||
* @param targetPlayer - target player UUID
|
* @param targetPlayer - target player UUID
|
||||||
* @return Level of player or zero if player is unknown or UUID is null
|
* @return Level of player or zero if player is unknown or UUID is null
|
||||||
*/
|
*/
|
||||||
@ -316,6 +321,7 @@ public class Level extends Addon {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets the player's level to a value
|
* Sets the player's level to a value
|
||||||
|
*
|
||||||
* @param world - world
|
* @param world - world
|
||||||
* @param targetPlayer - target player
|
* @param targetPlayer - target player
|
||||||
* @param level - level
|
* @param level - level
|
||||||
@ -326,6 +332,7 @@ public class Level extends Addon {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Zeros the initial island level
|
* Zeros the initial island level
|
||||||
|
*
|
||||||
* @param island - island
|
* @param island - island
|
||||||
* @param level - initial calculated island level
|
* @param level - initial calculated island level
|
||||||
*/
|
*/
|
||||||
@ -335,6 +342,7 @@ public class Level extends Addon {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the initial island level
|
* Get the initial island level
|
||||||
|
*
|
||||||
* @param island - island
|
* @param island - island
|
||||||
* @return level or 0 by default
|
* @return level or 0 by default
|
||||||
*/
|
*/
|
||||||
@ -344,19 +352,23 @@ public class Level extends Addon {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Calculates a user's island
|
* Calculates a user's island
|
||||||
|
*
|
||||||
* @param world - the world where this island is
|
* @param world - the world where this island is
|
||||||
* @param user - not used! See depecration message
|
* @param user - not used! See depecration message
|
||||||
* @param playerUUID - the target island member's UUID
|
* @param playerUUID - the target island member's UUID
|
||||||
* @deprecated Do not use this anymore. Use getManager().calculateLevel(playerUUID, island)
|
* @deprecated Do not use this anymore. Use
|
||||||
|
* getManager().calculateLevel(playerUUID, island)
|
||||||
*/
|
*/
|
||||||
@Deprecated(since = "2.3.0", forRemoval = true)
|
@Deprecated(since = "2.3.0", forRemoval = true)
|
||||||
public void calculateIslandLevel(World world, @Nullable User user, @NonNull UUID playerUUID) {
|
public void calculateIslandLevel(World world, @Nullable User user, @NonNull UUID playerUUID) {
|
||||||
Island island = getIslands().getIsland(world, playerUUID);
|
Island island = getIslands().getIsland(world, playerUUID);
|
||||||
if (island != null) getManager().calculateLevel(playerUUID, island);
|
if (island != null)
|
||||||
|
getManager().calculateLevel(playerUUID, island);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Provide the levels data for the target player
|
* Provide the levels data for the target player
|
||||||
|
*
|
||||||
* @param targetPlayer - UUID of target player
|
* @param targetPlayer - UUID of target player
|
||||||
* @return LevelsData object or null if not found. Only island levels are set!
|
* @return LevelsData object or null if not found. Only island levels are set!
|
||||||
* @deprecated Do not use this anymore. Use {@link #getIslandLevel(World, UUID)}
|
* @deprecated Do not use this anymore. Use {@link #getIslandLevel(World, UUID)}
|
||||||
@ -365,8 +377,7 @@ public class Level extends Addon {
|
|||||||
public LevelsData getLevelsData(UUID targetPlayer) {
|
public LevelsData getLevelsData(UUID targetPlayer) {
|
||||||
LevelsData ld = new LevelsData(targetPlayer);
|
LevelsData ld = new LevelsData(targetPlayer);
|
||||||
getPlugin().getAddonsManager().getGameModeAddons().stream()
|
getPlugin().getAddonsManager().getGameModeAddons().stream()
|
||||||
.filter(gm -> !settings.getGameModes().contains(gm.getDescription().getName()))
|
.filter(gm -> !settings.getGameModes().contains(gm.getDescription().getName())).forEach(gm -> {
|
||||||
.forEach(gm -> {
|
|
||||||
if (getSettings().isZeroNewIslandLevels()) {
|
if (getSettings().isZeroNewIslandLevels()) {
|
||||||
Island island = getIslands().getIsland(gm.getOverWorld(), targetPlayer);
|
Island island = getIslands().getIsland(gm.getOverWorld(), targetPlayer);
|
||||||
if (island != null) {
|
if (island != null) {
|
||||||
@ -387,6 +398,7 @@ public class Level extends Addon {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if Level addon is active in game mode
|
* Check if Level addon is active in game mode
|
||||||
|
*
|
||||||
* @param gm Game Mode Addon
|
* @param gm Game Mode Addon
|
||||||
* @return true if active, false if not
|
* @return true if active, false if not
|
||||||
*/
|
*/
|
||||||
@ -396,6 +408,7 @@ public class Level extends Addon {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks if Level addon is active in world
|
* Checks if Level addon is active in world
|
||||||
|
*
|
||||||
* @param world world
|
* @param world world
|
||||||
* @return true if active, false if not
|
* @return true if active, false if not
|
||||||
*/
|
*/
|
||||||
@ -422,8 +435,7 @@ public class Level extends Addon {
|
|||||||
*
|
*
|
||||||
* @return {@code Visit} of this object, {@code null} otherwise.
|
* @return {@code Visit} of this object, {@code null} otherwise.
|
||||||
*/
|
*/
|
||||||
public VisitAddon getVisitHook()
|
public VisitAddon getVisitHook() {
|
||||||
{
|
|
||||||
return this.visitHook;
|
return this.visitHook;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -432,8 +444,8 @@ public class Level extends Addon {
|
|||||||
*
|
*
|
||||||
* @return {@code Warp} of this object, {@code null} otherwise.
|
* @return {@code Warp} of this object, {@code null} otherwise.
|
||||||
*/
|
*/
|
||||||
public Warp getWarpHook()
|
public Warp getWarpHook() {
|
||||||
{
|
|
||||||
return this.warpHook;
|
return this.warpHook;
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -2,6 +2,7 @@ package world.bentobox.level;
|
|||||||
|
|
||||||
import java.math.BigInteger;
|
import java.math.BigInteger;
|
||||||
import java.text.DecimalFormat;
|
import java.text.DecimalFormat;
|
||||||
|
import java.util.AbstractMap;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
import java.util.LinkedHashMap;
|
import java.util.LinkedHashMap;
|
||||||
@ -31,7 +32,6 @@ import world.bentobox.level.objects.IslandLevels;
|
|||||||
import world.bentobox.level.objects.LevelsData;
|
import world.bentobox.level.objects.LevelsData;
|
||||||
import world.bentobox.level.objects.TopTenData;
|
import world.bentobox.level.objects.TopTenData;
|
||||||
|
|
||||||
|
|
||||||
public class LevelsManager {
|
public class LevelsManager {
|
||||||
private static final String INTOPTEN = "intopten";
|
private static final String INTOPTEN = "intopten";
|
||||||
private static final TreeMap<BigInteger, String> LEVELS;
|
private static final TreeMap<BigInteger, String> LEVELS;
|
||||||
@ -53,7 +53,6 @@ public class LevelsManager {
|
|||||||
// Top ten lists
|
// Top ten lists
|
||||||
private final Map<World, TopTenData> topTenLists;
|
private final Map<World, TopTenData> topTenLists;
|
||||||
|
|
||||||
|
|
||||||
public LevelsManager(Level addon) {
|
public LevelsManager(Level addon) {
|
||||||
this.addon = addon;
|
this.addon = addon;
|
||||||
// Get the BentoBox database
|
// Get the BentoBox database
|
||||||
@ -76,8 +75,7 @@ public class LevelsManager {
|
|||||||
// World
|
// World
|
||||||
.map(Bukkit::getWorld).filter(Objects::nonNull)
|
.map(Bukkit::getWorld).filter(Objects::nonNull)
|
||||||
// Island
|
// Island
|
||||||
.map(w -> addon.getIslands().getIsland(w, owner)).filter(Objects::nonNull)
|
.map(w -> addon.getIslands().getIsland(w, owner)).filter(Objects::nonNull).forEach(i -> {
|
||||||
.forEach(i -> {
|
|
||||||
// Make new database entry
|
// Make new database entry
|
||||||
World w = i.getWorld();
|
World w = i.getWorld();
|
||||||
IslandLevels il = new IslandLevels(i.getUniqueId());
|
IslandLevels il = new IslandLevels(i.getUniqueId());
|
||||||
@ -99,43 +97,26 @@ public class LevelsManager {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Add a score to the top players list
|
|
||||||
* @param world - world
|
|
||||||
* @param targetPlayer - target player
|
|
||||||
* @param lv - island level
|
|
||||||
*/
|
|
||||||
private void addToTopTen(@NonNull World world, @NonNull UUID targetPlayer, long lv) {
|
|
||||||
// Get top ten
|
|
||||||
Map<UUID, Long> topTen = topTenLists.computeIfAbsent(world, k -> new TopTenData(world)).getTopTen();
|
|
||||||
// Remove this player from the top list no matter what (we'll put them back later if required)
|
|
||||||
topTen.remove(targetPlayer);
|
|
||||||
|
|
||||||
// Get the island
|
|
||||||
Island island = addon.getIslands().getIsland(world, targetPlayer);
|
|
||||||
if (island != null && island.getOwner() != null && hasTopTenPerm(world, island.getOwner())) {
|
|
||||||
// Insert the owner into the top ten
|
|
||||||
topTen.put(island.getOwner(), lv);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Add an island to a top ten
|
* Add an island to a top ten
|
||||||
|
*
|
||||||
* @param island - island to add
|
* @param island - island to add
|
||||||
* @param lv - level
|
* @param lv - level
|
||||||
* @return true if successful, false if not added
|
* @return true if successful, false if not added
|
||||||
*/
|
*/
|
||||||
private boolean addToTopTen(Island island, long lv) {
|
private boolean addToTopTen(Island island, long lv) {
|
||||||
if (island != null && island.getOwner() != null && hasTopTenPerm(island.getWorld(), island.getOwner())) {
|
if (island != null && island.getOwner() != null && hasTopTenPerm(island.getWorld(), island.getOwner())) {
|
||||||
topTenLists.computeIfAbsent(island.getWorld(), k -> new TopTenData(island.getWorld()))
|
topTenLists.computeIfAbsent(island.getWorld(), k -> new TopTenData(island.getWorld())).getTopTen()
|
||||||
.getTopTen().put(island.getOwner(), lv);
|
.put(island.getUniqueId(), lv);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Calculate the island level, set all island member's levels to the result and try to add the owner to the top ten
|
* Calculate the island level, set all island member's levels to the result and
|
||||||
|
* try to add the owner to the top ten
|
||||||
|
*
|
||||||
* @param targetPlayer - uuid of targeted player - owner or team member
|
* @param targetPlayer - uuid of targeted player - owner or team member
|
||||||
* @param island - island to calculate
|
* @param island - island to calculate
|
||||||
* @return completable future with the results of the calculation
|
* @return completable future with the results of the calculation
|
||||||
@ -150,12 +131,13 @@ public class LevelsManager {
|
|||||||
}
|
}
|
||||||
// Add island to the pipeline
|
// Add island to the pipeline
|
||||||
addon.getPipeliner().addIsland(island).thenAccept(r -> {
|
addon.getPipeliner().addIsland(island).thenAccept(r -> {
|
||||||
// Results are irrelevant because the island is unowned or deleted, or IslandLevelCalcEvent is cancelled
|
// Results are irrelevant because the island is unowned or deleted, or
|
||||||
|
// IslandLevelCalcEvent is cancelled
|
||||||
if (r == null || fireIslandLevelCalcEvent(targetPlayer, island, r)) {
|
if (r == null || fireIslandLevelCalcEvent(targetPlayer, island, r)) {
|
||||||
result.complete(null);
|
result.complete(null);
|
||||||
}
|
}
|
||||||
// Save result
|
// Save result
|
||||||
setIslandResults(island.getWorld(), island.getOwner(), r);
|
setIslandResults(island, r);
|
||||||
// Save the island scan details
|
// Save the island scan details
|
||||||
result.complete(r);
|
result.complete(r);
|
||||||
});
|
});
|
||||||
@ -164,6 +146,7 @@ public class LevelsManager {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Fires the IslandLevelCalculatedEvent and returns true if it is canceled
|
* Fires the IslandLevelCalculatedEvent and returns true if it is canceled
|
||||||
|
*
|
||||||
* @param targetPlayer - target player
|
* @param targetPlayer - target player
|
||||||
* @param island - island
|
* @param island - island
|
||||||
* @param results - results set
|
* @param results - results set
|
||||||
@ -173,23 +156,28 @@ public class LevelsManager {
|
|||||||
// Fire post calculation event
|
// Fire post calculation event
|
||||||
IslandLevelCalculatedEvent ilce = new IslandLevelCalculatedEvent(targetPlayer, island, results);
|
IslandLevelCalculatedEvent ilce = new IslandLevelCalculatedEvent(targetPlayer, island, results);
|
||||||
Bukkit.getPluginManager().callEvent(ilce);
|
Bukkit.getPluginManager().callEvent(ilce);
|
||||||
if (ilce.isCancelled()) return true;
|
if (ilce.isCancelled())
|
||||||
|
return true;
|
||||||
// Set the values if they were altered
|
// Set the values if they were altered
|
||||||
results.setLevel((Long) ilce.getKeyValues().getOrDefault("level", results.getLevel()));
|
results.setLevel((Long) ilce.getKeyValues().getOrDefault("level", results.getLevel()));
|
||||||
results.setInitialLevel((Long) ilce.getKeyValues().getOrDefault("initialLevel", results.getInitialLevel()));
|
results.setInitialLevel((Long) ilce.getKeyValues().getOrDefault("initialLevel", results.getInitialLevel()));
|
||||||
results.setDeathHandicap((int) ilce.getKeyValues().getOrDefault("deathHandicap", results.getDeathHandicap()));
|
results.setDeathHandicap((int) ilce.getKeyValues().getOrDefault("deathHandicap", results.getDeathHandicap()));
|
||||||
results.setPointsToNextLevel((Long)ilce.getKeyValues().getOrDefault("pointsToNextLevel", results.getPointsToNextLevel()));
|
results.setPointsToNextLevel(
|
||||||
|
(Long) ilce.getKeyValues().getOrDefault("pointsToNextLevel", results.getPointsToNextLevel()));
|
||||||
results.setTotalPoints((Long) ilce.getKeyValues().getOrDefault("totalPoints", results.getTotalPoints()));
|
results.setTotalPoints((Long) ilce.getKeyValues().getOrDefault("totalPoints", results.getTotalPoints()));
|
||||||
return ((Boolean) ilce.getKeyValues().getOrDefault("isCancelled", false));
|
return ((Boolean) ilce.getKeyValues().getOrDefault("isCancelled", false));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the string representation of the level. May be converted to shorthand notation, e.g., 104556 = 10.5k
|
* Get the string representation of the level. May be converted to shorthand
|
||||||
|
* notation, e.g., 104556 = 10.5k
|
||||||
|
*
|
||||||
* @param lvl - long value to represent
|
* @param lvl - long value to represent
|
||||||
* @return string of the level.
|
* @return string of the level.
|
||||||
*/
|
*/
|
||||||
public String formatLevel(@Nullable Long lvl) {
|
public String formatLevel(@Nullable Long lvl) {
|
||||||
if (lvl == null) return "";
|
if (lvl == null)
|
||||||
|
return "";
|
||||||
String level = String.valueOf(lvl);
|
String level = String.valueOf(lvl);
|
||||||
// Asking for the level of another player
|
// Asking for the level of another player
|
||||||
if (addon.getSettings().isShorthand()) {
|
if (addon.getSettings().isShorthand()) {
|
||||||
@ -202,7 +190,8 @@ public class LevelsManager {
|
|||||||
// 1 527 314 -> 1.5M
|
// 1 527 314 -> 1.5M
|
||||||
// 3 874 130 021 -> 3.8G
|
// 3 874 130 021 -> 3.8G
|
||||||
// 4 002 317 889 -> 4.0T
|
// 4 002 317 889 -> 4.0T
|
||||||
level = new DecimalFormat("#.#").format(levelValue.divide(stage.getKey().divide(THOUSAND)).doubleValue()/1000.0) + stage.getValue();
|
level = new DecimalFormat("#.#").format(
|
||||||
|
levelValue.divide(stage.getKey().divide(THOUSAND)).doubleValue() / 1000.0) + stage.getValue();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return level;
|
return level;
|
||||||
@ -210,6 +199,7 @@ public class LevelsManager {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the initial level of the island. Used to zero island levels
|
* Get the initial level of the island. Used to zero island levels
|
||||||
|
*
|
||||||
* @param island - island
|
* @param island - island
|
||||||
* @return initial level of island
|
* @return initial level of island
|
||||||
*/
|
*/
|
||||||
@ -219,12 +209,15 @@ public class LevelsManager {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Get level of island from cache for a player.
|
* Get level of island from cache for a player.
|
||||||
|
*
|
||||||
* @param world - world where the island is
|
* @param world - world where the island is
|
||||||
* @param targetPlayer - target player UUID
|
* @param targetPlayer - target player UUID
|
||||||
* @return Level of the player's island or zero if player is unknown or UUID is null
|
* @return Level of the player's island or zero if player is unknown or UUID is
|
||||||
|
* null
|
||||||
*/
|
*/
|
||||||
public long getIslandLevel(@NonNull World world, @Nullable UUID targetPlayer) {
|
public long getIslandLevel(@NonNull World world, @Nullable UUID targetPlayer) {
|
||||||
if (targetPlayer == null) return 0L;
|
if (targetPlayer == null)
|
||||||
|
return 0L;
|
||||||
// Get the island
|
// Get the island
|
||||||
Island island = addon.getIslands().getIsland(world, targetPlayer);
|
Island island = addon.getIslands().getIsland(world, targetPlayer);
|
||||||
return island == null ? 0L : getLevelsData(island).getLevel();
|
return island == null ? 0L : getLevelsData(island).getLevel();
|
||||||
@ -232,12 +225,15 @@ public class LevelsManager {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the maximum level ever given to this island
|
* Get the maximum level ever given to this island
|
||||||
|
*
|
||||||
* @param world - world where the island is
|
* @param world - world where the island is
|
||||||
* @param targetPlayer - target player UUID
|
* @param targetPlayer - target player UUID
|
||||||
* @return Max level of the player's island or zero if player is unknown or UUID is null
|
* @return Max level of the player's island or zero if player is unknown or UUID
|
||||||
|
* is null
|
||||||
*/
|
*/
|
||||||
public long getIslandMaxLevel(@NonNull World world, @Nullable UUID targetPlayer) {
|
public long getIslandMaxLevel(@NonNull World world, @Nullable UUID targetPlayer) {
|
||||||
if (targetPlayer == null) return 0L;
|
if (targetPlayer == null)
|
||||||
|
return 0L;
|
||||||
// Get the island
|
// Get the island
|
||||||
Island island = addon.getIslands().getIsland(world, targetPlayer);
|
Island island = addon.getIslands().getIsland(world, targetPlayer);
|
||||||
return island == null ? 0L : getLevelsData(island).getMaxLevel();
|
return island == null ? 0L : getLevelsData(island).getMaxLevel();
|
||||||
@ -245,9 +241,11 @@ public class LevelsManager {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a formatted string of the target player's island level
|
* Returns a formatted string of the target player's island level
|
||||||
|
*
|
||||||
* @param world - world where the island is
|
* @param world - world where the island is
|
||||||
* @param targetPlayer - target player's UUID
|
* @param targetPlayer - target player's UUID
|
||||||
* @return Formatted level of player or zero if player is unknown or UUID is null
|
* @return Formatted level of player or zero if player is unknown or UUID is
|
||||||
|
* null
|
||||||
*/
|
*/
|
||||||
public String getIslandLevelString(@NonNull World world, @Nullable UUID targetPlayer) {
|
public String getIslandLevelString(@NonNull World world, @Nullable UUID targetPlayer) {
|
||||||
return formatLevel(getIslandLevel(world, targetPlayer));
|
return formatLevel(getIslandLevel(world, targetPlayer));
|
||||||
@ -255,6 +253,7 @@ public class LevelsManager {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Load a level data for the island from the cache or database.
|
* Load a level data for the island from the cache or database.
|
||||||
|
*
|
||||||
* @param island - UUID of island
|
* @param island - UUID of island
|
||||||
* @return IslandLevels object
|
* @return IslandLevels object
|
||||||
*/
|
*/
|
||||||
@ -281,72 +280,114 @@ public class LevelsManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the number of points required until the next level since the last level calc
|
* Get the number of points required until the next level since the last level
|
||||||
|
* calc
|
||||||
|
*
|
||||||
* @param world - world where the island is
|
* @param world - world where the island is
|
||||||
* @param targetPlayer - target player UUID
|
* @param targetPlayer - target player UUID
|
||||||
* @return string with the number required or blank if the player is unknown
|
* @return string with the number required or blank if the player is unknown
|
||||||
*/
|
*/
|
||||||
public String getPointsToNextString(@NonNull World world, @Nullable UUID targetPlayer) {
|
public String getPointsToNextString(@NonNull World world, @Nullable UUID targetPlayer) {
|
||||||
if (targetPlayer == null) return "";
|
if (targetPlayer == null)
|
||||||
|
return "";
|
||||||
Island island = addon.getIslands().getIsland(world, targetPlayer);
|
Island island = addon.getIslands().getIsland(world, targetPlayer);
|
||||||
return island == null ? "" : String.valueOf(getLevelsData(island).getPointsToNextLevel());
|
return island == null ? "" : String.valueOf(getLevelsData(island).getPointsToNextLevel());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the top ten for this world. Returns offline players or players with the intopten permission.
|
* Get the weighted top ten for this world. Weighting is based on number of
|
||||||
|
* players per team.
|
||||||
|
*
|
||||||
* @param world - world requested
|
* @param world - world requested
|
||||||
* @param size - size of the top ten
|
* @param size - size of the top ten
|
||||||
* @return sorted top ten map
|
* @return sorted top ten map. The key is the island unique ID
|
||||||
*/
|
*/
|
||||||
@NonNull
|
@NonNull
|
||||||
public Map<UUID, Long> getTopTen(@NonNull World world, int size) {
|
public Map<Island, Long> getWeightedTopTen(@NonNull World world, int size) {
|
||||||
|
createAndCleanRankings(world);
|
||||||
|
Map<Island, Long> weightedTopTen = topTenLists.get(world).getTopTen().entrySet().stream()
|
||||||
|
.map(en -> addon.getIslands().getIslandById(en.getKey()).map(island -> {
|
||||||
|
|
||||||
|
long value = (long) (en.getValue() / (double) Math.max(1, island.getMemberSet().size())); // Calculate
|
||||||
|
// weighted
|
||||||
|
// value
|
||||||
|
return new AbstractMap.SimpleEntry<>(island, value);
|
||||||
|
}).orElse(null)) // Handle islands that do not exist according to this ID - old deleted ones
|
||||||
|
.filter(Objects::nonNull) // Filter out null entries
|
||||||
|
.filter(en -> en.getValue() > 0) // Filter out entries with non-positive values
|
||||||
|
.sorted(Collections.reverseOrder(Map.Entry.comparingByValue())) // Sort in descending order of values
|
||||||
|
.limit(size) // Limit to the top 'size' entries
|
||||||
|
.collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue, (e1, e2) -> e1, // In case of key
|
||||||
|
// collision, choose
|
||||||
|
// the first one
|
||||||
|
LinkedHashMap::new // Preserves the order of entries
|
||||||
|
));
|
||||||
|
|
||||||
|
// Return the unmodifiable map
|
||||||
|
return Collections.unmodifiableMap(weightedTopTen);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the top ten for this world. Returns offline players or players with the
|
||||||
|
* intopten permission.
|
||||||
|
*
|
||||||
|
* @param world - world requested
|
||||||
|
* @param size - size of the top ten
|
||||||
|
* @return sorted top ten map. The key is the island unique ID
|
||||||
|
*/
|
||||||
|
@NonNull
|
||||||
|
public Map<String, Long> getTopTen(@NonNull World world, int size) {
|
||||||
createAndCleanRankings(world);
|
createAndCleanRankings(world);
|
||||||
// Return the sorted map
|
// Return the sorted map
|
||||||
return Collections.unmodifiableMap(topTenLists.get(world).getTopTen().entrySet().stream()
|
return Collections.unmodifiableMap(topTenLists.get(world).getTopTen().entrySet().stream()
|
||||||
.filter(e -> addon.getIslands().isOwner(world, e.getKey()))
|
.filter(l -> l.getValue() > 0).sorted(Collections.reverseOrder(Map.Entry.comparingByValue()))
|
||||||
.filter(l -> l.getValue() > 0)
|
.limit(size)
|
||||||
.sorted(Collections.reverseOrder(Map.Entry.comparingByValue())).limit(size)
|
.collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue, (e1, e2) -> e1, LinkedHashMap::new)));
|
||||||
.collect(Collectors.toMap(
|
|
||||||
Map.Entry::getKey, Map.Entry::getValue, (e1, e2) -> e1, LinkedHashMap::new)));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void createAndCleanRankings(@NonNull World world) {
|
void createAndCleanRankings(@NonNull World world) {
|
||||||
topTenLists.computeIfAbsent(world, TopTenData::new);
|
topTenLists.computeIfAbsent(world, TopTenData::new);
|
||||||
// Remove player from top ten if they are online and do not have the perm
|
// Remove player from top ten if they are online and do not have the perm
|
||||||
topTenLists.get(world).getTopTen().keySet().removeIf(u -> !hasTopTenPerm(world, u));
|
topTenLists.get(world).getTopTen().keySet().removeIf(u -> addon.getIslands().getIslandById(u)
|
||||||
|
.filter(i -> i.getOwner() == null || !hasTopTenPerm(world, i.getOwner())).isPresent());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return the topTenLists
|
* @return the topTenLists
|
||||||
*/
|
*/
|
||||||
protected Map<World, TopTenData> getTopTenLists() {
|
public Map<World, TopTenData> getTopTenLists() {
|
||||||
return topTenLists;
|
return topTenLists;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the rank of the player in the rankings
|
* Get the rank of the player in the rankings
|
||||||
|
*
|
||||||
* @param world - world
|
* @param world - world
|
||||||
* @param uuid - player UUID
|
* @param uuid - player UUID
|
||||||
* @return rank placing - note - placing of 1 means top ranked
|
* @return rank placing - note - placing of 1 means top ranked
|
||||||
*/
|
*/
|
||||||
public int getRank(@NonNull World world, UUID uuid) {
|
public int getRank(@NonNull World world, UUID uuid) {
|
||||||
createAndCleanRankings(world);
|
createAndCleanRankings(world);
|
||||||
Stream<Entry<UUID, Long>> stream = topTenLists.get(world).getTopTen().entrySet().stream()
|
Stream<Entry<String, Long>> stream = topTenLists.get(world).getTopTen().entrySet().stream()
|
||||||
.filter(e -> addon.getIslands().isOwner(world, e.getKey()))
|
.filter(l -> l.getValue() > 0).sorted(Collections.reverseOrder(Map.Entry.comparingByValue()));
|
||||||
.filter(l -> l.getValue() > 0)
|
// Get player's current island
|
||||||
.sorted(Collections.reverseOrder(Map.Entry.comparingByValue()));
|
Island island = addon.getIslands().getIsland(world, uuid);
|
||||||
return (int) (stream.takeWhile(x -> !x.getKey().equals(uuid)).map(Map.Entry::getKey).count() + 1);
|
String id = island == null ? null : island.getUniqueId();
|
||||||
|
return (int) (stream.takeWhile(x -> !x.getKey().equals(id)).map(Map.Entry::getKey).count() + 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks if player has the correct top ten perm to have their level saved
|
* Checks if player has the correct top ten perm to have their level saved
|
||||||
|
*
|
||||||
* @param world
|
* @param world
|
||||||
* @param targetPlayer
|
* @param targetPlayer
|
||||||
* @return true if player has the perm or the player is offline
|
* @return true if player has the perm or the player is offline
|
||||||
*/
|
*/
|
||||||
boolean hasTopTenPerm(@NonNull World world, @NonNull UUID targetPlayer) {
|
boolean hasTopTenPerm(@NonNull World world, @NonNull UUID targetPlayer) {
|
||||||
String permPrefix = addon.getPlugin().getIWM().getPermissionPrefix(world);
|
String permPrefix = addon.getPlugin().getIWM().getPermissionPrefix(world);
|
||||||
return Bukkit.getPlayer(targetPlayer) == null || Bukkit.getPlayer(targetPlayer).hasPermission(permPrefix + INTOPTEN);
|
return Bukkit.getPlayer(targetPlayer) == null
|
||||||
|
|| Bukkit.getPlayer(targetPlayer).hasPermission(permPrefix + INTOPTEN);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -358,7 +399,8 @@ public class LevelsManager {
|
|||||||
addon.log("Generating rankings");
|
addon.log("Generating rankings");
|
||||||
handler.loadObjects().forEach(il -> {
|
handler.loadObjects().forEach(il -> {
|
||||||
if (il.getLevel() > 0) {
|
if (il.getLevel() > 0) {
|
||||||
addon.getIslands().getIslandById(il.getUniqueId()).ifPresent(i -> this.addToTopTen(i, il.getLevel()));
|
addon.getIslands().getIslandById(il.getUniqueId())
|
||||||
|
.ifPresent(i -> this.addToTopTen(i, il.getLevel()));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
topTenLists.keySet().forEach(w -> addon.log("Generated rankings for " + w.getName()));
|
topTenLists.keySet().forEach(w -> addon.log("Generated rankings for " + w.getName()));
|
||||||
@ -366,11 +408,12 @@ public class LevelsManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Removes a player from a world's top ten and removes world from player's level data
|
* Removes an island from a world's top ten
|
||||||
|
*
|
||||||
* @param world - world
|
* @param world - world
|
||||||
* @param uuid - the player's uuid
|
* @param uuid - the island's uuid
|
||||||
*/
|
*/
|
||||||
public void removeEntry(World world, UUID uuid) {
|
public void removeEntry(World world, String uuid) {
|
||||||
if (topTenLists.containsKey(world)) {
|
if (topTenLists.containsKey(world)) {
|
||||||
topTenLists.get(world).getTopTen().remove(uuid);
|
topTenLists.get(world).getTopTen().remove(uuid);
|
||||||
}
|
}
|
||||||
@ -379,19 +422,23 @@ public class LevelsManager {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Set an initial island level
|
* Set an initial island level
|
||||||
|
*
|
||||||
* @param island - the island to set. Must have a non-null world
|
* @param island - the island to set. Must have a non-null world
|
||||||
* @param lv - initial island level
|
* @param lv - initial island level
|
||||||
*/
|
*/
|
||||||
public void setInitialIslandLevel(@NonNull Island island, long lv) {
|
public void setInitialIslandLevel(@NonNull Island island, long lv) {
|
||||||
if (island.getWorld() == null) return;
|
if (island.getWorld() == null)
|
||||||
|
return;
|
||||||
levelsCache.computeIfAbsent(island.getUniqueId(), IslandLevels::new).setInitialLevel(lv);
|
levelsCache.computeIfAbsent(island.getUniqueId(), IslandLevels::new).setInitialLevel(lv);
|
||||||
handler.saveObjectAsync(levelsCache.get(island.getUniqueId()));
|
handler.saveObjectAsync(levelsCache.get(island.getUniqueId()));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set the island level for the owner of the island that targetPlayer is a member
|
* Set the island level for the owner of the island that targetPlayer is a
|
||||||
|
* member
|
||||||
|
*
|
||||||
* @param world - world
|
* @param world - world
|
||||||
* @param targetPlayer - player, may be a team member
|
* @param island - island
|
||||||
* @param lv - level
|
* @param lv - level
|
||||||
*/
|
*/
|
||||||
public void setIslandLevel(@NonNull World world, @NonNull UUID targetPlayer, long lv) {
|
public void setIslandLevel(@NonNull World world, @NonNull UUID targetPlayer, long lv) {
|
||||||
@ -408,21 +455,21 @@ public class LevelsManager {
|
|||||||
}
|
}
|
||||||
handler.saveObjectAsync(levelsCache.get(id));
|
handler.saveObjectAsync(levelsCache.get(id));
|
||||||
// Update TopTen
|
// Update TopTen
|
||||||
addToTopTen(world, targetPlayer, levelsCache.get(id).getLevel());
|
addToTopTen(island, levelsCache.get(id).getLevel());
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set the island level for the owner of the island that targetPlayer is a member
|
* Set the island level for the owner of the island that targetPlayer is a
|
||||||
|
* member
|
||||||
|
*
|
||||||
* @param world - world
|
* @param world - world
|
||||||
* @param owner - owner of the island
|
* @param owner - owner of the island
|
||||||
* @param r - results of the calculation
|
* @param r - results of the calculation
|
||||||
*/
|
*/
|
||||||
private void setIslandResults(World world, @NonNull UUID owner, Results r) {
|
private void setIslandResults(Island island, Results r) {
|
||||||
// Get the island
|
if (island == null)
|
||||||
Island island = addon.getIslands().getIsland(world, owner);
|
return;
|
||||||
if (island == null) return;
|
|
||||||
IslandLevels ld = levelsCache.computeIfAbsent(island.getUniqueId(), IslandLevels::new);
|
IslandLevels ld = levelsCache.computeIfAbsent(island.getUniqueId(), IslandLevels::new);
|
||||||
ld.setLevel(r.getLevel());
|
ld.setLevel(r.getLevel());
|
||||||
ld.setUwCount(Maps.asMap(r.getUwCount().elementSet(), elem -> r.getUwCount().count(elem)));
|
ld.setUwCount(Maps.asMap(r.getUwCount().elementSet(), elem -> r.getUwCount().count(elem)));
|
||||||
@ -432,11 +479,12 @@ public class LevelsManager {
|
|||||||
levelsCache.put(island.getUniqueId(), ld);
|
levelsCache.put(island.getUniqueId(), ld);
|
||||||
handler.saveObjectAsync(ld);
|
handler.saveObjectAsync(ld);
|
||||||
// Update TopTen
|
// Update TopTen
|
||||||
addToTopTen(world, owner, ld.getLevel());
|
addToTopTen(island, ld.getLevel());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Removes island from cache when it is deleted
|
* Removes island from cache when it is deleted
|
||||||
|
*
|
||||||
* @param uniqueId - id of island
|
* @param uniqueId - id of island
|
||||||
*/
|
*/
|
||||||
public void deleteIsland(String uniqueId) {
|
public void deleteIsland(String uniqueId) {
|
||||||
|
@ -2,10 +2,12 @@ package world.bentobox.level;
|
|||||||
|
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
import java.util.Optional;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
import org.bukkit.World;
|
import org.bukkit.World;
|
||||||
|
import org.eclipse.jdt.annotation.Nullable;
|
||||||
|
|
||||||
import world.bentobox.bentobox.BentoBox;
|
import world.bentobox.bentobox.BentoBox;
|
||||||
import world.bentobox.bentobox.api.addons.GameModeAddon;
|
import world.bentobox.bentobox.api.addons.GameModeAddon;
|
||||||
@ -18,6 +20,7 @@ import world.bentobox.level.objects.TopTenData;
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Handles Level placeholders
|
* Handles Level placeholders
|
||||||
|
*
|
||||||
* @author tastybento
|
* @author tastybento
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
@ -32,127 +35,161 @@ public class PlaceholderManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
protected void registerPlaceholders(GameModeAddon gm) {
|
protected void registerPlaceholders(GameModeAddon gm) {
|
||||||
if (plugin.getPlaceholdersManager() == null) return;
|
if (plugin.getPlaceholdersManager() == null)
|
||||||
|
return;
|
||||||
PlaceholdersManager bpm = plugin.getPlaceholdersManager();
|
PlaceholdersManager bpm = plugin.getPlaceholdersManager();
|
||||||
// Island Level
|
// Island Level
|
||||||
bpm.registerPlaceholder(addon,
|
bpm.registerPlaceholder(addon, gm.getDescription().getName().toLowerCase() + "_island_level",
|
||||||
gm.getDescription().getName().toLowerCase() + "_island_level",
|
|
||||||
user -> addon.getManager().getIslandLevelString(gm.getOverWorld(), user.getUniqueId()));
|
user -> addon.getManager().getIslandLevelString(gm.getOverWorld(), user.getUniqueId()));
|
||||||
bpm.registerPlaceholder(addon,
|
// Unformatted island level
|
||||||
gm.getDescription().getName().toLowerCase() + "_island_level_raw",
|
bpm.registerPlaceholder(addon, gm.getDescription().getName().toLowerCase() + "_island_level_raw",
|
||||||
user -> String.valueOf(addon.getManager().getIslandLevel(gm.getOverWorld(), user.getUniqueId())));
|
user -> String.valueOf(addon.getManager().getIslandLevel(gm.getOverWorld(), user.getUniqueId())));
|
||||||
bpm.registerPlaceholder(addon,
|
// Total number of points counted before applying level formula
|
||||||
gm.getDescription().getName().toLowerCase() + "_island_total_points",
|
bpm.registerPlaceholder(addon, gm.getDescription().getName().toLowerCase() + "_island_total_points", user -> {
|
||||||
user -> {
|
|
||||||
IslandLevels data = addon.getManager().getLevelsData(addon.getIslands().getIsland(gm.getOverWorld(), user));
|
IslandLevels data = addon.getManager().getLevelsData(addon.getIslands().getIsland(gm.getOverWorld(), user));
|
||||||
return data.getTotalPoints() + "";
|
return data.getTotalPoints() + "";
|
||||||
});
|
});
|
||||||
|
// Points to the next level for player
|
||||||
bpm.registerPlaceholder(addon,
|
bpm.registerPlaceholder(addon, gm.getDescription().getName().toLowerCase() + "_points_to_next_level",
|
||||||
gm.getDescription().getName().toLowerCase() + "_points_to_next_level",
|
|
||||||
user -> addon.getManager().getPointsToNextString(gm.getOverWorld(), user.getUniqueId()));
|
user -> addon.getManager().getPointsToNextString(gm.getOverWorld(), user.getUniqueId()));
|
||||||
bpm.registerPlaceholder(addon,
|
// Maximum level this island has ever been. Current level maybe lower.
|
||||||
gm.getDescription().getName().toLowerCase() + "_island_level_max",
|
bpm.registerPlaceholder(addon, gm.getDescription().getName().toLowerCase() + "_island_level_max",
|
||||||
user -> String.valueOf(addon.getManager().getIslandMaxLevel(gm.getOverWorld(), user.getUniqueId())));
|
user -> String.valueOf(addon.getManager().getIslandMaxLevel(gm.getOverWorld(), user.getUniqueId())));
|
||||||
|
|
||||||
// Visited Island Level
|
// Visited Island Level
|
||||||
bpm.registerPlaceholder(addon,
|
bpm.registerPlaceholder(addon, gm.getDescription().getName().toLowerCase() + "_visited_island_level",
|
||||||
gm.getDescription().getName().toLowerCase() + "_visited_island_level", user -> getVisitedIslandLevel(gm, user));
|
user -> getVisitedIslandLevel(gm, user));
|
||||||
|
|
||||||
// Register Top Ten Placeholders
|
// Register Top Ten Placeholders
|
||||||
for (int i = 1; i < 11; i++) {
|
for (int i = 1; i < 11; i++) {
|
||||||
final int rank = i;
|
final int rank = i;
|
||||||
// Name
|
// Name
|
||||||
bpm.registerPlaceholder(addon,
|
bpm.registerPlaceholder(addon, gm.getDescription().getName().toLowerCase() + "_top_name_" + i,
|
||||||
gm.getDescription().getName().toLowerCase() + "_top_name_" + i, u -> getRankName(gm.getOverWorld(), rank));
|
u -> getRankName(gm.getOverWorld(), rank, false));
|
||||||
// Island Name
|
// Island Name
|
||||||
bpm.registerPlaceholder(addon,
|
bpm.registerPlaceholder(addon, gm.getDescription().getName().toLowerCase() + "_top_island_name_" + i,
|
||||||
gm.getDescription().getName().toLowerCase() + "_top_island_name_" + i, u -> getRankIslandName(gm.getOverWorld(), rank));
|
u -> getRankIslandName(gm.getOverWorld(), rank, false));
|
||||||
// Members
|
// Members
|
||||||
bpm.registerPlaceholder(addon,
|
bpm.registerPlaceholder(addon, gm.getDescription().getName().toLowerCase() + "_top_members_" + i,
|
||||||
gm.getDescription().getName().toLowerCase() + "_top_members_" + i, u -> getRankMembers(gm.getOverWorld(), rank));
|
u -> getRankMembers(gm.getOverWorld(), rank, false));
|
||||||
// Level
|
// Level
|
||||||
|
bpm.registerPlaceholder(addon, gm.getDescription().getName().toLowerCase() + "_top_value_" + i,
|
||||||
|
u -> getRankLevel(gm.getOverWorld(), rank, false));
|
||||||
|
// Weighted Level Name (Level / number of members)
|
||||||
|
bpm.registerPlaceholder(addon, gm.getDescription().getName().toLowerCase() + "_top_weighted_name_" + i,
|
||||||
|
u -> getRankName(gm.getOverWorld(), rank, true));
|
||||||
|
// Weighted Island Name
|
||||||
bpm.registerPlaceholder(addon,
|
bpm.registerPlaceholder(addon,
|
||||||
gm.getDescription().getName().toLowerCase() + "_top_value_" + i, u -> getRankLevel(gm.getOverWorld(), rank));
|
gm.getDescription().getName().toLowerCase() + "_top_weighted_island_name_" + i,
|
||||||
|
u -> getRankIslandName(gm.getOverWorld(), rank, true));
|
||||||
|
// Weighted Members
|
||||||
|
bpm.registerPlaceholder(addon, gm.getDescription().getName().toLowerCase() + "_top_weighted_members_" + i,
|
||||||
|
u -> getRankMembers(gm.getOverWorld(), rank, true));
|
||||||
|
// Weighted Level (Level / number of members)
|
||||||
|
bpm.registerPlaceholder(addon, gm.getDescription().getName().toLowerCase() + "_top_weighted_value_" + i,
|
||||||
|
u -> getRankLevel(gm.getOverWorld(), rank, true));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Personal rank
|
// Personal rank
|
||||||
bpm.registerPlaceholder(addon,
|
bpm.registerPlaceholder(addon, gm.getDescription().getName().toLowerCase() + "_rank_value",
|
||||||
gm.getDescription().getName().toLowerCase() + "_rank_value", u -> getRankValue(gm.getOverWorld(), u));
|
u -> getRankValue(gm.getOverWorld(), u));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the name of the player who holds the rank in this world
|
* Get the name of the owner of the island who holds the rank in this world.
|
||||||
|
*
|
||||||
* @param world world
|
* @param world world
|
||||||
* @param rank rank 1 to 10
|
* @param rank rank 1 to 10
|
||||||
|
* @param weighted if true, then the weighted rank name is returned
|
||||||
* @return rank name
|
* @return rank name
|
||||||
*/
|
*/
|
||||||
String getRankName(World world, int rank) {
|
String getRankName(World world, int rank, boolean weighted) {
|
||||||
if (rank < 1) rank = 1;
|
// Ensure rank is within bounds
|
||||||
if (rank > Level.TEN) rank = Level.TEN;
|
rank = Math.max(1, Math.min(rank, Level.TEN));
|
||||||
return addon.getPlayers().getName(addon.getManager().getTopTen(world, Level.TEN).keySet().stream().skip(rank - 1L).limit(1L).findFirst().orElse(null));
|
if (weighted) {
|
||||||
|
return addon.getManager().getWeightedTopTen(world, Level.TEN).keySet().stream().skip(rank - 1L).limit(1L)
|
||||||
|
.findFirst().map(Island::getOwner).map(addon.getPlayers()::getName).orElse("");
|
||||||
|
}
|
||||||
|
@Nullable
|
||||||
|
UUID owner = addon.getManager().getTopTen(world, Level.TEN).keySet().stream().skip(rank - 1L).limit(1L)
|
||||||
|
.findFirst().flatMap(addon.getIslands()::getIslandById).map(Island::getOwner).orElse(null);
|
||||||
|
|
||||||
|
return addon.getPlayers().getName(owner);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the island name for this rank
|
* Get the island name for this rank
|
||||||
|
*
|
||||||
* @param world world
|
* @param world world
|
||||||
* @param rank rank 1 to 10
|
* @param rank rank 1 to 10
|
||||||
|
* @param weighted if true, then the weighted rank name is returned
|
||||||
* @return name of island or nothing if there isn't one
|
* @return name of island or nothing if there isn't one
|
||||||
*/
|
*/
|
||||||
String getRankIslandName(World world, int rank) {
|
String getRankIslandName(World world, int rank, boolean weighted) {
|
||||||
if (rank < 1) rank = 1;
|
// Ensure rank is within bounds
|
||||||
if (rank > Level.TEN) rank = Level.TEN;
|
rank = Math.max(1, Math.min(rank, Level.TEN));
|
||||||
UUID owner = addon.getManager().getTopTen(world, Level.TEN).keySet().stream().skip(rank - 1L).limit(1L).findFirst().orElse(null);
|
if (weighted) {
|
||||||
if (owner != null) {
|
return addon.getManager().getWeightedTopTen(world, Level.TEN).keySet().stream().skip(rank - 1L).limit(1L)
|
||||||
Island island = addon.getIslands().getIsland(world, owner);
|
.findFirst().map(Island::getName).orElse("");
|
||||||
if (island != null) {
|
|
||||||
return island.getName() == null ? "" : island.getName();
|
|
||||||
}
|
}
|
||||||
}
|
return addon.getManager().getTopTen(world, Level.TEN).keySet().stream().skip(rank - 1L).limit(1L).findFirst()
|
||||||
return "";
|
.flatMap(addon.getIslands()::getIslandById).map(Island::getName).orElse("");
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets a comma separated string of island member names
|
* Gets a comma separated string of island member names
|
||||||
|
*
|
||||||
* @param world world
|
* @param world world
|
||||||
* @param rank rank to request
|
* @param rank rank to request
|
||||||
|
* @param weighted if true, then the weighted rank name is returned
|
||||||
* @return comma separated string of island member names
|
* @return comma separated string of island member names
|
||||||
*/
|
*/
|
||||||
String getRankMembers(World world, int rank) {
|
String getRankMembers(World world, int rank, boolean weighted) {
|
||||||
if (rank < 1) rank = 1;
|
// Ensure rank is within bounds
|
||||||
if (rank > Level.TEN) rank = Level.TEN;
|
rank = Math.max(1, Math.min(rank, Level.TEN));
|
||||||
UUID owner = addon.getManager().getTopTen(world, Level.TEN).keySet().stream().skip(rank - 1L).limit(1L).findFirst().orElse(null);
|
if (weighted) {
|
||||||
if (owner != null) {
|
return addon.getManager().getWeightedTopTen(world, Level.TEN).keySet().stream().skip(rank - 1L).limit(1L)
|
||||||
Island island = addon.getIslands().getIsland(world, owner);
|
.findFirst()
|
||||||
if (island != null) {
|
.map(is -> is.getMembers().entrySet().stream().filter(e -> e.getValue() >= RanksManager.MEMBER_RANK)
|
||||||
// Sort members by rank
|
.sorted(Collections.reverseOrder(Map.Entry.comparingByValue())).map(Map.Entry::getKey)
|
||||||
return island.getMembers().entrySet().stream()
|
.map(addon.getPlayers()::getName).collect(Collectors.joining(",")))
|
||||||
.filter(e -> e.getValue() >= RanksManager.MEMBER_RANK)
|
.orElse("");
|
||||||
.sorted(Collections.reverseOrder(Map.Entry.comparingByValue()))
|
|
||||||
.map(Map.Entry::getKey)
|
|
||||||
.map(addon.getPlayers()::getName)
|
|
||||||
.collect(Collectors.joining(","));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Optional<Island> island = addon.getManager().getTopTen(world, Level.TEN).keySet().stream().skip(rank - 1L)
|
||||||
|
.limit(1L).findFirst().flatMap(addon.getIslands()::getIslandById);
|
||||||
|
|
||||||
|
if (island.isPresent()) {
|
||||||
|
// Sort members by rank
|
||||||
|
return island.get().getMembers().entrySet().stream().filter(e -> e.getValue() >= RanksManager.MEMBER_RANK)
|
||||||
|
.sorted(Collections.reverseOrder(Map.Entry.comparingByValue())).map(Map.Entry::getKey)
|
||||||
|
.map(addon.getPlayers()::getName).collect(Collectors.joining(","));
|
||||||
}
|
}
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
|
|
||||||
String getRankLevel(World world, int rank) {
|
/**
|
||||||
if (rank < 1) rank = 1;
|
* Get the level for the rank requested
|
||||||
if (rank > Level.TEN) rank = Level.TEN;
|
*
|
||||||
return addon.getManager()
|
* @param world world
|
||||||
.formatLevel(addon.getManager()
|
* @param rank rank wanted
|
||||||
.getTopTen(world, Level.TEN)
|
* @param weighted true if weighted (level/number of team members)
|
||||||
.values()
|
* @return level for the rank requested
|
||||||
.stream()
|
*/
|
||||||
.skip(rank - 1L)
|
String getRankLevel(World world, int rank, boolean weighted) {
|
||||||
.limit(1L)
|
// Ensure rank is within bounds
|
||||||
.findFirst()
|
rank = Math.max(1, Math.min(rank, Level.TEN));
|
||||||
.orElse(null));
|
if (weighted) {
|
||||||
|
return addon.getManager().formatLevel(addon.getManager().getWeightedTopTen(world, Level.TEN).values()
|
||||||
|
.stream().skip(rank - 1L).limit(1L).findFirst().orElse(null));
|
||||||
|
}
|
||||||
|
return addon.getManager().formatLevel(addon.getManager().getTopTen(world, Level.TEN).values().stream()
|
||||||
|
.skip(rank - 1L).limit(1L).findFirst().orElse(null));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return the rank of the player in a world
|
* Return the rank of the player in a world
|
||||||
|
*
|
||||||
* @param world world
|
* @param world world
|
||||||
* @param user player
|
* @param user player
|
||||||
* @return rank where 1 is the top rank.
|
* @return rank where 1 is the top rank.
|
||||||
@ -163,11 +200,13 @@ public class PlaceholderManager {
|
|||||||
}
|
}
|
||||||
// Get the island level for this user
|
// Get the island level for this user
|
||||||
long level = addon.getManager().getIslandLevel(world, user.getUniqueId());
|
long level = addon.getManager().getIslandLevel(world, user.getUniqueId());
|
||||||
return String.valueOf(addon.getManager().getTopTenLists().getOrDefault(world, new TopTenData(world)).getTopTen().values().stream().filter(l -> l > level).count() + 1);
|
return String.valueOf(addon.getManager().getTopTenLists().getOrDefault(world, new TopTenData(world)).getTopTen()
|
||||||
|
.values().stream().filter(l -> l > level).count() + 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
String getVisitedIslandLevel(GameModeAddon gm, User user) {
|
String getVisitedIslandLevel(GameModeAddon gm, User user) {
|
||||||
if (user == null || !gm.inWorld(user.getWorld())) return "";
|
if (user == null || !gm.inWorld(user.getWorld()))
|
||||||
|
return "";
|
||||||
return addon.getIslands().getIslandAt(user.getLocation())
|
return addon.getIslands().getIslandAt(user.getLocation())
|
||||||
.map(island -> addon.getManager().getIslandLevelString(gm.getOverWorld(), island.getOwner()))
|
.map(island -> addon.getManager().getIslandLevelString(gm.getOverWorld(), island.getOwner()))
|
||||||
.orElse("0");
|
.orElse("0");
|
||||||
|
@ -0,0 +1,121 @@
|
|||||||
|
package world.bentobox.level.calculators;
|
||||||
|
|
||||||
|
import java.text.ParseException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Utility class to evaluate equations
|
||||||
|
*/
|
||||||
|
public class EquationEvaluator {
|
||||||
|
|
||||||
|
private static class Parser {
|
||||||
|
private final String input;
|
||||||
|
private int pos = -1;
|
||||||
|
private int currentChar;
|
||||||
|
|
||||||
|
@SuppressWarnings("unused")
|
||||||
|
private Parser() {
|
||||||
|
throw new IllegalStateException("Utility class");
|
||||||
|
}
|
||||||
|
|
||||||
|
public Parser(String input) {
|
||||||
|
this.input = input;
|
||||||
|
moveToNextChar();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void moveToNextChar() {
|
||||||
|
currentChar = (++pos < input.length()) ? input.charAt(pos) : -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean tryToEat(int charToEat) {
|
||||||
|
while (currentChar == ' ') {
|
||||||
|
moveToNextChar();
|
||||||
|
}
|
||||||
|
if (currentChar == charToEat) {
|
||||||
|
moveToNextChar();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public double evaluate() throws ParseException {
|
||||||
|
double result = parseExpression();
|
||||||
|
if (pos < input.length()) {
|
||||||
|
throw new ParseException("Unexpected character: " + (char) currentChar, pos);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private double parseExpression() throws ParseException {
|
||||||
|
double result = parseTerm();
|
||||||
|
while (true) {
|
||||||
|
if (tryToEat('+')) {
|
||||||
|
result += parseTerm();
|
||||||
|
} else if (tryToEat('-')) {
|
||||||
|
result -= parseTerm();
|
||||||
|
} else {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private double parseFactor() throws ParseException {
|
||||||
|
if (tryToEat('+')) {
|
||||||
|
return parseFactor(); // unary plus
|
||||||
|
}
|
||||||
|
if (tryToEat('-')) {
|
||||||
|
return -parseFactor(); // unary minus
|
||||||
|
}
|
||||||
|
double x;
|
||||||
|
int startPos = this.pos;
|
||||||
|
if (tryToEat('(')) { // parentheses
|
||||||
|
x = parseExpression();
|
||||||
|
tryToEat(')');
|
||||||
|
} else if ((currentChar >= '0' && currentChar <= '9') || currentChar == '.') { // numbers
|
||||||
|
while ((currentChar >= '0' && currentChar <= '9') || currentChar == '.') {
|
||||||
|
moveToNextChar();
|
||||||
|
}
|
||||||
|
x = Double.parseDouble(input.substring(startPos, this.pos));
|
||||||
|
} else if (currentChar >= 'a' && currentChar <= 'z') { // functions
|
||||||
|
while (currentChar >= 'a' && currentChar <= 'z') {
|
||||||
|
moveToNextChar();
|
||||||
|
}
|
||||||
|
String func = input.substring(startPos, this.pos);
|
||||||
|
x = parseFactor();
|
||||||
|
x = switch (func) {
|
||||||
|
case "sqrt" -> Math.sqrt(x);
|
||||||
|
case "sin" -> Math.sin(Math.toRadians(x));
|
||||||
|
case "cos" -> Math.cos(Math.toRadians(x));
|
||||||
|
case "tan" -> Math.tan(Math.toRadians(x));
|
||||||
|
case "log" -> Math.log(x);
|
||||||
|
default -> throw new ParseException("Unknown function: " + func, startPos);
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
throw new ParseException("Unexpected: " + (char) currentChar, startPos);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tryToEat('^')) {
|
||||||
|
x = Math.pow(x, parseFactor()); // exponentiation
|
||||||
|
}
|
||||||
|
|
||||||
|
return x;
|
||||||
|
}
|
||||||
|
|
||||||
|
private double parseTerm() throws ParseException {
|
||||||
|
double x = parseFactor();
|
||||||
|
for (;;) {
|
||||||
|
if (tryToEat('*'))
|
||||||
|
x *= parseFactor(); // multiplication
|
||||||
|
else if (tryToEat('/'))
|
||||||
|
x /= parseFactor(); // division
|
||||||
|
else
|
||||||
|
return x;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public static double eval(final String equation) throws ParseException {
|
||||||
|
return new Parser(equation).evaluate();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -1,6 +1,6 @@
|
|||||||
package world.bentobox.level.calculators;
|
package world.bentobox.level.calculators;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.text.ParseException;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
@ -16,9 +16,6 @@ import java.util.UUID;
|
|||||||
import java.util.concurrent.CompletableFuture;
|
import java.util.concurrent.CompletableFuture;
|
||||||
import java.util.concurrent.ConcurrentLinkedQueue;
|
import java.util.concurrent.ConcurrentLinkedQueue;
|
||||||
|
|
||||||
import com.songoda.ultimatestacker.UltimateStacker;
|
|
||||||
import com.songoda.ultimatestacker.core.compatibility.CompatibleMaterial;
|
|
||||||
import com.songoda.ultimatestacker.stackable.block.BlockStack;
|
|
||||||
import org.bukkit.Bukkit;
|
import org.bukkit.Bukkit;
|
||||||
import org.bukkit.Chunk;
|
import org.bukkit.Chunk;
|
||||||
import org.bukkit.ChunkSnapshot;
|
import org.bukkit.ChunkSnapshot;
|
||||||
@ -27,7 +24,11 @@ import org.bukkit.Material;
|
|||||||
import org.bukkit.Tag;
|
import org.bukkit.Tag;
|
||||||
import org.bukkit.World;
|
import org.bukkit.World;
|
||||||
import org.bukkit.World.Environment;
|
import org.bukkit.World.Environment;
|
||||||
import org.bukkit.block.*;
|
import org.bukkit.block.Block;
|
||||||
|
import org.bukkit.block.BlockState;
|
||||||
|
import org.bukkit.block.Container;
|
||||||
|
import org.bukkit.block.CreatureSpawner;
|
||||||
|
import org.bukkit.block.ShulkerBox;
|
||||||
import org.bukkit.block.data.BlockData;
|
import org.bukkit.block.data.BlockData;
|
||||||
import org.bukkit.block.data.type.Slab;
|
import org.bukkit.block.data.type.Slab;
|
||||||
import org.bukkit.inventory.ItemStack;
|
import org.bukkit.inventory.ItemStack;
|
||||||
@ -39,6 +40,9 @@ import com.bgsoftware.wildstacker.api.objects.StackedBarrel;
|
|||||||
import com.google.common.collect.Multiset;
|
import com.google.common.collect.Multiset;
|
||||||
import com.google.common.collect.Multiset.Entry;
|
import com.google.common.collect.Multiset.Entry;
|
||||||
import com.google.common.collect.Multisets;
|
import com.google.common.collect.Multisets;
|
||||||
|
import com.songoda.ultimatestacker.UltimateStacker;
|
||||||
|
import com.songoda.ultimatestacker.core.compatibility.CompatibleMaterial;
|
||||||
|
import com.songoda.ultimatestacker.stackable.block.BlockStack;
|
||||||
|
|
||||||
import dev.rosewood.rosestacker.api.RoseStackerAPI;
|
import dev.rosewood.rosestacker.api.RoseStackerAPI;
|
||||||
import us.lynuxcraft.deadsilenceiv.advancedchests.AdvancedChestsAPI;
|
import us.lynuxcraft.deadsilenceiv.advancedchests.AdvancedChestsAPI;
|
||||||
@ -54,121 +58,21 @@ import world.bentobox.level.calculators.Results.Result;
|
|||||||
public class IslandLevelCalculator {
|
public class IslandLevelCalculator {
|
||||||
private static final String LINE_BREAK = "==================================";
|
private static final String LINE_BREAK = "==================================";
|
||||||
public static final long MAX_AMOUNT = 10000000;
|
public static final long MAX_AMOUNT = 10000000;
|
||||||
private static final List<Material> CHESTS = Arrays.asList(Material.CHEST, Material.CHEST_MINECART, Material.TRAPPED_CHEST,
|
private static final List<Material> CHESTS = Arrays.asList(Material.CHEST, Material.CHEST_MINECART,
|
||||||
Material.SHULKER_BOX, Material.BLACK_SHULKER_BOX, Material.BLUE_SHULKER_BOX, Material.BROWN_SHULKER_BOX,
|
Material.TRAPPED_CHEST, Material.SHULKER_BOX, Material.BLACK_SHULKER_BOX, Material.BLUE_SHULKER_BOX,
|
||||||
Material.CYAN_SHULKER_BOX, Material.GRAY_SHULKER_BOX, Material.GREEN_SHULKER_BOX, Material.LIGHT_BLUE_SHULKER_BOX,
|
Material.BROWN_SHULKER_BOX, Material.CYAN_SHULKER_BOX, Material.GRAY_SHULKER_BOX,
|
||||||
Material.LIGHT_GRAY_SHULKER_BOX, Material.LIME_SHULKER_BOX, Material.MAGENTA_SHULKER_BOX, Material.ORANGE_SHULKER_BOX,
|
Material.GREEN_SHULKER_BOX, Material.LIGHT_BLUE_SHULKER_BOX, Material.LIGHT_GRAY_SHULKER_BOX,
|
||||||
|
Material.LIME_SHULKER_BOX, Material.MAGENTA_SHULKER_BOX, Material.ORANGE_SHULKER_BOX,
|
||||||
Material.PINK_SHULKER_BOX, Material.PURPLE_SHULKER_BOX, Material.RED_SHULKER_BOX, Material.RED_SHULKER_BOX,
|
Material.PINK_SHULKER_BOX, Material.PURPLE_SHULKER_BOX, Material.RED_SHULKER_BOX, Material.RED_SHULKER_BOX,
|
||||||
Material.WHITE_SHULKER_BOX, Material.YELLOW_SHULKER_BOX, Material.COMPOSTER, Material.BARREL, Material.DISPENSER,
|
Material.WHITE_SHULKER_BOX, Material.YELLOW_SHULKER_BOX, Material.COMPOSTER, Material.BARREL,
|
||||||
Material.DROPPER, Material.SMOKER, Material.BLAST_FURNACE);
|
Material.DISPENSER, Material.DROPPER, Material.SMOKER, Material.BLAST_FURNACE);
|
||||||
private static final int CHUNKS_TO_SCAN = 100;
|
private static final int CHUNKS_TO_SCAN = 100;
|
||||||
|
|
||||||
/**
|
|
||||||
* Method to evaluate a mathematical equation
|
|
||||||
* @param str - equation to evaluate
|
|
||||||
* @return value of equation
|
|
||||||
*/
|
|
||||||
private static double eval(final String str) throws IOException {
|
|
||||||
return new Object() {
|
|
||||||
int pos = -1;
|
|
||||||
int ch;
|
|
||||||
|
|
||||||
boolean eat(int charToEat) {
|
|
||||||
while (ch == ' ') nextChar();
|
|
||||||
if (ch == charToEat) {
|
|
||||||
nextChar();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
void nextChar() {
|
|
||||||
ch = (++pos < str.length()) ? str.charAt(pos) : -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
double parse() throws IOException {
|
|
||||||
nextChar();
|
|
||||||
double x = parseExpression();
|
|
||||||
if (pos < str.length()) throw new IOException("Unexpected: " + (char)ch);
|
|
||||||
return x;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Grammar:
|
|
||||||
// expression = term | expression `+` term | expression `-` term
|
|
||||||
// term = factor | term `*` factor | term `/` factor
|
|
||||||
// factor = `+` factor | `-` factor | `(` expression `)`
|
|
||||||
// | number | functionName factor | factor `^` factor
|
|
||||||
|
|
||||||
double parseExpression() throws IOException {
|
|
||||||
double x = parseTerm();
|
|
||||||
for (;;) {
|
|
||||||
if (eat('+')) x += parseTerm(); // addition
|
|
||||||
else if (eat('-')) x -= parseTerm(); // subtraction
|
|
||||||
else return x;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
double parseFactor() throws IOException {
|
|
||||||
if (eat('+')) return parseFactor(); // unary plus
|
|
||||||
if (eat('-')) return -parseFactor(); // unary minus
|
|
||||||
|
|
||||||
double x;
|
|
||||||
int startPos = this.pos;
|
|
||||||
if (eat('(')) { // parentheses
|
|
||||||
x = parseExpression();
|
|
||||||
eat(')');
|
|
||||||
} else if ((ch >= '0' && ch <= '9') || ch == '.') { // numbers
|
|
||||||
while ((ch >= '0' && ch <= '9') || ch == '.') nextChar();
|
|
||||||
x = Double.parseDouble(str.substring(startPos, this.pos));
|
|
||||||
} else if (ch >= 'a' && ch <= 'z') { // functions
|
|
||||||
while (ch >= 'a' && ch <= 'z') nextChar();
|
|
||||||
String func = str.substring(startPos, this.pos);
|
|
||||||
x = parseFactor();
|
|
||||||
switch (func) {
|
|
||||||
case "sqrt":
|
|
||||||
x = Math.sqrt(x);
|
|
||||||
break;
|
|
||||||
case "sin":
|
|
||||||
x = Math.sin(Math.toRadians(x));
|
|
||||||
break;
|
|
||||||
case "cos":
|
|
||||||
x = Math.cos(Math.toRadians(x));
|
|
||||||
break;
|
|
||||||
case "tan":
|
|
||||||
x = Math.tan(Math.toRadians(x));
|
|
||||||
break;
|
|
||||||
case "log":
|
|
||||||
x = Math.log(x);
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
throw new IOException("Unknown function: " + func);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
throw new IOException("Unexpected: " + (char)ch);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (eat('^')) x = Math.pow(x, parseFactor()); // exponentiation
|
|
||||||
|
|
||||||
return x;
|
|
||||||
}
|
|
||||||
|
|
||||||
double parseTerm() throws IOException {
|
|
||||||
double x = parseFactor();
|
|
||||||
for (;;) {
|
|
||||||
if (eat('*')) x *= parseFactor(); // multiplication
|
|
||||||
else if (eat('/')) x /= parseFactor(); // division
|
|
||||||
else return x;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}.parse();
|
|
||||||
}
|
|
||||||
private final Level addon;
|
private final Level addon;
|
||||||
private final Queue<Pair<Integer, Integer>> chunksToCheck;
|
private final Queue<Pair<Integer, Integer>> chunksToCheck;
|
||||||
private final Island island;
|
private final Island island;
|
||||||
private final Map<Material, Integer> limitCount;
|
private final Map<Material, Integer> limitCount;
|
||||||
private final CompletableFuture<Results> r;
|
private final CompletableFuture<Results> r;
|
||||||
|
|
||||||
|
|
||||||
private final Results results;
|
private final Results results;
|
||||||
private long duration;
|
private long duration;
|
||||||
private final boolean zeroIsland;
|
private final boolean zeroIsland;
|
||||||
@ -178,12 +82,13 @@ public class IslandLevelCalculator {
|
|||||||
private final Set<Chunk> chestBlocks = new HashSet<>();
|
private final Set<Chunk> chestBlocks = new HashSet<>();
|
||||||
private BukkitTask finishTask;
|
private BukkitTask finishTask;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor to get the level for an island
|
* Constructor to get the level for an island
|
||||||
|
*
|
||||||
* @param addon - Level addon
|
* @param addon - Level addon
|
||||||
* @param island - the island to scan
|
* @param island - the island to scan
|
||||||
* @param r - completable result that will be completed when the calculation is complete
|
* @param r - completable result that will be completed when the
|
||||||
|
* calculation is complete
|
||||||
* @param zeroIsland - true if the calculation is due to an island zeroing
|
* @param zeroIsland - true if the calculation is due to an island zeroing
|
||||||
*/
|
*/
|
||||||
public IslandLevelCalculator(Level addon, Island island, CompletableFuture<Results> r, boolean zeroIsland) {
|
public IslandLevelCalculator(Level addon, Island island, CompletableFuture<Results> r, boolean zeroIsland) {
|
||||||
@ -219,25 +124,29 @@ public class IslandLevelCalculator {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Calculate the level based on the raw points
|
* Calculate the level based on the raw points
|
||||||
|
*
|
||||||
* @param blockAndDeathPoints - raw points counted on island
|
* @param blockAndDeathPoints - raw points counted on island
|
||||||
* @return level of island
|
* @return level of island
|
||||||
*/
|
*/
|
||||||
private long calculateLevel(long blockAndDeathPoints) {
|
private long calculateLevel(long blockAndDeathPoints) {
|
||||||
String calcString = addon.getSettings().getLevelCalc();
|
String calcString = addon.getSettings().getLevelCalc();
|
||||||
String withValues = calcString.replace("blocks", String.valueOf(blockAndDeathPoints)).replace("level_cost", String.valueOf(this.addon.getSettings().getLevelCost()));
|
String withValues = calcString.replace("blocks", String.valueOf(blockAndDeathPoints)).replace("level_cost",
|
||||||
|
String.valueOf(this.addon.getSettings().getLevelCost()));
|
||||||
long evalWithValues;
|
long evalWithValues;
|
||||||
try {
|
try {
|
||||||
evalWithValues = (long)eval(withValues);
|
evalWithValues = (long) EquationEvaluator.eval(withValues);
|
||||||
return evalWithValues - (addon.getSettings().isZeroNewIslandLevels() ? results.initialLevel.get() : 0);
|
return evalWithValues - (addon.getSettings().isZeroNewIslandLevels() ? results.initialLevel.get() : 0);
|
||||||
|
|
||||||
} catch (IOException e) {
|
} catch (ParseException e) {
|
||||||
addon.getPlugin().logStacktrace(e);
|
addon.getPlugin().logStacktrace(e);
|
||||||
return 0L;
|
return 0L;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Adds value to the results based on the material and whether the block is below sea level or not
|
* Adds value to the results based on the material and whether the block is
|
||||||
|
* below sea level or not
|
||||||
|
*
|
||||||
* @param mat - material of the block
|
* @param mat - material of the block
|
||||||
* @param belowSeaLevel - true if below sea level
|
* @param belowSeaLevel - true if below sea level
|
||||||
*/
|
*/
|
||||||
@ -254,20 +163,22 @@ public class IslandLevelCalculator {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Get a set of all the chunks in island
|
* Get a set of all the chunks in island
|
||||||
|
*
|
||||||
* @param island - island
|
* @param island - island
|
||||||
* @return - set of pairs of x,z coordinates to check
|
* @return - set of pairs of x,z coordinates to check
|
||||||
*/
|
*/
|
||||||
private Queue<Pair<Integer, Integer>> getChunksToScan(Island island) {
|
private Queue<Pair<Integer, Integer>> getChunksToScan(Island island) {
|
||||||
Queue<Pair<Integer, Integer>> chunkQueue = new ConcurrentLinkedQueue<>();
|
Queue<Pair<Integer, Integer>> chunkQueue = new ConcurrentLinkedQueue<>();
|
||||||
for (int x = island.getMinProtectedX(); x < (island.getMinProtectedX() + island.getProtectionRange() * 2 + 16); x += 16) {
|
for (int x = island.getMinProtectedX(); x < (island.getMinProtectedX() + island.getProtectionRange() * 2
|
||||||
for (int z = island.getMinProtectedZ(); z < (island.getMinProtectedZ() + island.getProtectionRange() * 2 + 16); z += 16) {
|
+ 16); x += 16) {
|
||||||
|
for (int z = island.getMinProtectedZ(); z < (island.getMinProtectedZ() + island.getProtectionRange() * 2
|
||||||
|
+ 16); z += 16) {
|
||||||
chunkQueue.add(new Pair<>(x >> 4, z >> 4));
|
chunkQueue.add(new Pair<>(x >> 4, z >> 4));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return chunkQueue;
|
return chunkQueue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return the island
|
* @return the island
|
||||||
*/
|
*/
|
||||||
@ -277,6 +188,7 @@ public class IslandLevelCalculator {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the completable result for this calculation
|
* Get the completable result for this calculation
|
||||||
|
*
|
||||||
* @return the r
|
* @return the r
|
||||||
*/
|
*/
|
||||||
public CompletableFuture<Results> getR() {
|
public CompletableFuture<Results> getR() {
|
||||||
@ -285,12 +197,14 @@ public class IslandLevelCalculator {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the full analysis report
|
* Get the full analysis report
|
||||||
|
*
|
||||||
* @return a list of lines
|
* @return a list of lines
|
||||||
*/
|
*/
|
||||||
private List<String> getReport() {
|
private List<String> getReport() {
|
||||||
List<String> reportLines = new ArrayList<>();
|
List<String> reportLines = new ArrayList<>();
|
||||||
// provide counts
|
// provide counts
|
||||||
reportLines.add("Level Log for island in " + addon.getPlugin().getIWM().getFriendlyName(island.getWorld()) + " at " + Util.xyz(island.getCenter().toVector()));
|
reportLines.add("Level Log for island in " + addon.getPlugin().getIWM().getFriendlyName(island.getWorld())
|
||||||
|
+ " at " + Util.xyz(island.getCenter().toVector()));
|
||||||
reportLines.add("Island owner UUID = " + island.getOwner());
|
reportLines.add("Island owner UUID = " + island.getOwner());
|
||||||
reportLines.add("Total block value count = " + String.format("%,d", results.rawBlockCount.get()));
|
reportLines.add("Total block value count = " + String.format("%,d", results.rawBlockCount.get()));
|
||||||
reportLines.add("Formula to calculate island level: " + addon.getSettings().getLevelCalc());
|
reportLines.add("Formula to calculate island level: " + addon.getSettings().getLevelCalc());
|
||||||
@ -304,7 +218,8 @@ public class IslandLevelCalculator {
|
|||||||
reportLines.add(LINE_BREAK);
|
reportLines.add(LINE_BREAK);
|
||||||
int total = 0;
|
int total = 0;
|
||||||
if (!results.uwCount.isEmpty()) {
|
if (!results.uwCount.isEmpty()) {
|
||||||
reportLines.add("Underwater block count (Multiplier = x" + addon.getSettings().getUnderWaterMultiplier() + ") value");
|
reportLines.add("Underwater block count (Multiplier = x" + addon.getSettings().getUnderWaterMultiplier()
|
||||||
|
+ ") value");
|
||||||
reportLines.add("Total number of underwater blocks = " + String.format("%,d", results.uwCount.size()));
|
reportLines.add("Total number of underwater blocks = " + String.format("%,d", results.uwCount.size()));
|
||||||
reportLines.addAll(sortedReport(total, results.uwCount));
|
reportLines.addAll(sortedReport(total, results.uwCount));
|
||||||
}
|
}
|
||||||
@ -312,7 +227,8 @@ public class IslandLevelCalculator {
|
|||||||
reportLines.add("Total number of blocks = " + String.format("%,d", results.mdCount.size()));
|
reportLines.add("Total number of blocks = " + String.format("%,d", results.mdCount.size()));
|
||||||
reportLines.addAll(sortedReport(total, results.mdCount));
|
reportLines.addAll(sortedReport(total, results.mdCount));
|
||||||
|
|
||||||
reportLines.add("Blocks not counted because they exceeded limits: " + String.format("%,d",results.ofCount.size()));
|
reportLines.add(
|
||||||
|
"Blocks not counted because they exceeded limits: " + String.format("%,d", results.ofCount.size()));
|
||||||
Iterable<Multiset.Entry<Material>> entriesSortedByCount = results.ofCount.entrySet();
|
Iterable<Multiset.Entry<Material>> entriesSortedByCount = results.ofCount.entrySet();
|
||||||
Iterator<Entry<Material>> it = entriesSortedByCount.iterator();
|
Iterator<Entry<Material>> it = entriesSortedByCount.iterator();
|
||||||
while (it.hasNext()) {
|
while (it.hasNext()) {
|
||||||
@ -324,7 +240,8 @@ public class IslandLevelCalculator {
|
|||||||
limit = addon.getBlockConfig().getBlockLimits().get(generic);
|
limit = addon.getBlockConfig().getBlockLimits().get(generic);
|
||||||
explain = " - All types)";
|
explain = " - All types)";
|
||||||
}
|
}
|
||||||
reportLines.add(type.getElement().toString() + ": " + String.format("%,d",type.getCount()) + " blocks (max " + limit + explain);
|
reportLines.add(type.getElement().toString() + ": " + String.format("%,d", type.getCount())
|
||||||
|
+ " blocks (max " + limit + explain);
|
||||||
}
|
}
|
||||||
reportLines.add(LINE_BREAK);
|
reportLines.add(LINE_BREAK);
|
||||||
reportLines.add("Blocks on island that are not in config.yml");
|
reportLines.add("Blocks on island that are not in config.yml");
|
||||||
@ -346,9 +263,10 @@ public class IslandLevelCalculator {
|
|||||||
public Results getResults() {
|
public Results getResults() {
|
||||||
return results;
|
return results;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get value of a material
|
* Get value of a material World blocks trump regular block values
|
||||||
* World blocks trump regular block values
|
*
|
||||||
* @param md - Material to check
|
* @param md - Material to check
|
||||||
* @return value of a material
|
* @return value of a material
|
||||||
*/
|
*/
|
||||||
@ -364,9 +282,11 @@ public class IslandLevelCalculator {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Get a chunk async
|
* Get a chunk async
|
||||||
|
*
|
||||||
* @param env - the environment
|
* @param env - the environment
|
||||||
* @param pairList - chunk coordinate
|
* @param pairList - chunk coordinate
|
||||||
* @return a future chunk or future null if there is no chunk to load, e.g., there is no island nether
|
* @return a future chunk or future null if there is no chunk to load, e.g.,
|
||||||
|
* there is no island nether
|
||||||
*/
|
*/
|
||||||
private CompletableFuture<List<Chunk>> getWorldChunk(Environment env, Queue<Pair<Integer, Integer>> pairList) {
|
private CompletableFuture<List<Chunk>> getWorldChunk(Environment env, Queue<Pair<Integer, Integer>> pairList) {
|
||||||
if (worlds.containsKey(env)) {
|
if (worlds.containsKey(env)) {
|
||||||
@ -410,7 +330,9 @@ public class IslandLevelCalculator {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks if a block has been limited or not and whether a block has any value or not
|
* Checks if a block has been limited or not and whether a block has any value
|
||||||
|
* or not
|
||||||
|
*
|
||||||
* @param md Material
|
* @param md Material
|
||||||
* @return value of the block if can be counted
|
* @return value of the block if can be counted
|
||||||
*/
|
*/
|
||||||
@ -428,9 +350,9 @@ public class IslandLevelCalculator {
|
|||||||
return getValue(md);
|
return getValue(md);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Scan all containers in a chunk and count their blocks
|
* Scan all containers in a chunk and count their blocks
|
||||||
|
*
|
||||||
* @param chunk - the chunk to scan
|
* @param chunk - the chunk to scan
|
||||||
*/
|
*/
|
||||||
private void scanChests(Chunk chunk) {
|
private void scanChests(Chunk chunk) {
|
||||||
@ -455,7 +377,8 @@ public class IslandLevelCalculator {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void countItemStack(ItemStack i) {
|
private void countItemStack(ItemStack i) {
|
||||||
if (i == null || !i.getType().isBlock()) return;
|
if (i == null || !i.getType().isBlock())
|
||||||
|
return;
|
||||||
|
|
||||||
for (int c = 0; c < i.getAmount(); c++) {
|
for (int c = 0; c < i.getAmount(); c++) {
|
||||||
if (addon.getSettings().isIncludeShulkersInChest()
|
if (addon.getSettings().isIncludeShulkersInChest()
|
||||||
@ -469,10 +392,13 @@ public class IslandLevelCalculator {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Scan the chunk chests and count the blocks. Note that the chunks are a list of all the island chunks
|
* Scan the chunk chests and count the blocks. Note that the chunks are a list
|
||||||
* in a particular world, so the memory usage is high, but I think most servers can handle it.
|
* of all the island chunks in a particular world, so the memory usage is high,
|
||||||
|
* but I think most servers can handle it.
|
||||||
|
*
|
||||||
* @param chunks - a list of chunks to scan
|
* @param chunks - a list of chunks to scan
|
||||||
* @return future that completes when the scan is done and supplies a boolean that will be true if the scan was successful, false if not
|
* @return future that completes when the scan is done and supplies a boolean
|
||||||
|
* that will be true if the scan was successful, false if not
|
||||||
*/
|
*/
|
||||||
private CompletableFuture<Boolean> scanChunk(List<Chunk> chunks) {
|
private CompletableFuture<Boolean> scanChunk(List<Chunk> chunks) {
|
||||||
// If the chunk hasn't been generated, return
|
// If the chunk hasn't been generated, return
|
||||||
@ -482,11 +408,13 @@ public class IslandLevelCalculator {
|
|||||||
// Count blocks in chunk
|
// Count blocks in chunk
|
||||||
CompletableFuture<Boolean> result = new CompletableFuture<>();
|
CompletableFuture<Boolean> result = new CompletableFuture<>();
|
||||||
/*
|
/*
|
||||||
* At this point, we need to grab a snapshot of each chunk and then scan it async.
|
* At this point, we need to grab a snapshot of each chunk and then scan it
|
||||||
* At the end, we make the CompletableFuture true to show it is done.
|
* async. At the end, we make the CompletableFuture true to show it is done. I'm
|
||||||
* I'm not sure how much lag this will cause, but as all the chunks are loaded, maybe not that much.
|
* not sure how much lag this will cause, but as all the chunks are loaded,
|
||||||
|
* maybe not that much.
|
||||||
*/
|
*/
|
||||||
List<ChunkPair> preLoad = chunks.stream().map(c -> new ChunkPair(c.getWorld(), c, c.getChunkSnapshot())).toList();
|
List<ChunkPair> preLoad = chunks.stream().map(c -> new ChunkPair(c.getWorld(), c, c.getChunkSnapshot()))
|
||||||
|
.toList();
|
||||||
Bukkit.getScheduler().runTaskAsynchronously(BentoBox.getInstance(), () -> {
|
Bukkit.getScheduler().runTaskAsynchronously(BentoBox.getInstance(), () -> {
|
||||||
preLoad.forEach(this::scanAsync);
|
preLoad.forEach(this::scanAsync);
|
||||||
// Once they are all done, return to the main thread.
|
// Once they are all done, return to the main thread.
|
||||||
@ -495,21 +423,27 @@ public class IslandLevelCalculator {
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
record ChunkPair(World world, Chunk chunk, ChunkSnapshot chunkSnapshot) {}
|
record ChunkPair(World world, Chunk chunk, ChunkSnapshot chunkSnapshot) {
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Count the blocks on the island
|
* Count the blocks on the island
|
||||||
|
*
|
||||||
* @param cp chunk to scan
|
* @param cp chunk to scan
|
||||||
*/
|
*/
|
||||||
private void scanAsync(ChunkPair cp) {
|
private void scanAsync(ChunkPair cp) {
|
||||||
for (int x = 0; x < 16; x++) {
|
for (int x = 0; x < 16; x++) {
|
||||||
// Check if the block coordinate is inside the protection zone and if not, don't count it
|
// Check if the block coordinate is inside the protection zone and if not, don't
|
||||||
if (cp.chunkSnapshot.getX() * 16 + x < island.getMinProtectedX() || cp.chunkSnapshot.getX() * 16 + x >= island.getMinProtectedX() + island.getProtectionRange() * 2) {
|
// count it
|
||||||
|
if (cp.chunkSnapshot.getX() * 16 + x < island.getMinProtectedX() || cp.chunkSnapshot.getX() * 16
|
||||||
|
+ x >= island.getMinProtectedX() + island.getProtectionRange() * 2) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
for (int z = 0; z < 16; z++) {
|
for (int z = 0; z < 16; z++) {
|
||||||
// Check if the block coordinate is inside the protection zone and if not, don't count it
|
// Check if the block coordinate is inside the protection zone and if not, don't
|
||||||
if (cp.chunkSnapshot.getZ() * 16 + z < island.getMinProtectedZ() || cp.chunkSnapshot.getZ() * 16 + z >= island.getMinProtectedZ() + island.getProtectionRange() * 2) {
|
// count it
|
||||||
|
if (cp.chunkSnapshot.getZ() * 16 + z < island.getMinProtectedZ() || cp.chunkSnapshot.getZ() * 16
|
||||||
|
+ z >= island.getMinProtectedZ() + island.getProtectionRange() * 2) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
// Only count to the highest block in the world for some optimization
|
// Only count to the highest block in the world for some optimization
|
||||||
@ -523,16 +457,20 @@ public class IslandLevelCalculator {
|
|||||||
checkBlock(blockData.getMaterial(), belowSeaLevel);
|
checkBlock(blockData.getMaterial(), belowSeaLevel);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Hook for Wild Stackers (Blocks and Spawners Only) - this has to use the real chunk
|
// Hook for Wild Stackers (Blocks and Spawners Only) - this has to use the real
|
||||||
if (addon.isStackersEnabled() && (blockData.getMaterial().equals(Material.CAULDRON) || blockData.getMaterial().equals(Material.SPAWNER))) {
|
// chunk
|
||||||
stackedBlocks.add(new Location(cp.world, (double)x + cp.chunkSnapshot.getX() * 16, y, (double)z + cp.chunkSnapshot.getZ() * 16));
|
if (addon.isStackersEnabled() && (blockData.getMaterial().equals(Material.CAULDRON)
|
||||||
|
|| blockData.getMaterial().equals(Material.SPAWNER))) {
|
||||||
|
stackedBlocks.add(new Location(cp.world, (double) x + cp.chunkSnapshot.getX() * 16, y,
|
||||||
|
(double) z + cp.chunkSnapshot.getZ() * 16));
|
||||||
}
|
}
|
||||||
|
|
||||||
Block block = cp.chunk.getBlock(x, y, z);
|
Block block = cp.chunk.getBlock(x, y, z);
|
||||||
|
|
||||||
if (addon.isUltimateStackerEnabled()) {
|
if (addon.isUltimateStackerEnabled()) {
|
||||||
if (!blockData.getMaterial().equals(Material.AIR)) {
|
if (!blockData.getMaterial().equals(Material.AIR)) {
|
||||||
BlockStack stack = UltimateStacker.getInstance().getBlockStackManager().getBlock(block, CompatibleMaterial.getMaterial(block));
|
BlockStack stack = UltimateStacker.getInstance().getBlockStackManager().getBlock(block,
|
||||||
|
CompatibleMaterial.getMaterial(block));
|
||||||
if (stack != null) {
|
if (stack != null) {
|
||||||
int value = limitCount(blockData.getMaterial());
|
int value = limitCount(blockData.getMaterial());
|
||||||
if (belowSeaLevel) {
|
if (belowSeaLevel) {
|
||||||
@ -559,7 +497,9 @@ public class IslandLevelCalculator {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Scan the next chunk on the island
|
* Scan the next chunk on the island
|
||||||
* @return completable boolean future that will be true if more chunks are left to be scanned, and false if not
|
*
|
||||||
|
* @return completable boolean future that will be true if more chunks are left
|
||||||
|
* to be scanned, and false if not
|
||||||
*/
|
*/
|
||||||
public CompletableFuture<Boolean> scanNextChunk() {
|
public CompletableFuture<Boolean> scanNextChunk() {
|
||||||
if (chunksToCheck.isEmpty()) {
|
if (chunksToCheck.isEmpty()) {
|
||||||
@ -579,31 +519,28 @@ public class IslandLevelCalculator {
|
|||||||
CompletableFuture<Boolean> result = new CompletableFuture<>();
|
CompletableFuture<Boolean> result = new CompletableFuture<>();
|
||||||
// Get chunks and scan
|
// Get chunks and scan
|
||||||
// Get chunks and scan
|
// Get chunks and scan
|
||||||
getWorldChunk(Environment.THE_END, endPairList).thenAccept(endChunks ->
|
getWorldChunk(Environment.THE_END, endPairList).thenAccept(
|
||||||
scanChunk(endChunks).thenAccept(b ->
|
endChunks -> scanChunk(endChunks).thenAccept(b -> getWorldChunk(Environment.NETHER, netherPairList)
|
||||||
getWorldChunk(Environment.NETHER, netherPairList).thenAccept(netherChunks ->
|
.thenAccept(netherChunks -> scanChunk(netherChunks)
|
||||||
scanChunk(netherChunks).thenAccept(b2 ->
|
.thenAccept(b2 -> getWorldChunk(Environment.NORMAL, pairList)
|
||||||
getWorldChunk(Environment.NORMAL, pairList).thenAccept(normalChunks ->
|
.thenAccept(normalChunks -> scanChunk(normalChunks).thenAccept(b3 ->
|
||||||
scanChunk(normalChunks).thenAccept(b3 ->
|
|
||||||
// Complete the result now that all chunks have been scanned
|
// Complete the result now that all chunks have been scanned
|
||||||
result.complete(!chunksToCheck.isEmpty()))))
|
result.complete(!chunksToCheck.isEmpty())))))));
|
||||||
)
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
private Collection<String> sortedReport(int total, Multiset<Material> materialCount) {
|
private Collection<String> sortedReport(int total, Multiset<Material> materialCount) {
|
||||||
Collection<String> result = new ArrayList<>();
|
Collection<String> result = new ArrayList<>();
|
||||||
Iterable<Multiset.Entry<Material>> entriesSortedByCount = Multisets.copyHighestCountFirst(materialCount).entrySet();
|
Iterable<Multiset.Entry<Material>> entriesSortedByCount = Multisets.copyHighestCountFirst(materialCount)
|
||||||
|
.entrySet();
|
||||||
for (Entry<Material> en : entriesSortedByCount) {
|
for (Entry<Material> en : entriesSortedByCount) {
|
||||||
Material type = en.getElement();
|
Material type = en.getElement();
|
||||||
|
|
||||||
int value = getValue(type);
|
int value = getValue(type);
|
||||||
|
|
||||||
result.add(type.toString() + ":"
|
result.add(type.toString() + ":" + String.format("%,d", en.getCount()) + " blocks x " + value + " = "
|
||||||
+ String.format("%,d", en.getCount()) + " blocks x " + value + " = " + (value * en.getCount()));
|
+ (value * en.getCount()));
|
||||||
total += (value * en.getCount());
|
total += (value * en.getCount());
|
||||||
|
|
||||||
}
|
}
|
||||||
@ -612,34 +549,29 @@ public class IslandLevelCalculator {
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Finalizes the calculations and makes the report
|
* Finalizes the calculations and makes the report
|
||||||
*/
|
*/
|
||||||
public void tidyUp() {
|
public void tidyUp() {
|
||||||
// Finalize calculations
|
// Finalize calculations
|
||||||
results.rawBlockCount.addAndGet((long)(results.underWaterBlockCount.get() * addon.getSettings().getUnderWaterMultiplier()));
|
results.rawBlockCount
|
||||||
|
.addAndGet((long) (results.underWaterBlockCount.get() * addon.getSettings().getUnderWaterMultiplier()));
|
||||||
|
|
||||||
// Set the death penalty
|
// Set the death penalty
|
||||||
if (this.addon.getSettings().isSumTeamDeaths())
|
if (this.addon.getSettings().isSumTeamDeaths()) {
|
||||||
{
|
for (UUID uuid : this.island.getMemberSet()) {
|
||||||
for (UUID uuid : this.island.getMemberSet())
|
|
||||||
{
|
|
||||||
this.results.deathHandicap.addAndGet(this.addon.getPlayers().getDeaths(island.getWorld(), uuid));
|
this.results.deathHandicap.addAndGet(this.addon.getPlayers().getDeaths(island.getWorld(), uuid));
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
// At this point, it may be that the island has become unowned.
|
// At this point, it may be that the island has become unowned.
|
||||||
this.results.deathHandicap.set(this.island.getOwner() == null ? 0 :
|
this.results.deathHandicap.set(this.island.getOwner() == null ? 0
|
||||||
this.addon.getPlayers().getDeaths(island.getWorld(), this.island.getOwner()));
|
: this.addon.getPlayers().getDeaths(island.getWorld(), this.island.getOwner()));
|
||||||
}
|
}
|
||||||
|
|
||||||
long blockAndDeathPoints = this.results.rawBlockCount.get();
|
long blockAndDeathPoints = this.results.rawBlockCount.get();
|
||||||
this.results.totalPoints.set(blockAndDeathPoints);
|
this.results.totalPoints.set(blockAndDeathPoints);
|
||||||
|
|
||||||
if (this.addon.getSettings().getDeathPenalty() > 0)
|
if (this.addon.getSettings().getDeathPenalty() > 0) {
|
||||||
{
|
|
||||||
// Proper death penalty calculation.
|
// Proper death penalty calculation.
|
||||||
blockAndDeathPoints -= this.results.deathHandicap.get() * this.addon.getSettings().getDeathPenalty();
|
blockAndDeathPoints -= this.results.deathHandicap.get() * this.addon.getSettings().getDeathPenalty();
|
||||||
}
|
}
|
||||||
@ -674,11 +606,13 @@ public class IslandLevelCalculator {
|
|||||||
addon.getPlugin().logError("scanChunk not on Primary Thread!");
|
addon.getPlugin().logError("scanChunk not on Primary Thread!");
|
||||||
}
|
}
|
||||||
// Timeout check
|
// Timeout check
|
||||||
if (System.currentTimeMillis() - pipeliner.getInProcessQueue().get(this) > addon.getSettings().getCalculationTimeout() * 60000) {
|
if (System.currentTimeMillis()
|
||||||
|
- pipeliner.getInProcessQueue().get(this) > addon.getSettings().getCalculationTimeout() * 60000) {
|
||||||
// Done
|
// Done
|
||||||
pipeliner.getInProcessQueue().remove(this);
|
pipeliner.getInProcessQueue().remove(this);
|
||||||
getR().complete(new Results(Result.TIMEOUT));
|
getR().complete(new Results(Result.TIMEOUT));
|
||||||
addon.logError("Level calculation timed out after " + addon.getSettings().getCalculationTimeout() + "m for island: " + getIsland());
|
addon.logError("Level calculation timed out after " + addon.getSettings().getCalculationTimeout()
|
||||||
|
+ "m for island: " + getIsland());
|
||||||
if (!isNotZeroIsland()) {
|
if (!isNotZeroIsland()) {
|
||||||
addon.logError("Island level was being zeroed.");
|
addon.logError("Island level was being zeroed.");
|
||||||
}
|
}
|
||||||
@ -697,7 +631,8 @@ public class IslandLevelCalculator {
|
|||||||
long checkTime = System.currentTimeMillis();
|
long checkTime = System.currentTimeMillis();
|
||||||
finishTask = Bukkit.getScheduler().runTaskTimer(addon.getPlugin(), () -> {
|
finishTask = Bukkit.getScheduler().runTaskTimer(addon.getPlugin(), () -> {
|
||||||
// Check every half second if all the chests and stacks have been cleared
|
// Check every half second if all the chests and stacks have been cleared
|
||||||
if ((chestBlocks.isEmpty() && stackedBlocks.isEmpty()) || System.currentTimeMillis() - checkTime > MAX_AMOUNT) {
|
if ((chestBlocks.isEmpty() && stackedBlocks.isEmpty())
|
||||||
|
|| System.currentTimeMillis() - checkTime > MAX_AMOUNT) {
|
||||||
this.tidyUp();
|
this.tidyUp();
|
||||||
this.getR().complete(getResults());
|
this.getR().complete(getResults());
|
||||||
finishTask.cancel();
|
finishTask.cancel();
|
||||||
|
@ -0,0 +1,98 @@
|
|||||||
|
package world.bentobox.level.commands;
|
||||||
|
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Map.Entry;
|
||||||
|
import java.util.TreeMap;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
import org.bukkit.World;
|
||||||
|
|
||||||
|
import world.bentobox.bentobox.api.commands.CompositeCommand;
|
||||||
|
import world.bentobox.bentobox.api.localization.TextVariables;
|
||||||
|
import world.bentobox.bentobox.api.user.User;
|
||||||
|
import world.bentobox.level.Level;
|
||||||
|
import world.bentobox.level.objects.TopTenData;
|
||||||
|
|
||||||
|
public class AdminStatsCommand extends CompositeCommand {
|
||||||
|
|
||||||
|
private final Level level;
|
||||||
|
|
||||||
|
public AdminStatsCommand(Level addon, CompositeCommand parent) {
|
||||||
|
super(parent, "stats");
|
||||||
|
this.level = addon;
|
||||||
|
new AdminTopRemoveCommand(addon, this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setup() {
|
||||||
|
this.setPermission("admin.stats");
|
||||||
|
this.setOnlyPlayer(false);
|
||||||
|
this.setDescription("admin.stats.description");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean execute(User user, String label, List<String> args) {
|
||||||
|
user.sendMessage("admin.stats.title");
|
||||||
|
Map<World, TopTenData> topTenLists = level.getManager().getTopTenLists();
|
||||||
|
if (topTenLists.isEmpty()) {
|
||||||
|
user.sendMessage("admin.stats.no-data");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
for (Entry<World, TopTenData> en : topTenLists.entrySet()) {
|
||||||
|
user.sendMessage("admin.stats.world", TextVariables.NAME,
|
||||||
|
level.getPlugin().getIWM().getWorldName(en.getKey()));
|
||||||
|
Map<String, Long> topTen = en.getValue().getTopTen();
|
||||||
|
if (topTen.isEmpty()) {
|
||||||
|
user.sendMessage("admin.stats.no-data");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Calculating basic statistics
|
||||||
|
long sum = 0, max = Long.MIN_VALUE, min = Long.MAX_VALUE;
|
||||||
|
Map<Long, Integer> levelFrequency = new HashMap<>();
|
||||||
|
|
||||||
|
for (Long level : topTen.values()) {
|
||||||
|
sum += level;
|
||||||
|
max = Math.max(max, level);
|
||||||
|
min = Math.min(min, level);
|
||||||
|
levelFrequency.merge(level, 1, Integer::sum);
|
||||||
|
}
|
||||||
|
|
||||||
|
double average = sum / (double) topTen.size();
|
||||||
|
List<Long> sortedLevels = topTen.values().stream().sorted().collect(Collectors.toList());
|
||||||
|
long median = sortedLevels.get(sortedLevels.size() / 2);
|
||||||
|
Long mode = Collections.max(levelFrequency.entrySet(), Map.Entry.comparingByValue()).getKey();
|
||||||
|
|
||||||
|
// Logging basic statistics
|
||||||
|
user.sendMessage("admin.stats.average-level", TextVariables.NUMBER, String.valueOf(average));
|
||||||
|
user.sendMessage("admin.stats.median-level", TextVariables.NUMBER, String.valueOf(median));
|
||||||
|
user.sendMessage("admin.stats.mode-level", TextVariables.NUMBER, String.valueOf(mode));
|
||||||
|
user.sendMessage("admin.stats.highest-level", TextVariables.NUMBER, String.valueOf(max));
|
||||||
|
user.sendMessage("admin.stats.lowest-level", TextVariables.NUMBER, String.valueOf(min));
|
||||||
|
|
||||||
|
// Grouping data for distribution analysis
|
||||||
|
Map<String, Integer> rangeMap = new TreeMap<>();
|
||||||
|
for (Long level : topTen.values()) {
|
||||||
|
String range = getRange(level);
|
||||||
|
rangeMap.merge(range, 1, Integer::sum);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Logging distribution
|
||||||
|
user.sendMessage("admin.stats.distribution");
|
||||||
|
for (Map.Entry<String, Integer> entry : rangeMap.entrySet()) {
|
||||||
|
user.sendMessage(
|
||||||
|
entry.getKey() + ": " + entry.getValue() + " " + user.getTranslation("admin.stats.islands"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static String getRange(long level) {
|
||||||
|
long rangeStart = level / 100 * 100;
|
||||||
|
long rangeEnd = rangeStart + 99;
|
||||||
|
return rangeStart + "-" + rangeEnd;
|
||||||
|
}
|
||||||
|
}
|
@ -2,7 +2,7 @@ package world.bentobox.level.commands;
|
|||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.UUID;
|
import java.util.Optional;
|
||||||
|
|
||||||
import world.bentobox.bentobox.api.commands.CompositeCommand;
|
import world.bentobox.bentobox.api.commands.CompositeCommand;
|
||||||
import world.bentobox.bentobox.api.user.User;
|
import world.bentobox.bentobox.api.user.User;
|
||||||
@ -30,20 +30,16 @@ public class AdminTopCommand extends CompositeCommand {
|
|||||||
public boolean execute(User user, String label, List<String> args) {
|
public boolean execute(User user, String label, List<String> args) {
|
||||||
user.sendMessage("island.top.gui-title");
|
user.sendMessage("island.top.gui-title");
|
||||||
int rank = 0;
|
int rank = 0;
|
||||||
for (Map.Entry<UUID, Long> topTen : levelPlugin.getManager().getTopTen(getWorld(), Level.TEN).entrySet()) {
|
for (Map.Entry<String, Long> topTen : levelPlugin.getManager().getTopTen(getWorld(), Level.TEN).entrySet()) {
|
||||||
Island island = getPlugin().getIslands().getIsland(getWorld(), topTen.getKey());
|
Optional<Island> is = getPlugin().getIslands().getIslandById(topTen.getKey());
|
||||||
if (island != null) {
|
if (is.isPresent()) {
|
||||||
|
Island island = is.get();
|
||||||
rank++;
|
rank++;
|
||||||
user.sendMessage("admin.top.display",
|
user.sendMessage("admin.top.display", "[rank]", String.valueOf(rank), "[name]",
|
||||||
"[rank]",
|
this.getPlugin().getPlayers().getUser(island.getOwner()).getName(), "[level]",
|
||||||
String.valueOf(rank),
|
|
||||||
"[name]",
|
|
||||||
this.getPlugin().getPlayers().getUser(island.getOwner()).getName(),
|
|
||||||
"[level]",
|
|
||||||
String.valueOf(topTen.getValue()));
|
String.valueOf(topTen.getValue()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6,10 +6,12 @@ import java.util.Optional;
|
|||||||
import world.bentobox.bentobox.api.commands.CompositeCommand;
|
import world.bentobox.bentobox.api.commands.CompositeCommand;
|
||||||
import world.bentobox.bentobox.api.localization.TextVariables;
|
import world.bentobox.bentobox.api.localization.TextVariables;
|
||||||
import world.bentobox.bentobox.api.user.User;
|
import world.bentobox.bentobox.api.user.User;
|
||||||
|
import world.bentobox.bentobox.database.objects.Island;
|
||||||
import world.bentobox.level.Level;
|
import world.bentobox.level.Level;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Removes a player from the top ten
|
* Removes a player from the top ten
|
||||||
|
*
|
||||||
* @author tastybento
|
* @author tastybento
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
@ -31,8 +33,11 @@ public class AdminTopRemoveCommand extends CompositeCommand {
|
|||||||
this.setDescription("admin.top.remove.description");
|
this.setDescription("admin.top.remove.description");
|
||||||
}
|
}
|
||||||
|
|
||||||
/* (non-Javadoc)
|
/*
|
||||||
* @see world.bentobox.bentobox.api.commands.BentoBoxCommand#canExecute(world.bentobox.bentobox.api.user.User, java.lang.String, java.util.List)
|
* (non-Javadoc)
|
||||||
|
*
|
||||||
|
* @see world.bentobox.bentobox.api.commands.BentoBoxCommand#canExecute(world.
|
||||||
|
* bentobox.bentobox.api.user.User, java.lang.String, java.util.List)
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
public boolean canExecute(User user, String label, List<String> args) {
|
public boolean canExecute(User user, String label, List<String> args) {
|
||||||
@ -51,14 +56,18 @@ public class AdminTopRemoveCommand extends CompositeCommand {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean execute(User user, String label, List<String> args) {
|
public boolean execute(User user, String label, List<String> args) {
|
||||||
addon.getManager().removeEntry(getWorld(), target.getUniqueId());
|
// Removes islands that this target is an owner of
|
||||||
|
getIslands().getIslands(getWorld(), target.getUniqueId()).stream()
|
||||||
|
.filter(is -> target.getUniqueId().equals(is.getOwner()))
|
||||||
|
.forEach(island -> addon.getManager().removeEntry(getWorld(), island.getUniqueId()));
|
||||||
user.sendMessage("general.success");
|
user.sendMessage("general.success");
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Optional<List<String>> tabComplete(User user, String alias, List<String> args) {
|
public Optional<List<String>> tabComplete(User user, String alias, List<String> args) {
|
||||||
return Optional.of(addon.getManager().getTopTen(getWorld(), Level.TEN).keySet().stream().map(addon.getPlayers()::getName)
|
return Optional.of(addon.getManager().getTopTen(getWorld(), Level.TEN).keySet().stream()
|
||||||
.filter(n -> !n.isEmpty()).toList());
|
.map(getIslands()::getIslandById).flatMap(Optional::stream).map(Island::getOwner)
|
||||||
|
.map(addon.getPlayers()::getName).filter(n -> !n.isEmpty()).toList());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -60,10 +60,16 @@ public class BlockConfig {
|
|||||||
if (bWorld != null) {
|
if (bWorld != null) {
|
||||||
ConfigurationSection worldValues = worlds.getConfigurationSection(world);
|
ConfigurationSection worldValues = worlds.getConfigurationSection(world);
|
||||||
for (String material : Objects.requireNonNull(worldValues).getKeys(false)) {
|
for (String material : Objects.requireNonNull(worldValues).getKeys(false)) {
|
||||||
|
try {
|
||||||
Material mat = Material.valueOf(material);
|
Material mat = Material.valueOf(material);
|
||||||
Map<Material, Integer> values = worldBlockValues.getOrDefault(bWorld, new EnumMap<>(Material.class));
|
Map<Material, Integer> values = worldBlockValues.getOrDefault(bWorld,
|
||||||
|
new EnumMap<>(Material.class));
|
||||||
values.put(mat, worldValues.getInt(material));
|
values.put(mat, worldValues.getInt(material));
|
||||||
worldBlockValues.put(bWorld, values);
|
worldBlockValues.put(bWorld, values);
|
||||||
|
} catch (Exception e) {
|
||||||
|
addon.logError(
|
||||||
|
"Unknown material (" + material + ") in blockconfig.yml worlds section. Skipping...");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
addon.logWarning("Level Addon: No such world in blockconfig.yml : " + world);
|
addon.logWarning("Level Addon: No such world in blockconfig.yml : " + world);
|
||||||
@ -97,7 +103,7 @@ public class BlockConfig {
|
|||||||
Material mat = Material.valueOf(material);
|
Material mat = Material.valueOf(material);
|
||||||
bl.put(mat, limits.getInt(material, 0));
|
bl.put(mat, limits.getInt(material, 0));
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
addon.logWarning("Unknown material (" + material + ") in blockconfig.yml Limits section. Skipping...");
|
addon.logError("Unknown material (" + material + ") in blockconfig.yml Limits section. Skipping...");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return bl;
|
return bl;
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
package world.bentobox.level.config;
|
package world.bentobox.level.config;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
@ -126,6 +127,12 @@ public class ConfigSettings implements ConfigObject {
|
|||||||
@ConfigEntry(path = "include-shulkers-in-chest")
|
@ConfigEntry(path = "include-shulkers-in-chest")
|
||||||
private boolean includeShulkersInChest = false;
|
private boolean includeShulkersInChest = false;
|
||||||
|
|
||||||
|
@ConfigComment("")
|
||||||
|
@ConfigComment("Disables hooking with other plugins.")
|
||||||
|
@ConfigComment("Example: disabled-plugin-hooks: [UltimateStacker, RoseStacker]")
|
||||||
|
@ConfigEntry(path = "disabled-plugin-hooks")
|
||||||
|
private List<String> disabledPluginHooks = new ArrayList<>();
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return the gameModes
|
* @return the gameModes
|
||||||
@ -404,4 +411,12 @@ public class ConfigSettings implements ConfigObject {
|
|||||||
public void setIncludeShulkersInChest(boolean includeShulkersInChest) {
|
public void setIncludeShulkersInChest(boolean includeShulkersInChest) {
|
||||||
this.includeShulkersInChest = includeShulkersInChest;
|
this.includeShulkersInChest = includeShulkersInChest;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public List<String> getDisabledPluginHooks() {
|
||||||
|
return disabledPluginHooks;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDisabledPluginHooks(List<String> disabledPluginHooks) {
|
||||||
|
this.disabledPluginHooks = disabledPluginHooks;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,5 @@
|
|||||||
package world.bentobox.level.listeners;
|
package world.bentobox.level.listeners;
|
||||||
|
|
||||||
import java.util.UUID;
|
|
||||||
|
|
||||||
import org.bukkit.World;
|
import org.bukkit.World;
|
||||||
import org.bukkit.event.EventHandler;
|
import org.bukkit.event.EventHandler;
|
||||||
import org.bukkit.event.EventPriority;
|
import org.bukkit.event.EventPriority;
|
||||||
@ -21,7 +19,9 @@ import world.bentobox.bentobox.database.objects.Island;
|
|||||||
import world.bentobox.level.Level;
|
import world.bentobox.level.Level;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Listens for new islands or ownership changes and sets the level to zero automatically
|
* Listens for new islands or ownership changes and sets the level to zero
|
||||||
|
* automatically
|
||||||
|
*
|
||||||
* @author tastybento
|
* @author tastybento
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
@ -54,18 +54,14 @@ public class IslandActivitiesListeners implements Listener {
|
|||||||
private void zeroIsland(final Island island) {
|
private void zeroIsland(final Island island) {
|
||||||
// Clear the island setting
|
// Clear the island setting
|
||||||
if (island.getOwner() != null && island.getWorld() != null) {
|
if (island.getOwner() != null && island.getWorld() != null) {
|
||||||
addon.getPipeliner().zeroIsland(island).thenAccept(results ->
|
addon.getPipeliner().zeroIsland(island)
|
||||||
addon.getManager().setInitialIslandLevel(island, results.getLevel()));
|
.thenAccept(results -> addon.getManager().setInitialIslandLevel(island, results.getLevel()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@EventHandler(priority = EventPriority.MONITOR, ignoreCancelled = true)
|
@EventHandler(priority = EventPriority.MONITOR, ignoreCancelled = true)
|
||||||
public void onIslandDelete(IslandPreclearEvent e) {
|
public void onIslandDelete(IslandPreclearEvent e) {
|
||||||
|
remove(e.getIsland().getWorld(), e.getIsland().getUniqueId());
|
||||||
// Remove player from the top ten and level
|
|
||||||
UUID uuid = e.getIsland().getOwner();
|
|
||||||
World world = e.getIsland().getWorld();
|
|
||||||
remove(world, uuid);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@EventHandler(priority = EventPriority.MONITOR, ignoreCancelled = true)
|
@EventHandler(priority = EventPriority.MONITOR, ignoreCancelled = true)
|
||||||
@ -74,7 +70,7 @@ public class IslandActivitiesListeners implements Listener {
|
|||||||
addon.getManager().deleteIsland(e.getIsland().getUniqueId());
|
addon.getManager().deleteIsland(e.getIsland().getUniqueId());
|
||||||
}
|
}
|
||||||
|
|
||||||
private void remove(World world, UUID uuid) {
|
private void remove(World world, String uuid) {
|
||||||
if (uuid != null && world != null) {
|
if (uuid != null && world != null) {
|
||||||
addon.getManager().removeEntry(world, uuid);
|
addon.getManager().removeEntry(world, uuid);
|
||||||
}
|
}
|
||||||
@ -83,43 +79,43 @@ public class IslandActivitiesListeners implements Listener {
|
|||||||
@EventHandler(priority = EventPriority.NORMAL, ignoreCancelled = true)
|
@EventHandler(priority = EventPriority.NORMAL, ignoreCancelled = true)
|
||||||
public void onNewIslandOwner(TeamSetownerEvent e) {
|
public void onNewIslandOwner(TeamSetownerEvent e) {
|
||||||
|
|
||||||
// Remove player from the top ten and level
|
// Remove island from the top ten and level
|
||||||
remove(e.getIsland().getWorld(), e.getIsland().getOwner());
|
remove(e.getIsland().getWorld(), e.getIsland().getUniqueId());
|
||||||
}
|
}
|
||||||
|
|
||||||
@EventHandler(priority = EventPriority.NORMAL, ignoreCancelled = true)
|
@EventHandler(priority = EventPriority.NORMAL, ignoreCancelled = true)
|
||||||
public void onIsland(TeamJoinedEvent e) {
|
public void onIsland(TeamJoinedEvent e) {
|
||||||
|
// TODO: anything to do here?
|
||||||
// Remove player from the top ten and level
|
// Remove player from the top ten and level
|
||||||
remove(e.getIsland().getWorld(), e.getPlayerUUID());
|
// remove(e.getIsland().getWorld(), e.getPlayerUUID());
|
||||||
}
|
}
|
||||||
|
|
||||||
@EventHandler(priority = EventPriority.NORMAL, ignoreCancelled = true)
|
@EventHandler(priority = EventPriority.NORMAL, ignoreCancelled = true)
|
||||||
public void onIsland(IslandUnregisteredEvent e) {
|
public void onIsland(IslandUnregisteredEvent e) {
|
||||||
|
|
||||||
// Remove player from the top ten
|
// Remove island from the top ten
|
||||||
remove(e.getIsland().getWorld(), e.getPlayerUUID());
|
remove(e.getIsland().getWorld(), e.getIsland().getUniqueId());
|
||||||
}
|
}
|
||||||
|
|
||||||
@EventHandler(priority = EventPriority.NORMAL, ignoreCancelled = true)
|
@EventHandler(priority = EventPriority.NORMAL, ignoreCancelled = true)
|
||||||
public void onIsland(IslandRegisteredEvent e) {
|
public void onIsland(IslandRegisteredEvent e) {
|
||||||
|
// TODO: anything to do here?
|
||||||
// Remove player from the top ten
|
// Remove player from the top ten
|
||||||
remove(e.getIsland().getWorld(), e.getPlayerUUID());
|
// remove(e.getIsland().getWorld(), e.getPlayerUUID());
|
||||||
}
|
}
|
||||||
|
|
||||||
@EventHandler(priority = EventPriority.NORMAL, ignoreCancelled = true)
|
@EventHandler(priority = EventPriority.NORMAL, ignoreCancelled = true)
|
||||||
public void onIsland(TeamLeaveEvent e) {
|
public void onIsland(TeamLeaveEvent e) {
|
||||||
|
// TODO: anything to do here?
|
||||||
// Remove player from the top ten and level
|
// Remove player from the top ten and level
|
||||||
remove(e.getIsland().getWorld(), e.getPlayerUUID());
|
// remove(e.getIsland().getWorld(), e.getPlayerUUID());
|
||||||
}
|
}
|
||||||
|
|
||||||
@EventHandler(priority = EventPriority.NORMAL, ignoreCancelled = true)
|
@EventHandler(priority = EventPriority.NORMAL, ignoreCancelled = true)
|
||||||
public void onIsland(TeamKickEvent e) {
|
public void onIsland(TeamKickEvent e) {
|
||||||
|
//// TODO: anything to do here?
|
||||||
// Remove player from the top ten and level
|
// Remove player from the top ten and level
|
||||||
remove(e.getIsland().getWorld(), e.getPlayerUUID());
|
// remove(e.getIsland().getWorld(), e.getPlayerUUID());
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -3,56 +3,41 @@ package world.bentobox.level.objects;
|
|||||||
import java.util.LinkedHashMap;
|
import java.util.LinkedHashMap;
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.UUID;
|
|
||||||
|
|
||||||
import org.bukkit.World;
|
import org.bukkit.World;
|
||||||
|
|
||||||
import com.google.gson.annotations.Expose;
|
import com.google.gson.annotations.Expose;
|
||||||
|
|
||||||
import world.bentobox.bentobox.database.objects.DataObject;
|
|
||||||
import world.bentobox.bentobox.database.objects.Table;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This class stores the top ten.
|
* This class stores the top ten.
|
||||||
|
*
|
||||||
* @author tastybento
|
* @author tastybento
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
@Table(name = "TopTenData")
|
public class TopTenData {
|
||||||
public class TopTenData implements DataObject {
|
|
||||||
|
|
||||||
// UniqueId is the world name
|
// UniqueId is the world name
|
||||||
@Expose
|
@Expose
|
||||||
private String uniqueId = "";
|
private String uniqueId = "";
|
||||||
@Expose
|
@Expose
|
||||||
private Map<UUID, Long> topTen = new LinkedHashMap<>();
|
private Map<String, Long> topTen = new LinkedHashMap<>();
|
||||||
|
|
||||||
public TopTenData(World k) {
|
public TopTenData(World k) {
|
||||||
uniqueId = k.getName().toLowerCase(Locale.ENGLISH);
|
uniqueId = k.getName().toLowerCase(Locale.ENGLISH);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
|
||||||
public String getUniqueId() {
|
|
||||||
// This is the world name
|
|
||||||
return uniqueId;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setUniqueId(String uniqueId) {
|
|
||||||
// This is the world name - make it always lowercase
|
|
||||||
this.uniqueId = uniqueId.toLowerCase(Locale.ENGLISH);
|
|
||||||
}
|
|
||||||
/**
|
/**
|
||||||
* @return the topTen
|
* @return the topTen
|
||||||
*/
|
*/
|
||||||
public Map<UUID, Long> getTopTen() {
|
public Map<String, Long> getTopTen() {
|
||||||
return topTen;
|
return topTen;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param topTen the topTen to set
|
* @param topTen the topTen to set
|
||||||
*/
|
*/
|
||||||
public void setTopTen(Map<UUID, Long> topTen) {
|
public void setTopTen(Map<String, Long> topTen) {
|
||||||
this.topTen = topTen;
|
this.topTen = topTen;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
package world.bentobox.level.panels;
|
package world.bentobox.level.panels;
|
||||||
|
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Comparator;
|
import java.util.Comparator;
|
||||||
@ -30,40 +29,32 @@ import world.bentobox.level.Level;
|
|||||||
import world.bentobox.level.objects.IslandLevels;
|
import world.bentobox.level.objects.IslandLevels;
|
||||||
import world.bentobox.level.util.Utils;
|
import world.bentobox.level.util.Utils;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This class opens GUI that shows generator view for user.
|
* This class opens GUI that shows generator view for user.
|
||||||
*/
|
*/
|
||||||
public class DetailsPanel
|
public class DetailsPanel {
|
||||||
{
|
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
// Section: Internal Constructor
|
// Section: Internal Constructor
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This is internal constructor. It is used internally in current class to avoid creating objects everywhere.
|
* This is internal constructor. It is used internally in current class to avoid
|
||||||
|
* creating objects everywhere.
|
||||||
*
|
*
|
||||||
* @param addon Level object
|
* @param addon Level object
|
||||||
* @param world World where user is operating
|
* @param world World where user is operating
|
||||||
* @param user User who opens panel
|
* @param user User who opens panel
|
||||||
*/
|
*/
|
||||||
private DetailsPanel(Level addon,
|
private DetailsPanel(Level addon, World world, User user) {
|
||||||
World world,
|
|
||||||
User user)
|
|
||||||
{
|
|
||||||
this.addon = addon;
|
this.addon = addon;
|
||||||
this.world = world;
|
this.world = world;
|
||||||
this.user = user;
|
this.user = user;
|
||||||
|
|
||||||
this.island = this.addon.getIslands().getIsland(world, user);
|
this.island = this.addon.getIslands().getIsland(world, user);
|
||||||
|
|
||||||
if (this.island != null)
|
if (this.island != null) {
|
||||||
{
|
|
||||||
this.levelsData = this.addon.getManager().getLevelsData(this.island);
|
this.levelsData = this.addon.getManager().getLevelsData(this.island);
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
this.levelsData = null;
|
this.levelsData = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -75,21 +66,17 @@ public class DetailsPanel
|
|||||||
this.updateFilters();
|
this.updateFilters();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method builds this GUI.
|
* This method builds this GUI.
|
||||||
*/
|
*/
|
||||||
private void build()
|
private void build() {
|
||||||
{
|
if (this.island == null || this.levelsData == null) {
|
||||||
if (this.island == null || this.levelsData == null)
|
|
||||||
{
|
|
||||||
// Nothing to see.
|
// Nothing to see.
|
||||||
Utils.sendMessage(this.user, this.user.getTranslation("general.errors.no-island"));
|
Utils.sendMessage(this.user, this.user.getTranslation("general.errors.no-island"));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.levelsData.getMdCount().isEmpty() && this.levelsData.getUwCount().isEmpty())
|
if (this.levelsData.getMdCount().isEmpty() && this.levelsData.getUwCount().isEmpty()) {
|
||||||
{
|
|
||||||
// Nothing to see.
|
// Nothing to see.
|
||||||
Utils.sendMessage(this.user, this.user.getTranslation("level.conversations.no-data"));
|
Utils.sendMessage(this.user, this.user.getTranslation("level.conversations.no-data"));
|
||||||
return;
|
return;
|
||||||
@ -117,16 +104,13 @@ public class DetailsPanel
|
|||||||
panelBuilder.build();
|
panelBuilder.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method updates filter of elements based on tabs.
|
* This method updates filter of elements based on tabs.
|
||||||
*/
|
*/
|
||||||
private void updateFilters()
|
private void updateFilters() {
|
||||||
{
|
|
||||||
this.materialCountList.clear();
|
this.materialCountList.clear();
|
||||||
|
|
||||||
switch (this.activeTab)
|
switch (this.activeTab) {
|
||||||
{
|
|
||||||
case ALL_BLOCKS -> {
|
case ALL_BLOCKS -> {
|
||||||
Map<Material, Integer> materialCountMap = new EnumMap<>(Material.class);
|
Map<Material, Integer> materialCountMap = new EnumMap<>(Material.class);
|
||||||
|
|
||||||
@ -136,9 +120,8 @@ public class DetailsPanel
|
|||||||
this.levelsData.getUwCount().forEach((material, count) -> materialCountMap.put(material,
|
this.levelsData.getUwCount().forEach((material, count) -> materialCountMap.put(material,
|
||||||
materialCountMap.computeIfAbsent(material, key -> 0) + count));
|
materialCountMap.computeIfAbsent(material, key -> 0) + count));
|
||||||
|
|
||||||
materialCountMap.entrySet().stream().sorted((Map.Entry.comparingByKey())).
|
materialCountMap.entrySet().stream().sorted((Map.Entry.comparingByKey()))
|
||||||
forEachOrdered(entry ->
|
.forEachOrdered(entry -> this.materialCountList.add(new Pair<>(entry.getKey(), entry.getValue())));
|
||||||
this.materialCountList.add(new Pair<>(entry.getKey(), entry.getValue())));
|
|
||||||
}
|
}
|
||||||
case ABOVE_SEA_LEVEL -> this.levelsData.getMdCount().entrySet().stream().sorted((Map.Entry.comparingByKey()))
|
case ABOVE_SEA_LEVEL -> this.levelsData.getMdCount().entrySet().stream().sorted((Map.Entry.comparingByKey()))
|
||||||
.forEachOrdered(entry -> this.materialCountList.add(new Pair<>(entry.getKey(), entry.getValue())));
|
.forEachOrdered(entry -> this.materialCountList.add(new Pair<>(entry.getKey(), entry.getValue())));
|
||||||
@ -157,57 +140,44 @@ public class DetailsPanel
|
|||||||
|
|
||||||
Comparator<Pair<Material, Integer>> sorter;
|
Comparator<Pair<Material, Integer>> sorter;
|
||||||
|
|
||||||
switch (this.activeFilter)
|
switch (this.activeFilter) {
|
||||||
{
|
case COUNT -> {
|
||||||
case COUNT ->
|
sorter = (o1, o2) -> {
|
||||||
{
|
if (o1.getValue().equals(o2.getValue())) {
|
||||||
sorter = (o1, o2) ->
|
|
||||||
{
|
|
||||||
if (o1.getValue().equals(o2.getValue()))
|
|
||||||
{
|
|
||||||
String o1Name = Utils.prettifyObject(o1.getKey(), this.user);
|
String o1Name = Utils.prettifyObject(o1.getKey(), this.user);
|
||||||
String o2Name = Utils.prettifyObject(o2.getKey(), this.user);
|
String o2Name = Utils.prettifyObject(o2.getKey(), this.user);
|
||||||
|
|
||||||
return String.CASE_INSENSITIVE_ORDER.compare(o1Name, o2Name);
|
return String.CASE_INSENSITIVE_ORDER.compare(o1Name, o2Name);
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
return Integer.compare(o2.getValue(), o1.getValue());
|
return Integer.compare(o2.getValue(), o1.getValue());
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
case VALUE ->
|
case VALUE -> {
|
||||||
{
|
sorter = (o1, o2) -> {
|
||||||
sorter = (o1, o2) ->
|
|
||||||
{
|
|
||||||
int blockLimit = this.addon.getBlockConfig().getBlockLimits().getOrDefault(o1.getKey(), 0);
|
int blockLimit = this.addon.getBlockConfig().getBlockLimits().getOrDefault(o1.getKey(), 0);
|
||||||
int o1Count = blockLimit > 0 ? Math.min(o1.getValue(), blockLimit) : o1.getValue();
|
int o1Count = blockLimit > 0 ? Math.min(o1.getValue(), blockLimit) : o1.getValue();
|
||||||
|
|
||||||
blockLimit = this.addon.getBlockConfig().getBlockLimits().getOrDefault(o2.getKey(), 0);
|
blockLimit = this.addon.getBlockConfig().getBlockLimits().getOrDefault(o2.getKey(), 0);
|
||||||
int o2Count = blockLimit > 0 ? Math.min(o2.getValue(), blockLimit) : o2.getValue();
|
int o2Count = blockLimit > 0 ? Math.min(o2.getValue(), blockLimit) : o2.getValue();
|
||||||
|
|
||||||
long o1Value = (long) o1Count *
|
long o1Value = (long) o1Count
|
||||||
this.addon.getBlockConfig().getBlockValues().getOrDefault(o1.getKey(), 0);
|
* this.addon.getBlockConfig().getBlockValues().getOrDefault(o1.getKey(), 0);
|
||||||
long o2Value = (long) o2Count *
|
long o2Value = (long) o2Count
|
||||||
this.addon.getBlockConfig().getBlockValues().getOrDefault(o2.getKey(), 0);
|
* this.addon.getBlockConfig().getBlockValues().getOrDefault(o2.getKey(), 0);
|
||||||
|
|
||||||
if (o1Value == o2Value)
|
if (o1Value == o2Value) {
|
||||||
{
|
|
||||||
String o1Name = Utils.prettifyObject(o1.getKey(), this.user);
|
String o1Name = Utils.prettifyObject(o1.getKey(), this.user);
|
||||||
String o2Name = Utils.prettifyObject(o2.getKey(), this.user);
|
String o2Name = Utils.prettifyObject(o2.getKey(), this.user);
|
||||||
|
|
||||||
return String.CASE_INSENSITIVE_ORDER.compare(o1Name, o2Name);
|
return String.CASE_INSENSITIVE_ORDER.compare(o1Name, o2Name);
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
return Long.compare(o2Value, o1Value);
|
return Long.compare(o2Value, o1Value);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
default ->
|
default -> {
|
||||||
{
|
sorter = (o1, o2) -> {
|
||||||
sorter = (o1, o2) ->
|
|
||||||
{
|
|
||||||
String o1Name = Utils.prettifyObject(o1.getKey(), this.user);
|
String o1Name = Utils.prettifyObject(o1.getKey(), this.user);
|
||||||
String o2Name = Utils.prettifyObject(o2.getKey(), this.user);
|
String o2Name = Utils.prettifyObject(o2.getKey(), this.user);
|
||||||
|
|
||||||
@ -221,12 +191,10 @@ public class DetailsPanel
|
|||||||
this.pageIndex = 0;
|
this.pageIndex = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
// Section: Tab Button Type
|
// Section: Tab Button Type
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create tab button panel item.
|
* Create tab button panel item.
|
||||||
*
|
*
|
||||||
@ -234,24 +202,20 @@ public class DetailsPanel
|
|||||||
* @param slot the slot
|
* @param slot the slot
|
||||||
* @return the panel item
|
* @return the panel item
|
||||||
*/
|
*/
|
||||||
private PanelItem createTabButton(ItemTemplateRecord template, TemplatedPanel.ItemSlot slot)
|
private PanelItem createTabButton(ItemTemplateRecord template, TemplatedPanel.ItemSlot slot) {
|
||||||
{
|
|
||||||
PanelItemBuilder builder = new PanelItemBuilder();
|
PanelItemBuilder builder = new PanelItemBuilder();
|
||||||
|
|
||||||
if (template.icon() != null)
|
if (template.icon() != null) {
|
||||||
{
|
|
||||||
// Set icon
|
// Set icon
|
||||||
builder.icon(template.icon().clone());
|
builder.icon(template.icon().clone());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (template.title() != null)
|
if (template.title() != null) {
|
||||||
{
|
|
||||||
// Set title
|
// Set title
|
||||||
builder.name(this.user.getTranslation(this.world, template.title()));
|
builder.name(this.user.getTranslation(this.world, template.title()));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (template.description() != null)
|
if (template.description() != null) {
|
||||||
{
|
|
||||||
// Set description
|
// Set description
|
||||||
builder.description(this.user.getTranslation(this.world, template.description()));
|
builder.description(this.user.getTranslation(this.world, template.description()));
|
||||||
}
|
}
|
||||||
@ -261,17 +225,13 @@ public class DetailsPanel
|
|||||||
// Get only possible actions, by removing all inactive ones.
|
// Get only possible actions, by removing all inactive ones.
|
||||||
List<ItemTemplateRecord.ActionRecords> activeActions = new ArrayList<>(template.actions());
|
List<ItemTemplateRecord.ActionRecords> activeActions = new ArrayList<>(template.actions());
|
||||||
|
|
||||||
activeActions.removeIf(action ->
|
activeActions.removeIf(action -> "VIEW".equalsIgnoreCase(action.actionType()) && this.activeTab == tab);
|
||||||
"VIEW".equalsIgnoreCase(action.actionType()) && this.activeTab == tab);
|
|
||||||
|
|
||||||
// Add Click handler
|
// Add Click handler
|
||||||
builder.clickHandler((panel, user, clickType, i) ->
|
builder.clickHandler((panel, user, clickType, i) -> {
|
||||||
{
|
for (ItemTemplateRecord.ActionRecords action : activeActions) {
|
||||||
for (ItemTemplateRecord.ActionRecords action : activeActions)
|
|
||||||
{
|
|
||||||
if ((clickType == action.clickType() || ClickType.UNKNOWN.equals(action.clickType()))
|
if ((clickType == action.clickType() || ClickType.UNKNOWN.equals(action.clickType()))
|
||||||
&& "VIEW".equalsIgnoreCase(action.actionType()))
|
&& "VIEW".equalsIgnoreCase(action.actionType())) {
|
||||||
{
|
|
||||||
this.activeTab = tab;
|
this.activeTab = tab;
|
||||||
|
|
||||||
// Update filters.
|
// Update filters.
|
||||||
@ -284,15 +244,12 @@ public class DetailsPanel
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Collect tooltips.
|
// Collect tooltips.
|
||||||
List<String> tooltips = activeActions.stream().
|
List<String> tooltips = activeActions.stream().filter(action -> action.tooltip() != null)
|
||||||
filter(action -> action.tooltip() != null).
|
.map(action -> this.user.getTranslation(this.world, action.tooltip())).filter(text -> !text.isBlank())
|
||||||
map(action -> this.user.getTranslation(this.world, action.tooltip())).
|
.collect(Collectors.toCollection(() -> new ArrayList<>(template.actions().size())));
|
||||||
filter(text -> !text.isBlank()).
|
|
||||||
collect(Collectors.toCollection(() -> new ArrayList<>(template.actions().size())));
|
|
||||||
|
|
||||||
// Add tooltips.
|
// Add tooltips.
|
||||||
if (!tooltips.isEmpty())
|
if (!tooltips.isEmpty()) {
|
||||||
{
|
|
||||||
// Empty line and tooltips.
|
// Empty line and tooltips.
|
||||||
builder.description("");
|
builder.description("");
|
||||||
builder.description(tooltips);
|
builder.description(tooltips);
|
||||||
@ -303,7 +260,6 @@ public class DetailsPanel
|
|||||||
return builder.build();
|
return builder.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create next button panel item.
|
* Create next button panel item.
|
||||||
*
|
*
|
||||||
@ -311,77 +267,61 @@ public class DetailsPanel
|
|||||||
* @param slot the slot
|
* @param slot the slot
|
||||||
* @return the panel item
|
* @return the panel item
|
||||||
*/
|
*/
|
||||||
private PanelItem createFilterButton(ItemTemplateRecord template, TemplatedPanel.ItemSlot slot)
|
private PanelItem createFilterButton(ItemTemplateRecord template, TemplatedPanel.ItemSlot slot) {
|
||||||
{
|
|
||||||
PanelItemBuilder builder = new PanelItemBuilder();
|
PanelItemBuilder builder = new PanelItemBuilder();
|
||||||
|
|
||||||
if (template.icon() != null)
|
if (template.icon() != null) {
|
||||||
{
|
|
||||||
// Set icon
|
// Set icon
|
||||||
builder.icon(template.icon().clone());
|
builder.icon(template.icon().clone());
|
||||||
}
|
}
|
||||||
|
|
||||||
Filter filter;
|
Filter filter;
|
||||||
|
|
||||||
if (slot.amountMap().getOrDefault("FILTER", 0) > 1)
|
if (slot.amountMap().getOrDefault("FILTER", 0) > 1) {
|
||||||
{
|
|
||||||
filter = Enums.getIfPresent(Filter.class, String.valueOf(template.dataMap().get("filter"))).or(Filter.NAME);
|
filter = Enums.getIfPresent(Filter.class, String.valueOf(template.dataMap().get("filter"))).or(Filter.NAME);
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
filter = this.activeFilter;
|
filter = this.activeFilter;
|
||||||
}
|
}
|
||||||
|
|
||||||
final String reference = "level.gui.buttons.filters.";
|
final String reference = "level.gui.buttons.filters.";
|
||||||
|
|
||||||
if (template.title() != null)
|
if (template.title() != null) {
|
||||||
{
|
|
||||||
// Set title
|
// Set title
|
||||||
builder.name(this.user.getTranslation(this.world, template.title().replace("[filter]", filter.name().toLowerCase())));
|
builder.name(this.user.getTranslation(this.world,
|
||||||
}
|
template.title().replace("[filter]", filter.name().toLowerCase())));
|
||||||
else
|
} else {
|
||||||
{
|
|
||||||
builder.name(this.user.getTranslation(this.world, reference + filter.name().toLowerCase() + ".name"));
|
builder.name(this.user.getTranslation(this.world, reference + filter.name().toLowerCase() + ".name"));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (template.description() != null)
|
if (template.description() != null) {
|
||||||
{
|
|
||||||
// Set description
|
// Set description
|
||||||
builder.description(this.user.getTranslation(this.world, template.description().replace("[filter]", filter.name().toLowerCase())));
|
builder.description(this.user.getTranslation(this.world,
|
||||||
}
|
template.description().replace("[filter]", filter.name().toLowerCase())));
|
||||||
else
|
} else {
|
||||||
{
|
builder.name(
|
||||||
builder.name(this.user.getTranslation(this.world, reference + filter.name().toLowerCase() + ".description"));
|
this.user.getTranslation(this.world, reference + filter.name().toLowerCase() + ".description"));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get only possible actions, by removing all inactive ones.
|
// Get only possible actions, by removing all inactive ones.
|
||||||
List<ItemTemplateRecord.ActionRecords> activeActions = new ArrayList<>(template.actions());
|
List<ItemTemplateRecord.ActionRecords> activeActions = new ArrayList<>(template.actions());
|
||||||
|
|
||||||
// Add Click handler
|
// Add Click handler
|
||||||
builder.clickHandler((panel, user, clickType, i) ->
|
builder.clickHandler((panel, user, clickType, i) -> {
|
||||||
{
|
for (ItemTemplateRecord.ActionRecords action : activeActions) {
|
||||||
for (ItemTemplateRecord.ActionRecords action : activeActions)
|
if (clickType == action.clickType() || ClickType.UNKNOWN.equals(action.clickType())) {
|
||||||
{
|
if ("UP".equalsIgnoreCase(action.actionType())) {
|
||||||
if (clickType == action.clickType() || ClickType.UNKNOWN.equals(action.clickType()))
|
|
||||||
{
|
|
||||||
if ("UP".equalsIgnoreCase(action.actionType()))
|
|
||||||
{
|
|
||||||
this.activeFilter = Utils.getNextValue(Filter.values(), filter);
|
this.activeFilter = Utils.getNextValue(Filter.values(), filter);
|
||||||
|
|
||||||
// Update filters.
|
// Update filters.
|
||||||
this.updateFilters();
|
this.updateFilters();
|
||||||
this.build();
|
this.build();
|
||||||
}
|
} else if ("DOWN".equalsIgnoreCase(action.actionType())) {
|
||||||
else if ("DOWN".equalsIgnoreCase(action.actionType()))
|
|
||||||
{
|
|
||||||
this.activeFilter = Utils.getPreviousValue(Filter.values(), filter);
|
this.activeFilter = Utils.getPreviousValue(Filter.values(), filter);
|
||||||
|
|
||||||
// Update filters.
|
// Update filters.
|
||||||
this.updateFilters();
|
this.updateFilters();
|
||||||
this.build();
|
this.build();
|
||||||
}
|
} else if ("SELECT".equalsIgnoreCase(action.actionType())) {
|
||||||
else if ("SELECT".equalsIgnoreCase(action.actionType()))
|
|
||||||
{
|
|
||||||
this.activeFilter = filter;
|
this.activeFilter = filter;
|
||||||
|
|
||||||
// Update filters.
|
// Update filters.
|
||||||
@ -395,15 +335,12 @@ public class DetailsPanel
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Collect tooltips.
|
// Collect tooltips.
|
||||||
List<String> tooltips = activeActions.stream().
|
List<String> tooltips = activeActions.stream().filter(action -> action.tooltip() != null)
|
||||||
filter(action -> action.tooltip() != null).
|
.map(action -> this.user.getTranslation(this.world, action.tooltip())).filter(text -> !text.isBlank())
|
||||||
map(action -> this.user.getTranslation(this.world, action.tooltip())).
|
.collect(Collectors.toCollection(() -> new ArrayList<>(template.actions().size())));
|
||||||
filter(text -> !text.isBlank()).
|
|
||||||
collect(Collectors.toCollection(() -> new ArrayList<>(template.actions().size())));
|
|
||||||
|
|
||||||
// Add tooltips.
|
// Add tooltips.
|
||||||
if (!tooltips.isEmpty())
|
if (!tooltips.isEmpty()) {
|
||||||
{
|
|
||||||
// Empty line and tooltips.
|
// Empty line and tooltips.
|
||||||
builder.description("");
|
builder.description("");
|
||||||
builder.description(tooltips);
|
builder.description(tooltips);
|
||||||
@ -414,12 +351,10 @@ public class DetailsPanel
|
|||||||
return builder.build();
|
return builder.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
// Section: Create common buttons
|
// Section: Create common buttons
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create next button panel item.
|
* Create next button panel item.
|
||||||
*
|
*
|
||||||
@ -427,13 +362,11 @@ public class DetailsPanel
|
|||||||
* @param slot the slot
|
* @param slot the slot
|
||||||
* @return the panel item
|
* @return the panel item
|
||||||
*/
|
*/
|
||||||
private PanelItem createNextButton(ItemTemplateRecord template, TemplatedPanel.ItemSlot slot)
|
private PanelItem createNextButton(ItemTemplateRecord template, TemplatedPanel.ItemSlot slot) {
|
||||||
{
|
|
||||||
long size = this.materialCountList.size();
|
long size = this.materialCountList.size();
|
||||||
|
|
||||||
if (size <= slot.amountMap().getOrDefault("BLOCK", 1) ||
|
if (size <= slot.amountMap().getOrDefault("BLOCK", 1)
|
||||||
1.0 * size / slot.amountMap().getOrDefault("BLOCK", 1) <= this.pageIndex + 1)
|
|| 1.0 * size / slot.amountMap().getOrDefault("BLOCK", 1) <= this.pageIndex + 1) {
|
||||||
{
|
|
||||||
// There are no next elements
|
// There are no next elements
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@ -442,37 +375,30 @@ public class DetailsPanel
|
|||||||
|
|
||||||
PanelItemBuilder builder = new PanelItemBuilder();
|
PanelItemBuilder builder = new PanelItemBuilder();
|
||||||
|
|
||||||
if (template.icon() != null)
|
if (template.icon() != null) {
|
||||||
{
|
|
||||||
ItemStack clone = template.icon().clone();
|
ItemStack clone = template.icon().clone();
|
||||||
|
|
||||||
if (Boolean.TRUE.equals(template.dataMap().getOrDefault("indexing", false)))
|
if (Boolean.TRUE.equals(template.dataMap().getOrDefault("indexing", false))) {
|
||||||
{
|
|
||||||
clone.setAmount(nextPageIndex);
|
clone.setAmount(nextPageIndex);
|
||||||
}
|
}
|
||||||
|
|
||||||
builder.icon(clone);
|
builder.icon(clone);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (template.title() != null)
|
if (template.title() != null) {
|
||||||
{
|
|
||||||
builder.name(this.user.getTranslation(this.world, template.title()));
|
builder.name(this.user.getTranslation(this.world, template.title()));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (template.description() != null)
|
if (template.description() != null) {
|
||||||
{
|
builder.description(this.user.getTranslation(this.world, template.description(), TextVariables.NUMBER,
|
||||||
builder.description(this.user.getTranslation(this.world, template.description(),
|
String.valueOf(nextPageIndex)));
|
||||||
TextVariables.NUMBER, String.valueOf(nextPageIndex)));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add ClickHandler
|
// Add ClickHandler
|
||||||
builder.clickHandler((panel, user, clickType, i) ->
|
builder.clickHandler((panel, user, clickType, i) -> {
|
||||||
{
|
for (ItemTemplateRecord.ActionRecords action : template.actions()) {
|
||||||
for (ItemTemplateRecord.ActionRecords action : template.actions())
|
if ((clickType == action.clickType() || ClickType.UNKNOWN.equals(action.clickType()))
|
||||||
{
|
&& "NEXT".equalsIgnoreCase(action.actionType())) {
|
||||||
if ((clickType == action.clickType() || ClickType.UNKNOWN.equals(action.clickType())) &&
|
|
||||||
"NEXT".equalsIgnoreCase(action.actionType()))
|
|
||||||
{
|
|
||||||
this.pageIndex++;
|
this.pageIndex++;
|
||||||
this.build();
|
this.build();
|
||||||
}
|
}
|
||||||
@ -483,15 +409,12 @@ public class DetailsPanel
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Collect tooltips.
|
// Collect tooltips.
|
||||||
List<String> tooltips = template.actions().stream().
|
List<String> tooltips = template.actions().stream().filter(action -> action.tooltip() != null)
|
||||||
filter(action -> action.tooltip() != null).
|
.map(action -> this.user.getTranslation(this.world, action.tooltip())).filter(text -> !text.isBlank())
|
||||||
map(action -> this.user.getTranslation(this.world, action.tooltip())).
|
.collect(Collectors.toCollection(() -> new ArrayList<>(template.actions().size())));
|
||||||
filter(text -> !text.isBlank()).
|
|
||||||
collect(Collectors.toCollection(() -> new ArrayList<>(template.actions().size())));
|
|
||||||
|
|
||||||
// Add tooltips.
|
// Add tooltips.
|
||||||
if (!tooltips.isEmpty())
|
if (!tooltips.isEmpty()) {
|
||||||
{
|
|
||||||
// Empty line and tooltips.
|
// Empty line and tooltips.
|
||||||
builder.description("");
|
builder.description("");
|
||||||
builder.description(tooltips);
|
builder.description(tooltips);
|
||||||
@ -500,7 +423,6 @@ public class DetailsPanel
|
|||||||
return builder.build();
|
return builder.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create previous button panel item.
|
* Create previous button panel item.
|
||||||
*
|
*
|
||||||
@ -508,10 +430,8 @@ public class DetailsPanel
|
|||||||
* @param slot the slot
|
* @param slot the slot
|
||||||
* @return the panel item
|
* @return the panel item
|
||||||
*/
|
*/
|
||||||
private PanelItem createPreviousButton(ItemTemplateRecord template, TemplatedPanel.ItemSlot slot)
|
private PanelItem createPreviousButton(ItemTemplateRecord template, TemplatedPanel.ItemSlot slot) {
|
||||||
{
|
if (this.pageIndex == 0) {
|
||||||
if (this.pageIndex == 0)
|
|
||||||
{
|
|
||||||
// There are no next elements
|
// There are no next elements
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@ -520,37 +440,30 @@ public class DetailsPanel
|
|||||||
|
|
||||||
PanelItemBuilder builder = new PanelItemBuilder();
|
PanelItemBuilder builder = new PanelItemBuilder();
|
||||||
|
|
||||||
if (template.icon() != null)
|
if (template.icon() != null) {
|
||||||
{
|
|
||||||
ItemStack clone = template.icon().clone();
|
ItemStack clone = template.icon().clone();
|
||||||
|
|
||||||
if (Boolean.TRUE.equals(template.dataMap().getOrDefault("indexing", false)))
|
if (Boolean.TRUE.equals(template.dataMap().getOrDefault("indexing", false))) {
|
||||||
{
|
|
||||||
clone.setAmount(previousPageIndex);
|
clone.setAmount(previousPageIndex);
|
||||||
}
|
}
|
||||||
|
|
||||||
builder.icon(clone);
|
builder.icon(clone);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (template.title() != null)
|
if (template.title() != null) {
|
||||||
{
|
|
||||||
builder.name(this.user.getTranslation(this.world, template.title()));
|
builder.name(this.user.getTranslation(this.world, template.title()));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (template.description() != null)
|
if (template.description() != null) {
|
||||||
{
|
builder.description(this.user.getTranslation(this.world, template.description(), TextVariables.NUMBER,
|
||||||
builder.description(this.user.getTranslation(this.world, template.description(),
|
String.valueOf(previousPageIndex)));
|
||||||
TextVariables.NUMBER, String.valueOf(previousPageIndex)));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add ClickHandler
|
// Add ClickHandler
|
||||||
builder.clickHandler((panel, user, clickType, i) ->
|
builder.clickHandler((panel, user, clickType, i) -> {
|
||||||
{
|
for (ItemTemplateRecord.ActionRecords action : template.actions()) {
|
||||||
for (ItemTemplateRecord.ActionRecords action : template.actions())
|
|
||||||
{
|
|
||||||
if ((clickType == action.clickType() || ClickType.UNKNOWN.equals(action.clickType()))
|
if ((clickType == action.clickType() || ClickType.UNKNOWN.equals(action.clickType()))
|
||||||
&& "PREVIOUS".equalsIgnoreCase(action.actionType()))
|
&& "PREVIOUS".equalsIgnoreCase(action.actionType())) {
|
||||||
{
|
|
||||||
this.pageIndex--;
|
this.pageIndex--;
|
||||||
this.build();
|
this.build();
|
||||||
}
|
}
|
||||||
@ -561,15 +474,12 @@ public class DetailsPanel
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Collect tooltips.
|
// Collect tooltips.
|
||||||
List<String> tooltips = template.actions().stream().
|
List<String> tooltips = template.actions().stream().filter(action -> action.tooltip() != null)
|
||||||
filter(action -> action.tooltip() != null).
|
.map(action -> this.user.getTranslation(this.world, action.tooltip())).filter(text -> !text.isBlank())
|
||||||
map(action -> this.user.getTranslation(this.world, action.tooltip())).
|
.collect(Collectors.toCollection(() -> new ArrayList<>(template.actions().size())));
|
||||||
filter(text -> !text.isBlank()).
|
|
||||||
collect(Collectors.toCollection(() -> new ArrayList<>(template.actions().size())));
|
|
||||||
|
|
||||||
// Add tooltips.
|
// Add tooltips.
|
||||||
if (!tooltips.isEmpty())
|
if (!tooltips.isEmpty()) {
|
||||||
{
|
|
||||||
// Empty line and tooltips.
|
// Empty line and tooltips.
|
||||||
builder.description("");
|
builder.description("");
|
||||||
builder.description(tooltips);
|
builder.description(tooltips);
|
||||||
@ -578,12 +488,10 @@ public class DetailsPanel
|
|||||||
return builder.build();
|
return builder.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
// Section: Create Material Button
|
// Section: Create Material Button
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create material button panel item.
|
* Create material button panel item.
|
||||||
*
|
*
|
||||||
@ -591,18 +499,15 @@ public class DetailsPanel
|
|||||||
* @param slot the slot
|
* @param slot the slot
|
||||||
* @return the panel item
|
* @return the panel item
|
||||||
*/
|
*/
|
||||||
private PanelItem createMaterialButton(ItemTemplateRecord template, TemplatedPanel.ItemSlot slot)
|
private PanelItem createMaterialButton(ItemTemplateRecord template, TemplatedPanel.ItemSlot slot) {
|
||||||
{
|
if (this.materialCountList.isEmpty()) {
|
||||||
if (this.materialCountList.isEmpty())
|
|
||||||
{
|
|
||||||
// Does not contain any generators.
|
// Does not contain any generators.
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
int index = this.pageIndex * slot.amountMap().getOrDefault("BLOCK", 1) + slot.slot();
|
int index = this.pageIndex * slot.amountMap().getOrDefault("BLOCK", 1) + slot.slot();
|
||||||
|
|
||||||
if (index >= this.materialCountList.size())
|
if (index >= this.materialCountList.size()) {
|
||||||
{
|
|
||||||
// Out of index.
|
// Out of index.
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@ -610,7 +515,6 @@ public class DetailsPanel
|
|||||||
return this.createMaterialButton(template, this.materialCountList.get(index));
|
return this.createMaterialButton(template, this.materialCountList.get(index));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method creates button for material.
|
* This method creates button for material.
|
||||||
*
|
*
|
||||||
@ -618,102 +522,84 @@ public class DetailsPanel
|
|||||||
* @param materialCount materialCount which button must be created.
|
* @param materialCount materialCount which button must be created.
|
||||||
* @return PanelItem for generator tier.
|
* @return PanelItem for generator tier.
|
||||||
*/
|
*/
|
||||||
private PanelItem createMaterialButton(ItemTemplateRecord template,
|
private PanelItem createMaterialButton(ItemTemplateRecord template, Pair<Material, Integer> materialCount) {
|
||||||
Pair<Material, Integer> materialCount)
|
|
||||||
{
|
|
||||||
PanelItemBuilder builder = new PanelItemBuilder();
|
PanelItemBuilder builder = new PanelItemBuilder();
|
||||||
|
|
||||||
if (template.icon() != null)
|
if (template.icon() != null) {
|
||||||
{
|
|
||||||
builder.icon(template.icon().clone());
|
builder.icon(template.icon().clone());
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
builder.icon(PanelUtils.getMaterialItem(materialCount.getKey()));
|
builder.icon(PanelUtils.getMaterialItem(materialCount.getKey()));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (materialCount.getValue() < 64)
|
if (materialCount.getValue() < 64) {
|
||||||
{
|
|
||||||
builder.amount(materialCount.getValue());
|
builder.amount(materialCount.getValue());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (template.title() != null)
|
if (template.title() != null) {
|
||||||
{
|
builder.name(this.user.getTranslation(this.world, template.title(), TextVariables.NUMBER,
|
||||||
builder.name(this.user.getTranslation(this.world, template.title(),
|
String.valueOf(materialCount.getValue()), "[material]",
|
||||||
TextVariables.NUMBER, String.valueOf(materialCount.getValue()),
|
Utils.prettifyObject(materialCount.getKey(), this.user)));
|
||||||
"[material]", Utils.prettifyObject(materialCount.getKey(), this.user)));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
String description = Utils.prettifyDescription(materialCount.getKey(), this.user);
|
String description = Utils.prettifyDescription(materialCount.getKey(), this.user);
|
||||||
|
|
||||||
final String reference = "level.gui.buttons.material.";
|
final String reference = "level.gui.buttons.material.";
|
||||||
String blockId = this.user.getTranslationOrNothing(reference + "id",
|
String blockId = this.user.getTranslationOrNothing(reference + "id", "[id]", materialCount.getKey().name());
|
||||||
"[id]", materialCount.getKey().name());
|
|
||||||
|
|
||||||
int blockValue = this.addon.getBlockConfig().getBlockValues().getOrDefault(materialCount.getKey(), 0);
|
int blockValue = this.addon.getBlockConfig().getBlockValues().getOrDefault(materialCount.getKey(), 0);
|
||||||
String value = blockValue > 0 ? this.user.getTranslationOrNothing(reference + "value",
|
String value = blockValue > 0
|
||||||
TextVariables.NUMBER, String.valueOf(blockValue)) : "";
|
? this.user.getTranslationOrNothing(reference + "value", TextVariables.NUMBER,
|
||||||
|
String.valueOf(blockValue))
|
||||||
|
: "";
|
||||||
|
|
||||||
int blockLimit = this.addon.getBlockConfig().getBlockLimits().getOrDefault(materialCount.getKey(), 0);
|
int blockLimit = this.addon.getBlockConfig().getBlockLimits().getOrDefault(materialCount.getKey(), 0);
|
||||||
String limit = blockLimit > 0 ? this.user.getTranslationOrNothing(reference + "limit",
|
String limit = blockLimit > 0
|
||||||
TextVariables.NUMBER, String.valueOf(blockLimit)) : "";
|
? this.user.getTranslationOrNothing(reference + "limit", TextVariables.NUMBER,
|
||||||
|
String.valueOf(blockLimit))
|
||||||
|
: "";
|
||||||
|
|
||||||
String count = this.user.getTranslationOrNothing(reference + "count",
|
String count = this.user.getTranslationOrNothing(reference + "count", TextVariables.NUMBER,
|
||||||
TextVariables.NUMBER, String.valueOf(materialCount.getValue()));
|
String.valueOf(materialCount.getValue()));
|
||||||
|
|
||||||
long calculatedValue = (long) Math.min(blockLimit > 0 ? blockLimit : Integer.MAX_VALUE, materialCount.getValue()) * blockValue;
|
long calculatedValue = (long) Math.min(blockLimit > 0 ? blockLimit : Integer.MAX_VALUE,
|
||||||
|
materialCount.getValue()) * blockValue;
|
||||||
String valueText = calculatedValue > 0 ? this.user.getTranslationOrNothing(reference + "calculated",
|
String valueText = calculatedValue > 0 ? this.user.getTranslationOrNothing(reference + "calculated",
|
||||||
TextVariables.NUMBER, String.valueOf(calculatedValue)) : "";
|
TextVariables.NUMBER, String.valueOf(calculatedValue)) : "";
|
||||||
|
|
||||||
if (template.description() != null)
|
if (template.description() != null) {
|
||||||
{
|
builder.description(this.user
|
||||||
builder.description(this.user.getTranslation(this.world, template.description(),
|
.getTranslation(this.world, template.description(), "[description]", description, "[id]", blockId,
|
||||||
"[description]", description,
|
"[value]", value, "[calculated]", valueText, "[limit]", limit, "[count]", count)
|
||||||
"[id]", blockId,
|
.replaceAll("(?m)^[ \\t]*\\r?\\n", "").replaceAll("(?<!\\\\)\\|", "\n").replace("\\\\\\|", "|"));
|
||||||
"[value]", value,
|
|
||||||
"[calculated]", valueText,
|
|
||||||
"[limit]", limit,
|
|
||||||
"[count]", count).
|
|
||||||
replaceAll("(?m)^[ \\t]*\\r?\\n", "").
|
|
||||||
replaceAll("(?<!\\\\)\\|", "\n").
|
|
||||||
replace("\\\\\\|", "|"));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return builder.build();
|
return builder.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
// Section: Other Methods
|
// Section: Other Methods
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method is used to open UserPanel outside this class. It will be much easier to open panel with single method
|
* This method is used to open UserPanel outside this class. It will be much
|
||||||
* call then initializing new object.
|
* easier to open panel with single method call then initializing new object.
|
||||||
*
|
*
|
||||||
* @param addon Level object
|
* @param addon Level object
|
||||||
* @param world World where user is operating
|
* @param world World where user is operating
|
||||||
* @param user User who opens panel
|
* @param user User who opens panel
|
||||||
*/
|
*/
|
||||||
public static void openPanel(Level addon,
|
public static void openPanel(Level addon, World world, User user) {
|
||||||
World world,
|
|
||||||
User user)
|
|
||||||
{
|
|
||||||
new DetailsPanel(addon, world, user).build();
|
new DetailsPanel(addon, world, user).build();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
// Section: Enums
|
// Section: Enums
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This enum holds possible tabs for current gui.
|
* This enum holds possible tabs for current gui.
|
||||||
*/
|
*/
|
||||||
private enum Tab
|
private enum Tab {
|
||||||
{
|
|
||||||
/**
|
/**
|
||||||
* All block Tab
|
* All block Tab
|
||||||
*/
|
*/
|
||||||
@ -732,12 +618,10 @@ public class DetailsPanel
|
|||||||
SPAWNER
|
SPAWNER
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sorting order of blocks.
|
* Sorting order of blocks.
|
||||||
*/
|
*/
|
||||||
private enum Filter
|
private enum Filter {
|
||||||
{
|
|
||||||
/**
|
/**
|
||||||
* By name
|
* By name
|
||||||
*/
|
*/
|
||||||
@ -752,7 +636,6 @@ public class DetailsPanel
|
|||||||
COUNT
|
COUNT
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
// Section: Variables
|
// Section: Variables
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
|
@ -5,10 +5,11 @@
|
|||||||
|
|
||||||
package world.bentobox.level.panels;
|
package world.bentobox.level.panels;
|
||||||
|
|
||||||
|
|
||||||
import java.io.File;
|
import java.io.File;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Optional;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
@ -29,50 +30,44 @@ import world.bentobox.bentobox.managers.RanksManager;
|
|||||||
import world.bentobox.level.Level;
|
import world.bentobox.level.Level;
|
||||||
import world.bentobox.level.util.Utils;
|
import world.bentobox.level.util.Utils;
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This panel opens top likes panel
|
* This panel opens top likes panel
|
||||||
*/
|
*/
|
||||||
public class TopLevelPanel
|
public class TopLevelPanel {
|
||||||
{
|
|
||||||
|
|
||||||
|
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
// Section: Internal Constructor
|
// Section: Internal Constructor
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This is internal constructor. It is used internally in current class to avoid creating objects everywhere.
|
* This is internal constructor. It is used internally in current class to avoid
|
||||||
|
* creating objects everywhere.
|
||||||
*
|
*
|
||||||
* @param addon Level object.
|
* @param addon Level object.
|
||||||
* @param user User who opens Panel.
|
* @param user User who opens Panel.
|
||||||
* @param world World where gui is opened
|
* @param world World where gui is opened
|
||||||
* @param permissionPrefix Permission Prefix
|
* @param permissionPrefix Permission Prefix
|
||||||
*/
|
*/
|
||||||
private TopLevelPanel(Level addon, User user, World world, String permissionPrefix)
|
private TopLevelPanel(Level addon, User user, World world, String permissionPrefix) {
|
||||||
{
|
|
||||||
this.addon = addon;
|
this.addon = addon;
|
||||||
this.user = user;
|
this.user = user;
|
||||||
this.world = world;
|
this.world = world;
|
||||||
|
|
||||||
this.iconPermission = permissionPrefix + "level.icon";
|
this.iconPermission = permissionPrefix + "level.icon";
|
||||||
|
topIslands = new ArrayList<>();
|
||||||
this.topIslands = this.addon.getManager().getTopTen(this.world, 10).entrySet().stream().
|
for (Map.Entry<String, Long> en : addon.getManager().getTopTen(this.world, Level.TEN).entrySet()) {
|
||||||
map(entry -> {
|
Optional<Island> is = addon.getIslands().getIslandById(en.getKey());
|
||||||
Island island = this.addon.getIslandsManager().getIsland(this.world, entry.getKey());
|
if (is.isPresent()) {
|
||||||
return new IslandTopRecord(island, entry.getValue());
|
topIslands.add(new IslandTopRecord(is.get(), en.getValue()));
|
||||||
}).
|
}
|
||||||
collect(Collectors.toList());
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Build method manages current panel opening. It uses BentoBox PanelAPI that is easy to use and users can get nice
|
* Build method manages current panel opening. It uses BentoBox PanelAPI that is
|
||||||
* panels.
|
* easy to use and users can get nice panels.
|
||||||
*/
|
*/
|
||||||
public void build()
|
public void build() {
|
||||||
{
|
|
||||||
TemplatedPanelBuilder panelBuilder = new TemplatedPanelBuilder();
|
TemplatedPanelBuilder panelBuilder = new TemplatedPanelBuilder();
|
||||||
|
|
||||||
panelBuilder.user(this.user);
|
panelBuilder.user(this.user);
|
||||||
@ -87,47 +82,38 @@ public class TopLevelPanel
|
|||||||
panelBuilder.build();
|
panelBuilder.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
// Section: Methods
|
// Section: Methods
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates fallback based on template.
|
* Creates fallback based on template.
|
||||||
|
*
|
||||||
* @param template Template record for fallback button.
|
* @param template Template record for fallback button.
|
||||||
* @param index Place of the fallback.
|
* @param index Place of the fallback.
|
||||||
* @return Fallback panel item.
|
* @return Fallback panel item.
|
||||||
*/
|
*/
|
||||||
private PanelItem createFallback(ItemTemplateRecord template, long index)
|
private PanelItem createFallback(ItemTemplateRecord template, long index) {
|
||||||
{
|
if (template == null) {
|
||||||
if (template == null)
|
|
||||||
{
|
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
PanelItemBuilder builder = new PanelItemBuilder();
|
PanelItemBuilder builder = new PanelItemBuilder();
|
||||||
|
|
||||||
if (template.icon() != null)
|
if (template.icon() != null) {
|
||||||
{
|
|
||||||
builder.icon(template.icon().clone());
|
builder.icon(template.icon().clone());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (template.title() != null)
|
if (template.title() != null) {
|
||||||
{
|
builder.name(
|
||||||
builder.name(this.user.getTranslation(this.world, template.title(),
|
this.user.getTranslation(this.world, template.title(), TextVariables.NAME, String.valueOf(index)));
|
||||||
TextVariables.NAME, String.valueOf(index)));
|
} else {
|
||||||
}
|
builder.name(this.user.getTranslation(this.world, REFERENCE, TextVariables.NAME, String.valueOf(index)));
|
||||||
else
|
|
||||||
{
|
|
||||||
builder.name(this.user.getTranslation(this.world, REFERENCE,
|
|
||||||
TextVariables.NAME, String.valueOf(index)));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (template.description() != null)
|
if (template.description() != null) {
|
||||||
{
|
builder.description(this.user.getTranslation(this.world, template.description(), TextVariables.NUMBER,
|
||||||
builder.description(this.user.getTranslation(this.world, template.description(),
|
String.valueOf(index)));
|
||||||
TextVariables.NUMBER, String.valueOf(index)));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
builder.amount(index != 0 ? (int) index : 1);
|
builder.amount(index != 0 ? (int) index : 1);
|
||||||
@ -135,46 +121,40 @@ public class TopLevelPanel
|
|||||||
return builder.build();
|
return builder.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method creates player icon with warp functionality.
|
* This method creates player icon with warp functionality.
|
||||||
*
|
*
|
||||||
* @return PanelItem for PanelBuilder.
|
* @return PanelItem for PanelBuilder.
|
||||||
*/
|
*/
|
||||||
private PanelItem createPlayerButton(ItemTemplateRecord template, TemplatedPanel.ItemSlot itemSlot)
|
private PanelItem createPlayerButton(ItemTemplateRecord template, TemplatedPanel.ItemSlot itemSlot) {
|
||||||
{
|
|
||||||
int index = (int) template.dataMap().getOrDefault("index", 0);
|
int index = (int) template.dataMap().getOrDefault("index", 0);
|
||||||
|
|
||||||
if (index < 1)
|
if (index < 1) {
|
||||||
{
|
|
||||||
return this.createFallback(template.fallback(), index);
|
return this.createFallback(template.fallback(), index);
|
||||||
}
|
}
|
||||||
|
|
||||||
IslandTopRecord islandTopRecord = this.topIslands.size() < index ? null : this.topIslands.get(index - 1);
|
IslandTopRecord islandTopRecord = this.topIslands.size() < index ? null : this.topIslands.get(index - 1);
|
||||||
|
|
||||||
if (islandTopRecord == null)
|
if (islandTopRecord == null) {
|
||||||
{
|
|
||||||
return this.createFallback(template.fallback(), index);
|
return this.createFallback(template.fallback(), index);
|
||||||
}
|
}
|
||||||
|
|
||||||
return this.createIslandIcon(template, islandTopRecord, index);
|
return this.createIslandIcon(template, islandTopRecord, index);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method creates button from template for given island top record.
|
* This method creates button from template for given island top record.
|
||||||
|
*
|
||||||
* @param template Icon Template.
|
* @param template Icon Template.
|
||||||
* @param islandTopRecord Island Top Record.
|
* @param islandTopRecord Island Top Record.
|
||||||
* @param index Place Index.
|
* @param index Place Index.
|
||||||
* @return PanelItem for PanelBuilder.
|
* @return PanelItem for PanelBuilder.
|
||||||
*/
|
*/
|
||||||
private PanelItem createIslandIcon(ItemTemplateRecord template, IslandTopRecord islandTopRecord, int index)
|
private PanelItem createIslandIcon(ItemTemplateRecord template, IslandTopRecord islandTopRecord, int index) {
|
||||||
{
|
|
||||||
// Get player island.
|
// Get player island.
|
||||||
Island island = islandTopRecord.island();
|
Island island = islandTopRecord.island();
|
||||||
|
|
||||||
if (island == null)
|
if (island == null) {
|
||||||
{
|
|
||||||
return this.createFallback(template.fallback(), index);
|
return this.createFallback(template.fallback(), index);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -189,23 +169,19 @@ public class TopLevelPanel
|
|||||||
// Get only possible actions, by removing all inactive ones.
|
// Get only possible actions, by removing all inactive ones.
|
||||||
List<ItemTemplateRecord.ActionRecords> activeActions = new ArrayList<>(template.actions());
|
List<ItemTemplateRecord.ActionRecords> activeActions = new ArrayList<>(template.actions());
|
||||||
|
|
||||||
activeActions.removeIf(action ->
|
activeActions.removeIf(action -> {
|
||||||
{
|
switch (action.actionType().toUpperCase()) {
|
||||||
switch (action.actionType().toUpperCase())
|
|
||||||
{
|
|
||||||
case "WARP" -> {
|
case "WARP" -> {
|
||||||
return island.getOwner() == null ||
|
return island.getOwner() == null || this.addon.getWarpHook() == null
|
||||||
this.addon.getWarpHook() == null ||
|
|| !this.addon.getWarpHook().getWarpSignsManager().hasWarp(this.world, island.getOwner());
|
||||||
!this.addon.getWarpHook().getWarpSignsManager().hasWarp(this.world, island.getOwner());
|
|
||||||
}
|
}
|
||||||
case "VISIT" -> {
|
case "VISIT" -> {
|
||||||
return island.getOwner() == null ||
|
return island.getOwner() == null || this.addon.getVisitHook() == null
|
||||||
this.addon.getVisitHook() == null ||
|
|| !this.addon.getVisitHook().getAddonManager().preprocessTeleportation(this.user, island, true);
|
||||||
!this.addon.getVisitHook().getAddonManager().preprocessTeleportation(this.user, island);
|
|
||||||
}
|
}
|
||||||
case "VIEW" -> {
|
case "VIEW" -> {
|
||||||
return island.getOwner() == null ||
|
return island.getOwner() == null
|
||||||
!island.getMemberSet(RanksManager.MEMBER_RANK).contains(this.user.getUniqueId());
|
|| !island.getMemberSet(RanksManager.MEMBER_RANK).contains(this.user.getUniqueId());
|
||||||
}
|
}
|
||||||
default -> {
|
default -> {
|
||||||
return false;
|
return false;
|
||||||
@ -214,33 +190,30 @@ public class TopLevelPanel
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Add Click handler
|
// Add Click handler
|
||||||
builder.clickHandler((panel, user, clickType, i) ->
|
builder.clickHandler((panel, user, clickType, i) -> {
|
||||||
{
|
for (ItemTemplateRecord.ActionRecords action : activeActions) {
|
||||||
for (ItemTemplateRecord.ActionRecords action : activeActions)
|
if (clickType == action.clickType() || action.clickType() == ClickType.UNKNOWN) {
|
||||||
{
|
switch (action.actionType().toUpperCase()) {
|
||||||
if (clickType == action.clickType() || action.clickType() == ClickType.UNKNOWN)
|
|
||||||
{
|
|
||||||
switch (action.actionType().toUpperCase())
|
|
||||||
{
|
|
||||||
case "WARP" -> {
|
case "WARP" -> {
|
||||||
this.user.closeInventory();
|
this.user.closeInventory();
|
||||||
this.addon.getWarpHook().getWarpSignsManager().warpPlayer(this.world, this.user, island.getOwner());
|
this.addon.getWarpHook().getWarpSignsManager().warpPlayer(this.world, this.user,
|
||||||
|
island.getOwner());
|
||||||
}
|
}
|
||||||
case "VISIT" ->
|
case "VISIT" ->
|
||||||
// The command call implementation solves necessity to check for all visits options,
|
// The command call implementation solves necessity to check for all visits
|
||||||
// like cool down, confirmation and preprocess in single go. Would it be better to write
|
// options,
|
||||||
|
// like cool down, confirmation and preprocess in single go. Would it be better
|
||||||
|
// to write
|
||||||
// all logic here?
|
// all logic here?
|
||||||
|
|
||||||
this.addon.getPlugin().getIWM().getAddon(this.world).
|
this.addon.getPlugin().getIWM().getAddon(this.world).flatMap(GameModeAddon::getPlayerCommand)
|
||||||
flatMap(GameModeAddon::getPlayerCommand).ifPresent(command ->
|
.ifPresent(command -> {
|
||||||
{
|
String mainCommand = this.addon.getVisitHook().getSettings().getPlayerMainCommand();
|
||||||
String mainCommand =
|
|
||||||
this.addon.getVisitHook().getSettings().getPlayerMainCommand();
|
|
||||||
|
|
||||||
if (!mainCommand.isBlank())
|
if (!mainCommand.isBlank()) {
|
||||||
{
|
|
||||||
this.user.closeInventory();
|
this.user.closeInventory();
|
||||||
this.user.performCommand(command.getTopLabel() + " " + mainCommand + " " + island.getOwner());
|
this.user.performCommand(
|
||||||
|
command.getTopLabel() + " " + mainCommand + " " + island.getOwner());
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -262,15 +235,12 @@ public class TopLevelPanel
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Collect tooltips.
|
// Collect tooltips.
|
||||||
List<String> tooltips = activeActions.stream().
|
List<String> tooltips = activeActions.stream().filter(action -> action.tooltip() != null)
|
||||||
filter(action -> action.tooltip() != null).
|
.map(action -> this.user.getTranslation(this.world, action.tooltip())).filter(text -> !text.isBlank())
|
||||||
map(action -> this.user.getTranslation(this.world, action.tooltip())).
|
.collect(Collectors.toCollection(() -> new ArrayList<>(template.actions().size())));
|
||||||
filter(text -> !text.isBlank()).
|
|
||||||
collect(Collectors.toCollection(() -> new ArrayList<>(template.actions().size())));
|
|
||||||
|
|
||||||
// Add tooltips.
|
// Add tooltips.
|
||||||
if (!tooltips.isEmpty())
|
if (!tooltips.isEmpty()) {
|
||||||
{
|
|
||||||
// Empty line and tooltips.
|
// Empty line and tooltips.
|
||||||
builder.description("");
|
builder.description("");
|
||||||
builder.description(tooltips);
|
builder.description(tooltips);
|
||||||
@ -279,104 +249,75 @@ public class TopLevelPanel
|
|||||||
return builder.build();
|
return builder.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Populate given panel item builder name with values from template and island objects.
|
* Populate given panel item builder name with values from template and island
|
||||||
|
* objects.
|
||||||
*
|
*
|
||||||
* @param builder the builder
|
* @param builder the builder
|
||||||
* @param template the template
|
* @param template the template
|
||||||
* @param island the island
|
* @param island the island
|
||||||
*/
|
*/
|
||||||
private void populateIslandTitle(PanelItemBuilder builder,
|
private void populateIslandTitle(PanelItemBuilder builder, ItemTemplateRecord template, Island island) {
|
||||||
ItemTemplateRecord template,
|
|
||||||
Island island)
|
|
||||||
{
|
|
||||||
|
|
||||||
// Get Island Name
|
// Get Island Name
|
||||||
String nameText;
|
String nameText;
|
||||||
|
|
||||||
if (island.getName() == null || island.getName().isEmpty())
|
if (island.getName() == null || island.getName().isEmpty()) {
|
||||||
{
|
nameText = this.user.getTranslation(REFERENCE + "owners-island", PLAYER,
|
||||||
nameText = this.user.getTranslation(REFERENCE + "owners-island",
|
island.getOwner() == null ? this.user.getTranslation(REFERENCE + "unknown")
|
||||||
PLAYER,
|
: this.addon.getPlayers().getName(island.getOwner()));
|
||||||
island.getOwner() == null ?
|
} else {
|
||||||
this.user.getTranslation(REFERENCE + "unknown") :
|
|
||||||
this.addon.getPlayers().getName(island.getOwner()));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
nameText = island.getName();
|
nameText = island.getName();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Template specific title is always more important than custom one.
|
// Template specific title is always more important than custom one.
|
||||||
if (template.title() != null && !template.title().isBlank())
|
if (template.title() != null && !template.title().isBlank()) {
|
||||||
{
|
builder.name(this.user.getTranslation(this.world, template.title(), TextVariables.NAME, nameText));
|
||||||
builder.name(this.user.getTranslation(this.world, template.title(),
|
} else {
|
||||||
TextVariables.NAME, nameText));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
builder.name(this.user.getTranslation(REFERENCE + "name", TextVariables.NAME, nameText));
|
builder.name(this.user.getTranslation(REFERENCE + "name", TextVariables.NAME, nameText));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Populate given panel item builder icon with values from template and island objects.
|
* Populate given panel item builder icon with values from template and island
|
||||||
|
* objects.
|
||||||
*
|
*
|
||||||
* @param builder the builder
|
* @param builder the builder
|
||||||
* @param template the template
|
* @param template the template
|
||||||
* @param island the island
|
* @param island the island
|
||||||
*/
|
*/
|
||||||
private void populateIslandIcon(PanelItemBuilder builder,
|
private void populateIslandIcon(PanelItemBuilder builder, ItemTemplateRecord template, Island island) {
|
||||||
ItemTemplateRecord template,
|
|
||||||
Island island)
|
|
||||||
{
|
|
||||||
User owner = island.getOwner() == null ? null : User.getInstance(island.getOwner());
|
User owner = island.getOwner() == null ? null : User.getInstance(island.getOwner());
|
||||||
|
|
||||||
// Get permission or island icon
|
// Get permission or island icon
|
||||||
String permissionIcon = owner == null ? null :
|
String permissionIcon = owner == null ? null : Utils.getPermissionValue(owner, this.iconPermission, null);
|
||||||
Utils.getPermissionValue(owner, this.iconPermission, null);
|
|
||||||
|
|
||||||
Material material;
|
Material material;
|
||||||
|
|
||||||
if (permissionIcon != null && !permissionIcon.equals("*"))
|
if (permissionIcon != null && !permissionIcon.equals("*")) {
|
||||||
{
|
|
||||||
material = Material.matchMaterial(permissionIcon);
|
material = Material.matchMaterial(permissionIcon);
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
material = null;
|
material = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (material != null)
|
if (material != null) {
|
||||||
{
|
if (!material.equals(Material.PLAYER_HEAD)) {
|
||||||
if (!material.equals(Material.PLAYER_HEAD))
|
|
||||||
{
|
|
||||||
builder.icon(material);
|
builder.icon(material);
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
builder.icon(owner.getName());
|
builder.icon(owner.getName());
|
||||||
}
|
}
|
||||||
}
|
} else if (template.icon() != null) {
|
||||||
else if (template.icon() != null)
|
|
||||||
{
|
|
||||||
builder.icon(template.icon().clone());
|
builder.icon(template.icon().clone());
|
||||||
}
|
} else if (owner != null) {
|
||||||
else if (owner != null)
|
|
||||||
{
|
|
||||||
builder.icon(owner.getName());
|
builder.icon(owner.getName());
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
builder.icon(Material.PLAYER_HEAD);
|
builder.icon(Material.PLAYER_HEAD);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Populate given panel item builder description with values from template and island objects.
|
* Populate given panel item builder description with values from template and
|
||||||
|
* island objects.
|
||||||
*
|
*
|
||||||
* @param builder the builder
|
* @param builder the builder
|
||||||
* @param template the template
|
* @param template the template
|
||||||
@ -384,94 +325,72 @@ public class TopLevelPanel
|
|||||||
* @param islandTopRecord the top record object
|
* @param islandTopRecord the top record object
|
||||||
* @param index place index.
|
* @param index place index.
|
||||||
*/
|
*/
|
||||||
private void populateIslandDescription(PanelItemBuilder builder,
|
private void populateIslandDescription(PanelItemBuilder builder, ItemTemplateRecord template, Island island,
|
||||||
ItemTemplateRecord template,
|
IslandTopRecord islandTopRecord, int index) {
|
||||||
Island island,
|
|
||||||
IslandTopRecord islandTopRecord,
|
|
||||||
int index)
|
|
||||||
{
|
|
||||||
// Get Owner Name
|
// Get Owner Name
|
||||||
String ownerText = this.user.getTranslation(REFERENCE + "owner",
|
String ownerText = this.user.getTranslation(REFERENCE + "owner", PLAYER,
|
||||||
PLAYER,
|
island.getOwner() == null ? this.user.getTranslation(REFERENCE + "unknown")
|
||||||
island.getOwner() == null ?
|
: this.addon.getPlayers().getName(island.getOwner()));
|
||||||
this.user.getTranslation(REFERENCE + "unknown") :
|
|
||||||
this.addon.getPlayers().getName(island.getOwner()));
|
|
||||||
|
|
||||||
// Get Members Text
|
// Get Members Text
|
||||||
String memberText;
|
String memberText;
|
||||||
|
|
||||||
if (island.getMemberSet().size() > 1)
|
if (island.getMemberSet().size() > 1) {
|
||||||
{
|
|
||||||
StringBuilder memberBuilder = new StringBuilder(
|
StringBuilder memberBuilder = new StringBuilder(
|
||||||
this.user.getTranslationOrNothing(REFERENCE + "members-title"));
|
this.user.getTranslationOrNothing(REFERENCE + "members-title"));
|
||||||
|
|
||||||
for (UUID uuid : island.getMemberSet())
|
for (UUID uuid : island.getMemberSet()) {
|
||||||
{
|
|
||||||
User member = User.getInstance(uuid);
|
User member = User.getInstance(uuid);
|
||||||
|
|
||||||
if (memberBuilder.length() > 0)
|
if (memberBuilder.length() > 0) {
|
||||||
{
|
|
||||||
memberBuilder.append("\n");
|
memberBuilder.append("\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
memberBuilder.append(
|
memberBuilder.append(this.user.getTranslationOrNothing(REFERENCE + "member", PLAYER, member.getName()));
|
||||||
this.user.getTranslationOrNothing(REFERENCE + "member",
|
|
||||||
PLAYER, member.getName()));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
memberText = memberBuilder.toString();
|
memberText = memberBuilder.toString();
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
memberText = "";
|
memberText = "";
|
||||||
}
|
}
|
||||||
|
|
||||||
String placeText = this.user.getTranslation(REFERENCE + "place",
|
String placeText = this.user.getTranslation(REFERENCE + "place", TextVariables.NUMBER, String.valueOf(index));
|
||||||
TextVariables.NUMBER, String.valueOf(index));
|
|
||||||
|
|
||||||
String levelText = this.user.getTranslation(REFERENCE + "level",
|
String levelText = this.user.getTranslation(REFERENCE + "level", TextVariables.NUMBER,
|
||||||
TextVariables.NUMBER, this.addon.getManager().formatLevel(islandTopRecord.level()));
|
this.addon.getManager().formatLevel(islandTopRecord.level()));
|
||||||
|
|
||||||
// Template specific description is always more important than custom one.
|
// Template specific description is always more important than custom one.
|
||||||
if (template.description() != null && !template.description().isBlank())
|
if (template.description() != null && !template.description().isBlank()) {
|
||||||
{
|
builder.description(this.user
|
||||||
builder.description(this.user.getTranslation(this.world, template.description(),
|
.getTranslation(this.world, template.description(), "[owner]", ownerText, "[members]", memberText,
|
||||||
"[owner]", ownerText,
|
"[level]", levelText, "[place]", placeText)
|
||||||
"[members]", memberText,
|
.replaceAll("(?m)^[ \\t]*\\r?\\n", "").replaceAll("(?<!\\\\)\\|", "\n").replace("\\\\\\|", "|")); // Not
|
||||||
"[level]", levelText,
|
// a
|
||||||
"[place]", placeText).
|
// regex
|
||||||
replaceAll("(?m)^[ \\t]*\\r?\\n", "").
|
// -
|
||||||
replaceAll("(?<!\\\\)\\|", "\n").
|
// replace
|
||||||
replace("\\\\\\|", "|")); // Not a regex - replace is more efficient
|
// is
|
||||||
}
|
// more
|
||||||
else
|
// efficient
|
||||||
{
|
} else {
|
||||||
// Now combine everything.
|
// Now combine everything.
|
||||||
String descriptionText = this.user.getTranslation(REFERENCE + "description",
|
String descriptionText = this.user.getTranslation(REFERENCE + "description", "[owner]", ownerText,
|
||||||
"[owner]", ownerText,
|
"[members]", memberText, "[level]", levelText, "[place]", placeText);
|
||||||
"[members]", memberText,
|
|
||||||
"[level]", levelText,
|
|
||||||
"[place]", placeText);
|
|
||||||
|
|
||||||
builder.description(descriptionText.
|
builder.description(descriptionText.replaceAll("(?m)^[ \\t]*\\r?\\n", "").replaceAll("(?<!\\\\)\\|", "\n")
|
||||||
replaceAll("(?m)^[ \\t]*\\r?\\n", "").
|
.replace("\\\\\\|", "|")); // Not a regex - replace is more efficient
|
||||||
replaceAll("(?<!\\\\)\\|", "\n").
|
|
||||||
replace("\\\\\\|", "|")); // Not a regex - replace is more efficient
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create viewer button panel item.
|
* Create viewer button panel item.
|
||||||
*
|
*
|
||||||
* @return PanelItem for PanelBuilder.
|
* @return PanelItem for PanelBuilder.
|
||||||
*/
|
*/
|
||||||
private PanelItem createViewerButton(ItemTemplateRecord template, TemplatedPanel.ItemSlot itemSlot)
|
private PanelItem createViewerButton(ItemTemplateRecord template, TemplatedPanel.ItemSlot itemSlot) {
|
||||||
{
|
|
||||||
Island island = this.addon.getIslands().getIsland(this.world, this.user);
|
Island island = this.addon.getIslands().getIsland(this.world, this.user);
|
||||||
|
|
||||||
if (island == null || island.getOwner() == null)
|
if (island == null || island.getOwner() == null) {
|
||||||
{
|
|
||||||
// Player do not have an island.
|
// Player do not have an island.
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@ -484,18 +403,16 @@ public class TopLevelPanel
|
|||||||
return this.createIslandIcon(template, topRecord, place);
|
return this.createIslandIcon(template, topRecord, place);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This method is used to open UserPanel outside this class. It will be much easier to open panel with single method
|
* This method is used to open UserPanel outside this class. It will be much
|
||||||
* call then initializing new object.
|
* easier to open panel with single method call then initializing new object.
|
||||||
*
|
*
|
||||||
* @param addon Level Addon object
|
* @param addon Level Addon object
|
||||||
* @param user User who opens panel
|
* @param user User who opens panel
|
||||||
* @param world World where gui is opened
|
* @param world World where gui is opened
|
||||||
* @param permissionPrefix Permission Prefix
|
* @param permissionPrefix Permission Prefix
|
||||||
*/
|
*/
|
||||||
public static void openPanel(Level addon, User user, World world, String permissionPrefix)
|
public static void openPanel(Level addon, User user, World world, String permissionPrefix) {
|
||||||
{
|
|
||||||
new TopLevelPanel(addon, user, world, permissionPrefix).build();
|
new TopLevelPanel(addon, user, world, permissionPrefix).build();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -512,9 +429,12 @@ public class TopLevelPanel
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* This record is used internally. It converts user -> level to island -> level.
|
* This record is used internally. It converts user -> level to island -> level.
|
||||||
|
*
|
||||||
|
* @param island island
|
||||||
|
* @param level level
|
||||||
*/
|
*/
|
||||||
private record IslandTopRecord(Island island, Long level) {}
|
private record IslandTopRecord(Island island, Long level) {
|
||||||
|
}
|
||||||
|
|
||||||
// ---------------------------------------------------------------------
|
// ---------------------------------------------------------------------
|
||||||
// Section: Variables
|
// Section: Variables
|
||||||
|
@ -287,7 +287,7 @@ blocks:
|
|||||||
GRANITE_SLAB: 1
|
GRANITE_SLAB: 1
|
||||||
GRANITE_STAIRS: 1
|
GRANITE_STAIRS: 1
|
||||||
GRANITE_WALL: 1
|
GRANITE_WALL: 1
|
||||||
GRASS: 4
|
SHORT_GRASS: 4
|
||||||
GRASS_BLOCK: 4
|
GRASS_BLOCK: 4
|
||||||
GRAVEL: 1
|
GRAVEL: 1
|
||||||
GRAY_BANNER: 2
|
GRAY_BANNER: 2
|
||||||
|
@ -22,8 +22,18 @@ admin:
|
|||||||
remove:
|
remove:
|
||||||
description: "remove player from Top Ten"
|
description: "remove player from Top Ten"
|
||||||
parameters: "<player>"
|
parameters: "<player>"
|
||||||
|
stats:
|
||||||
|
description: "show stats on islands on this server"
|
||||||
|
title: "Server Island Stats"
|
||||||
|
world: "&a [name]"
|
||||||
|
no-data: "&c No data to process."
|
||||||
|
average-level: "Average Island Level: [number]"
|
||||||
|
median-level: "Median Island Level: [number]"
|
||||||
|
mode-level: "Mode Island Level: [number]"
|
||||||
|
highest-level: "Highest Island Level: [number]"
|
||||||
|
lowest-level: "Lowest Island Level: [number]"
|
||||||
|
distribution: "Island Level Distribution:"
|
||||||
|
islands: "islands"
|
||||||
island:
|
island:
|
||||||
level:
|
level:
|
||||||
parameters: "[player]"
|
parameters: "[player]"
|
||||||
|
184
src/main/resources/locales/uk.yml
Normal file
184
src/main/resources/locales/uk.yml
Normal file
@ -0,0 +1,184 @@
|
|||||||
|
---
|
||||||
|
admin:
|
||||||
|
level:
|
||||||
|
parameters: "<player>"
|
||||||
|
description: розрахувати рівень острова для гравця
|
||||||
|
sethandicap:
|
||||||
|
parameters: "<player> <handicap>"
|
||||||
|
description: встановити гандикап острова, як правило, рівень острова стартера
|
||||||
|
changed: "&a Початковий гандикап острова змінено з [number] на [new_number]."
|
||||||
|
invalid-level: "&c Недійсний гандикап. Використовуйте ціле число."
|
||||||
|
levelstatus:
|
||||||
|
description: показати, скільки островів у черзі на сканування
|
||||||
|
islands-in-queue: "&a Острови в черзі: [number]"
|
||||||
|
top:
|
||||||
|
description: показати першу десятку списку
|
||||||
|
unknown-world: "&c Невідомий світ!"
|
||||||
|
display: "&f[rank]. &a[name] &7- &b[level]"
|
||||||
|
remove:
|
||||||
|
description: видалити гравця з першої десятки
|
||||||
|
parameters: "<player>"
|
||||||
|
stats:
|
||||||
|
description: показати статистику островів на цьому сервері
|
||||||
|
title: Статистика острова сервера
|
||||||
|
world: "&a [name]"
|
||||||
|
no-data: "&c Немає даних для обробки."
|
||||||
|
average-level: 'Середній рівень острова: [number]'
|
||||||
|
median-level: 'Середній рівень острова: [number]'
|
||||||
|
mode-level: 'Рівень острова режиму: [number]'
|
||||||
|
highest-level: 'Найвищий рівень острова: [number]'
|
||||||
|
lowest-level: 'Найнижчий рівень острова: [number]'
|
||||||
|
distribution: 'Розподіл на рівні острова:'
|
||||||
|
islands: острови
|
||||||
|
island:
|
||||||
|
level:
|
||||||
|
parameters: "[player]"
|
||||||
|
description: обчисліть свій рівень острова або покажіть рівень [player]
|
||||||
|
calculating: "&a Розрахунок рівня..."
|
||||||
|
estimated-wait: "&a Приблизне очікування: [number] секунд"
|
||||||
|
in-queue: "&a Ви номер [number] у черзі"
|
||||||
|
island-level-is: "&a Рівень острова &b[level]"
|
||||||
|
required-points-to-next-level: "&a [points] потрібні бали до наступного рівня"
|
||||||
|
deaths: "&c([number] смерті)"
|
||||||
|
cooldown: "&c Ви повинні зачекати &b[time] &c секунд, поки ви зможете зробити
|
||||||
|
це знову"
|
||||||
|
in-progress: "&6 Розрахунок рівня острова триває..."
|
||||||
|
time-out: "&c Розрахунок рівня тривав занадто довго. Будь-ласка спробуйте пізніше."
|
||||||
|
top:
|
||||||
|
description: показати першу десятку
|
||||||
|
gui-title: "& Десятка Кращих"
|
||||||
|
gui-heading: "&6[name]: &B[rank]"
|
||||||
|
island-level: "&b Рівень [level]"
|
||||||
|
warp-to: "&A Варп на острів [name]."
|
||||||
|
level-details:
|
||||||
|
above-sea-level-blocks: Блоки над рівнем моря
|
||||||
|
spawners: Спавера
|
||||||
|
underwater-blocks: Підводні блоки
|
||||||
|
all-blocks: Всі блоки
|
||||||
|
no-island: "&c Немає острова!"
|
||||||
|
names-island: острів [name].
|
||||||
|
syntax: "[name] x [number]"
|
||||||
|
hint: "&c Запустіть рівень, щоб переглянути звіт про блокування"
|
||||||
|
level:
|
||||||
|
commands:
|
||||||
|
value:
|
||||||
|
parameters: "[hand|<material>]"
|
||||||
|
description: показує значення блоків. Додайте 'hand' в кінці, щоб відобразити
|
||||||
|
значення предмета в руках.
|
||||||
|
gui:
|
||||||
|
titles:
|
||||||
|
top: "&0&l Топ островів"
|
||||||
|
detail-panel: "&0&l острів [name]."
|
||||||
|
value-panel: "&0&l Значення блоку"
|
||||||
|
buttons:
|
||||||
|
island:
|
||||||
|
empty: "&f&l [name]. місце"
|
||||||
|
name: "&f&l [name]"
|
||||||
|
description: |-
|
||||||
|
[owner]
|
||||||
|
[members]
|
||||||
|
[place]
|
||||||
|
[level]
|
||||||
|
owners-island: Острів [player].
|
||||||
|
owner: "&7&l Власник: &r&b [player]"
|
||||||
|
members-title: "&7&l Члени:"
|
||||||
|
member: "&b - [player]"
|
||||||
|
unknown: невідомий
|
||||||
|
place: "&7&o [number]. &r&7 місце"
|
||||||
|
level: "&7 Рівень: &o [number]"
|
||||||
|
material:
|
||||||
|
name: "&f&l [number] x [material]"
|
||||||
|
description: |-
|
||||||
|
[description]
|
||||||
|
[count]
|
||||||
|
[value]
|
||||||
|
[calculated]
|
||||||
|
[limit]
|
||||||
|
[id]
|
||||||
|
id: "&7 Ідентифікатор блоку: &e [id]"
|
||||||
|
value: "&7 Значення блоку: &e [number]"
|
||||||
|
limit: "&7 Обмеження блоку: &e [number]"
|
||||||
|
count: "&7 Кількість блоків: &e [number]"
|
||||||
|
calculated: "&7 Розраховане значення: &e [number]"
|
||||||
|
all_blocks:
|
||||||
|
name: "&f&l Усі блоки"
|
||||||
|
description: |-
|
||||||
|
&7 Показати всі блоки
|
||||||
|
&7 на острові.
|
||||||
|
above_sea_level:
|
||||||
|
name: "&f&l Блоки над рівнем моря"
|
||||||
|
description: |-
|
||||||
|
&7 Показувати лише блоки
|
||||||
|
&7, які знаходяться над морем
|
||||||
|
&7 рівень.
|
||||||
|
underwater:
|
||||||
|
name: "&f&l Блоки під рівнем моря"
|
||||||
|
description: |-
|
||||||
|
&7 Показувати лише блоки
|
||||||
|
&7, які знаходяться нижче моря
|
||||||
|
&7 рівень.
|
||||||
|
spawner:
|
||||||
|
name: "&f&l Спанера"
|
||||||
|
description: "&7 Відображати лише спавнери."
|
||||||
|
filters:
|
||||||
|
name:
|
||||||
|
name: "&f&l Сортувати за назвою"
|
||||||
|
description: "&7 Сортувати всі блоки за назвою."
|
||||||
|
value:
|
||||||
|
name: "&f&l Сортувати за значенням"
|
||||||
|
description: "&7 Сортувати всі блоки за їх значенням."
|
||||||
|
count:
|
||||||
|
name: "&f&l Сортувати за кількістю"
|
||||||
|
description: "&7 Відсортуйте всі блоки за їх кількістю."
|
||||||
|
value:
|
||||||
|
name: "&f&l [material]"
|
||||||
|
description: |-
|
||||||
|
[description]
|
||||||
|
[value]
|
||||||
|
[underwater]
|
||||||
|
[limit]
|
||||||
|
[id]
|
||||||
|
id: "&7 Ідентифікатор блоку: &e [id]"
|
||||||
|
value: "&7 Значення блоку: &e [number]"
|
||||||
|
underwater: "&7 Нижче рівня моря: &e [number]"
|
||||||
|
limit: "&7 Обмеження блоку: &e [number]"
|
||||||
|
previous:
|
||||||
|
name: "&f&l Попередня сторінка"
|
||||||
|
description: "&7 Перейти на сторінку [number]."
|
||||||
|
next:
|
||||||
|
name: "&f&l Наступна сторінка"
|
||||||
|
description: "&7 Перейти на сторінку [number]."
|
||||||
|
search:
|
||||||
|
name: "&f&l Пошук"
|
||||||
|
description: |-
|
||||||
|
&7 Пошук конкретного
|
||||||
|
&7 значення.
|
||||||
|
search: "&b Значення: [value]"
|
||||||
|
tips:
|
||||||
|
click-to-view: "&e Натисніть &7, щоб переглянути."
|
||||||
|
click-to-previous: "&e Натисніть &7, щоб переглянути попередню сторінку."
|
||||||
|
click-to-next: "&e Натисніть &7, щоб переглянути наступну сторінку."
|
||||||
|
click-to-select: "&e Натисніть &7, щоб вибрати."
|
||||||
|
left-click-to-cycle-up: "&e Клацніть лівою кнопкою миші &7, щоб перейти вгору."
|
||||||
|
right-click-to-cycle-down: "&e Клацніть правою кнопкою миші &7, щоб перейти
|
||||||
|
вниз."
|
||||||
|
left-click-to-change: "&e Клацніть лівою кнопкою миші &7 для редагування."
|
||||||
|
right-click-to-clear: "&e Клацніть правою кнопкою миші &7, щоб очистити."
|
||||||
|
click-to-asc: "&e Клацніть &7, щоб відсортувати в порядку збільшення."
|
||||||
|
click-to-desc: "&e Клацніть &7, щоб відсортувати в порядку зменшення."
|
||||||
|
click-to-warp: "&e Натисніть &7, щоб деформувати."
|
||||||
|
click-to-visit: "&e Натисніть &7, щоб відвідати."
|
||||||
|
right-click-to-visit: "&e Клацніть правою кнопкою миші &7, щоб відвідати."
|
||||||
|
conversations:
|
||||||
|
prefix: "&l&6 [BentoBox]: &r"
|
||||||
|
no-data: "&c Запустіть рівень, щоб переглянути звіт про блокування."
|
||||||
|
cancel-string: cancel
|
||||||
|
exit-string: cancel, exit, quit
|
||||||
|
write-search: "&e Введіть пошукове значення. (Напишіть 'cancel', щоб вийти)"
|
||||||
|
search-updated: "&a Значення пошуку оновлено."
|
||||||
|
cancelled: "&c Розмова скасована!"
|
||||||
|
no-value: "&c Цей предмет не має цінності."
|
||||||
|
unknown-item: "&c '[material]' не існує в грі."
|
||||||
|
value: "&7 Значення '[material]' таке: &e[value]"
|
||||||
|
value-underwater: "&7 Значення '[material]' нижче рівня моря: &e[value]"
|
||||||
|
empty-hand: "&c У вашій руці немає блоків"
|
@ -1,25 +1,49 @@
|
|||||||
|
# Name of panel used for indentification in the code
|
||||||
detail_panel:
|
detail_panel:
|
||||||
|
# Title of the panel shown to the user. This is a reference and the reference will be translatable in the locale file
|
||||||
title: level.gui.titles.detail-panel
|
title: level.gui.titles.detail-panel
|
||||||
|
# The type of panel to show. Options are INVENTORY, HOPPER, DROPPER. INVENTORY is that standard chest inventory and
|
||||||
|
# the others refer to the inventories shown for those items.
|
||||||
type: INVENTORY
|
type: INVENTORY
|
||||||
|
# The background of the panel. These items will be shown if other items are not there. STAINED_GLASS_PANEs give a good effect.
|
||||||
background:
|
background:
|
||||||
icon: BLACK_STAINED_GLASS_PANE
|
icon: BLACK_STAINED_GLASS_PANE
|
||||||
title: "&b&r" # Empty text
|
# Each item may have text applied to it, but usually for background items, nothing is shown.
|
||||||
|
title: "&b&r" # Empty text. This is using the Bukkit chat color coding with &'s.
|
||||||
border:
|
border:
|
||||||
|
# The border of each panel may be shown as a different item.
|
||||||
|
# It can be used to provide a contrast to items in the panel.
|
||||||
icon: BLACK_STAINED_GLASS_PANE
|
icon: BLACK_STAINED_GLASS_PANE
|
||||||
title: "&b&r" # Empty text
|
title: "&b&r" # Empty text
|
||||||
|
# This tag indicates which rows in the panel must be shown. The panel will be sized vertically accordingly. This does not include the borders.
|
||||||
|
# This can be a list and rows must be between 1 and 6, if used.
|
||||||
force-shown: []
|
force-shown: []
|
||||||
|
# The content section contains details of each item/button in the panel. The numbers indicate the rows and then then columns of each item.
|
||||||
content:
|
content:
|
||||||
|
# Row number
|
||||||
1:
|
1:
|
||||||
|
# Column number
|
||||||
2:
|
2:
|
||||||
|
# Icon is a Bukkit Material.
|
||||||
icon: STONE
|
icon: STONE
|
||||||
|
# Title of the button shown to the user. This is a reference and the reference will be translatable in the locale file
|
||||||
title: level.gui.buttons.all_blocks.name
|
title: level.gui.buttons.all_blocks.name
|
||||||
|
# Description of the button shown to the user in the lore. This is a reference and the reference will be translatable in the locale file
|
||||||
description: level.gui.buttons.all_blocks.description
|
description: level.gui.buttons.all_blocks.description
|
||||||
|
# The data section is a key-value list of data relavent for this button. It is interpreted by the code implemented the panel.
|
||||||
|
# The convention is to specify the type and the panel tab that will open if pressed. These are Enums in the code.
|
||||||
data:
|
data:
|
||||||
|
# Type button will go to the ALL_BLOCKS tab when clicked.
|
||||||
type: TAB
|
type: TAB
|
||||||
tab: ALL_BLOCKS
|
tab: ALL_BLOCKS
|
||||||
|
# Actions cover what happens if the button is clicked or the mouse is moved over it. There can be multiple actions possible for different
|
||||||
|
# click-types.
|
||||||
actions:
|
actions:
|
||||||
|
# Each action has an arbitrary descriptive name to define it.
|
||||||
view:
|
view:
|
||||||
|
# The click-type is the same as the bukkit {@link org.bukkit.event.inventory.ClickType}. UNKNOWN is the default.
|
||||||
click-type: unknown
|
click-type: unknown
|
||||||
|
# tooltip is a locale reference that will be translated for the user and shown when they hover over the button.
|
||||||
tooltip: level.gui.tips.click-to-view
|
tooltip: level.gui.tips.click-to-view
|
||||||
3:
|
3:
|
||||||
icon: GRASS_BLOCK
|
icon: GRASS_BLOCK
|
||||||
@ -57,12 +81,12 @@ detail_panel:
|
|||||||
9:
|
9:
|
||||||
# You can create multiple buttons. By default it is one.
|
# You can create multiple buttons. By default it is one.
|
||||||
icon: IRON_TRAPDOOR
|
icon: IRON_TRAPDOOR
|
||||||
# [filter] is placeholder for different filter types. It will be replaced with name, value, count.
|
# [filter] is a placeholder for different filter types. It will be replaced with name, value, count.
|
||||||
title: level.gui.buttons.filters.[filter].name
|
title: level.gui.buttons.filters.[filter].name
|
||||||
description: level.gui.buttons.filters.[filter].description
|
description: level.gui.buttons.filters.[filter].description
|
||||||
data:
|
data:
|
||||||
type: FILTER
|
type: FILTER
|
||||||
# the value of filter button. Suggestion is to leave fist value to name if you use single button.
|
# the value of filter button. Suggestion is to leave first value to name if you use single button.
|
||||||
filter: NAME
|
filter: NAME
|
||||||
actions:
|
actions:
|
||||||
up:
|
up:
|
||||||
@ -76,6 +100,7 @@ detail_panel:
|
|||||||
# click-type: unknown
|
# click-type: unknown
|
||||||
# tooltip: level.gui.tips.click-to-select
|
# tooltip: level.gui.tips.click-to-select
|
||||||
2:
|
2:
|
||||||
|
# If a button is used repeatedly then it can be mentioned by name and then defined in the 'reusable' section
|
||||||
2: material_button
|
2: material_button
|
||||||
3: material_button
|
3: material_button
|
||||||
4: material_button
|
4: material_button
|
||||||
@ -85,6 +110,17 @@ detail_panel:
|
|||||||
8: material_button
|
8: material_button
|
||||||
3:
|
3:
|
||||||
1:
|
1:
|
||||||
|
# In this case, the icon is defined as a TIPPED_ARROW with and enchantment applied. The format for the enchantment is
|
||||||
|
# define in {@link world.bentobox.bentobox.util.ItemParser} and available for POTIONS or TIPPED_ARROWs.
|
||||||
|
# Format TIPPED_ARROW:NAME:<LEVEL>:<EXTENDED>:<SPLASH/LINGER>:QTY
|
||||||
|
# LEVEL, EXTENDED, SPLASH, LINGER are optional.
|
||||||
|
# LEVEL is a number, 1 or 2
|
||||||
|
# LINGER is for V1.9 servers and later
|
||||||
|
# Examples:
|
||||||
|
# TIPPED_ARROW:STRENGTH:1:EXTENDED:SPLASH:1
|
||||||
|
# TIPPED_ARROW:INSTANT_DAMAGE:2::LINGER:2
|
||||||
|
# TIPPED_ARROW:JUMP:2:NOTEXTENDED:NOSPLASH:1
|
||||||
|
# TIPPED_ARROW:WEAKNESS::::1 - any weakness enchantment
|
||||||
icon: TIPPED_ARROW:INSTANT_HEAL::::1
|
icon: TIPPED_ARROW:INSTANT_HEAL::::1
|
||||||
title: level.gui.buttons.previous.name
|
title: level.gui.buttons.previous.name
|
||||||
description: level.gui.buttons.previous.description
|
description: level.gui.buttons.previous.description
|
||||||
@ -121,8 +157,12 @@ detail_panel:
|
|||||||
6: material_button
|
6: material_button
|
||||||
7: material_button
|
7: material_button
|
||||||
8: material_button
|
8: material_button
|
||||||
|
# This is where reuable buttons are defined.
|
||||||
reusable:
|
reusable:
|
||||||
|
# This is the name of the button that is referenced
|
||||||
material_button:
|
material_button:
|
||||||
|
# If the icon for a button is not defined, it defaults to AIR and so effectively will not be shown.
|
||||||
|
# icons are usually not defined if the icon is going to be dynamically set in the panel, e.g. in this case the material will vary
|
||||||
#icon: STONE
|
#icon: STONE
|
||||||
title: level.gui.buttons.material.name
|
title: level.gui.buttons.material.name
|
||||||
description: level.gui.buttons.material.description
|
description: level.gui.buttons.material.description
|
||||||
|
@ -1,15 +1,28 @@
|
|||||||
|
# Name of panel used for indentification in the code
|
||||||
top_panel:
|
top_panel:
|
||||||
|
# Title of the panel shown to the user. This is a reference and the reference will be translatable in the locale file
|
||||||
title: level.gui.titles.top
|
title: level.gui.titles.top
|
||||||
|
# The type of panel to show. Options are INVENTORY, HOPPER, DROPPER. INVENTORY is that standard chest inventory and
|
||||||
|
# the others refer to the inventories shown for those items.
|
||||||
type: INVENTORY
|
type: INVENTORY
|
||||||
|
# The background of the panel. These items will be shown if other items are not there. STAINED_GLASS_PANEs give a good effect.
|
||||||
background:
|
background:
|
||||||
icon: BLACK_STAINED_GLASS_PANE
|
icon: BLACK_STAINED_GLASS_PANE
|
||||||
|
# Each item may have text applied to it, but usually for background items, nothing is shown.
|
||||||
title: "&b&r" # Empty text
|
title: "&b&r" # Empty text
|
||||||
border:
|
border:
|
||||||
|
# The border of each panel may be shown as a different item.
|
||||||
|
# It can be used to provide a contrast to items in the panel.
|
||||||
icon: BLACK_STAINED_GLASS_PANE
|
icon: BLACK_STAINED_GLASS_PANE
|
||||||
title: "&b&r" # Empty text
|
title: "&b&r" # Empty text
|
||||||
|
# This tag indicates which rows in the panel must be shown. The panel will be sized vertically accordingly. This does not include the borders.
|
||||||
|
# This can be a list and rows must be between 1 and 6, if used.
|
||||||
force-shown: [2,3,4,5]
|
force-shown: [2,3,4,5]
|
||||||
|
# The content section contains details of each item/button in the panel. The numbers indicate the rows and then then columns of each item.
|
||||||
content:
|
content:
|
||||||
|
# Row number
|
||||||
2:
|
2:
|
||||||
|
# Column number
|
||||||
5:
|
5:
|
||||||
#icon: PLAYER_HEAD
|
#icon: PLAYER_HEAD
|
||||||
title: level.gui.buttons.island.name
|
title: level.gui.buttons.island.name
|
||||||
|
@ -174,15 +174,14 @@ public class LevelTest {
|
|||||||
IslandWorldManager iwm = mock(IslandWorldManager.class);
|
IslandWorldManager iwm = mock(IslandWorldManager.class);
|
||||||
when(plugin.getIWM()).thenReturn(iwm);
|
when(plugin.getIWM()).thenReturn(iwm);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// Player has island to begin with
|
// Player has island to begin with
|
||||||
when(im.getIsland(Mockito.any(), Mockito.any(UUID.class))).thenReturn(island);
|
when(im.getIsland(Mockito.any(), Mockito.any(UUID.class))).thenReturn(island);
|
||||||
when(plugin.getIslands()).thenReturn(im);
|
when(plugin.getIslands()).thenReturn(im);
|
||||||
|
|
||||||
// Locales
|
// Locales
|
||||||
// Return the reference (USE THIS IN THE FUTURE)
|
// Return the reference (USE THIS IN THE FUTURE)
|
||||||
when(user.getTranslation(Mockito.anyString())).thenAnswer((Answer<String>) invocation -> invocation.getArgument(0, String.class));
|
when(user.getTranslation(Mockito.anyString()))
|
||||||
|
.thenAnswer((Answer<String>) invocation -> invocation.getArgument(0, String.class));
|
||||||
|
|
||||||
// Server
|
// Server
|
||||||
PowerMockito.mockStatic(Bukkit.class);
|
PowerMockito.mockStatic(Bukkit.class);
|
||||||
@ -196,14 +195,16 @@ public class LevelTest {
|
|||||||
File dataFolder = new File("addons/Level");
|
File dataFolder = new File("addons/Level");
|
||||||
addon.setDataFolder(dataFolder);
|
addon.setDataFolder(dataFolder);
|
||||||
addon.setFile(jFile);
|
addon.setFile(jFile);
|
||||||
AddonDescription desc = new AddonDescription.Builder("bentobox", "Level", "1.3").description("test").authors("tastybento").build();
|
AddonDescription desc = new AddonDescription.Builder("bentobox", "Level", "1.3").description("test")
|
||||||
|
.authors("tastybento").build();
|
||||||
addon.setDescription(desc);
|
addon.setDescription(desc);
|
||||||
addon.setSettings(new ConfigSettings());
|
addon.setSettings(new ConfigSettings());
|
||||||
// Addons manager
|
// Addons manager
|
||||||
when(plugin.getAddonsManager()).thenReturn(am);
|
when(plugin.getAddonsManager()).thenReturn(am);
|
||||||
// One game mode
|
// One game mode
|
||||||
when(am.getGameModeAddons()).thenReturn(Collections.singletonList(gameMode));
|
when(am.getGameModeAddons()).thenReturn(Collections.singletonList(gameMode));
|
||||||
AddonDescription desc2 = new AddonDescription.Builder("bentobox", "BSkyBlock", "1.3").description("test").authors("tasty").build();
|
AddonDescription desc2 = new AddonDescription.Builder("bentobox", "BSkyBlock", "1.3").description("test")
|
||||||
|
.authors("tasty").build();
|
||||||
when(gameMode.getDescription()).thenReturn(desc2);
|
when(gameMode.getDescription()).thenReturn(desc2);
|
||||||
when(gameMode.getOverWorld()).thenReturn(world);
|
when(gameMode.getOverWorld()).thenReturn(world);
|
||||||
|
|
||||||
@ -219,7 +220,6 @@ public class LevelTest {
|
|||||||
when(plugin.getFlagsManager()).thenReturn(fm);
|
when(plugin.getFlagsManager()).thenReturn(fm);
|
||||||
when(fm.getFlags()).thenReturn(Collections.emptyList());
|
when(fm.getFlags()).thenReturn(Collections.emptyList());
|
||||||
|
|
||||||
|
|
||||||
// Bukkit
|
// Bukkit
|
||||||
PowerMockito.mockStatic(Bukkit.class);
|
PowerMockito.mockStatic(Bukkit.class);
|
||||||
when(Bukkit.getScheduler()).thenReturn(scheduler);
|
when(Bukkit.getScheduler()).thenReturn(scheduler);
|
||||||
@ -260,10 +260,7 @@ public class LevelTest {
|
|||||||
|
|
||||||
private static void deleteAll(File file) throws IOException {
|
private static void deleteAll(File file) throws IOException {
|
||||||
if (file.exists()) {
|
if (file.exists()) {
|
||||||
Files.walk(file.toPath())
|
Files.walk(file.toPath()).sorted(Comparator.reverseOrder()).map(Path::toFile).forEach(File::delete);
|
||||||
.sorted(Comparator.reverseOrder())
|
|
||||||
.map(Path::toFile)
|
|
||||||
.forEach(File::delete);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -276,7 +273,7 @@ public class LevelTest {
|
|||||||
verify(plugin).logWarning("[Level] Level Addon: No such world in blockconfig.yml : acidisland_world");
|
verify(plugin).logWarning("[Level] Level Addon: No such world in blockconfig.yml : acidisland_world");
|
||||||
verify(plugin).log("[Level] Level hooking into BSkyBlock");
|
verify(plugin).log("[Level] Level hooking into BSkyBlock");
|
||||||
verify(cmd, times(3)).getAddon(); // 3 commands
|
verify(cmd, times(3)).getAddon(); // 3 commands
|
||||||
verify(adminCmd, times(4)).getAddon(); // Four commands
|
verify(adminCmd, times(5)).getAddon(); // Five commands
|
||||||
// Placeholders
|
// Placeholders
|
||||||
verify(phm).registerPlaceholder(eq(addon), eq("bskyblock_island_level"), any());
|
verify(phm).registerPlaceholder(eq(addon), eq("bskyblock_island_level"), any());
|
||||||
verify(phm).registerPlaceholder(eq(addon), eq("bskyblock_visited_island_level"), any());
|
verify(phm).registerPlaceholder(eq(addon), eq("bskyblock_visited_island_level"), any());
|
||||||
|
@ -82,7 +82,6 @@ public class LevelsManagerTest {
|
|||||||
@Mock
|
@Mock
|
||||||
private Settings pluginSettings;
|
private Settings pluginSettings;
|
||||||
|
|
||||||
|
|
||||||
// Class under test
|
// Class under test
|
||||||
private LevelsManager lm;
|
private LevelsManager lm;
|
||||||
@Mock
|
@Mock
|
||||||
@ -114,12 +113,11 @@ public class LevelsManagerTest {
|
|||||||
@Mock
|
@Mock
|
||||||
private BukkitScheduler scheduler;
|
private BukkitScheduler scheduler;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
@BeforeClass
|
@BeforeClass
|
||||||
public static void beforeClass() {
|
public static void beforeClass() {
|
||||||
// This has to be done beforeClass otherwise the tests will interfere with each other
|
// This has to be done beforeClass otherwise the tests will interfere with each
|
||||||
|
// other
|
||||||
handler = mock(AbstractDatabaseHandler.class);
|
handler = mock(AbstractDatabaseHandler.class);
|
||||||
// Database
|
// Database
|
||||||
PowerMockito.mockStatic(DatabaseSetup.class);
|
PowerMockito.mockStatic(DatabaseSetup.class);
|
||||||
@ -162,10 +160,9 @@ public class LevelsManagerTest {
|
|||||||
when(island.getMemberSet()).thenReturn(iset);
|
when(island.getMemberSet()).thenReturn(iset);
|
||||||
when(island.getOwner()).thenReturn(uuid);
|
when(island.getOwner()).thenReturn(uuid);
|
||||||
when(island.getWorld()).thenReturn(world);
|
when(island.getWorld()).thenReturn(world);
|
||||||
when(island.getUniqueId()).thenReturn(UUID.randomUUID().toString());
|
when(island.getUniqueId()).thenReturn(uuid.toString());
|
||||||
// Default to uuid's being island owners
|
// Default to uuid's being island owners
|
||||||
when(im.isOwner(eq(world), any())).thenReturn(true);
|
when(im.hasIsland(eq(world), any(UUID.class))).thenReturn(true);
|
||||||
when(im.getOwner(any(), any(UUID.class))).thenAnswer(in -> in.getArgument(1, UUID.class));
|
|
||||||
when(im.getIsland(world, uuid)).thenReturn(island);
|
when(im.getIsland(world, uuid)).thenReturn(island);
|
||||||
when(im.getIslandById(anyString())).thenReturn(Optional.of(island));
|
when(im.getIslandById(anyString())).thenReturn(Optional.of(island));
|
||||||
|
|
||||||
@ -247,15 +244,13 @@ public class LevelsManagerTest {
|
|||||||
|
|
||||||
private static void deleteAll(File file) throws IOException {
|
private static void deleteAll(File file) throws IOException {
|
||||||
if (file.exists()) {
|
if (file.exists()) {
|
||||||
Files.walk(file.toPath())
|
Files.walk(file.toPath()).sorted(Comparator.reverseOrder()).map(Path::toFile).forEach(File::delete);
|
||||||
.sorted(Comparator.reverseOrder())
|
|
||||||
.map(Path::toFile)
|
|
||||||
.forEach(File::delete);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.LevelsManager#calculateLevel(UUID, world.bentobox.bentobox.database.objects.Island)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.LevelsManager#calculateLevel(UUID, world.bentobox.bentobox.database.objects.Island)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testCalculateLevel() {
|
public void testCalculateLevel() {
|
||||||
@ -283,7 +278,8 @@ public class LevelsManagerTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.LevelsManager#getInitialLevel(world.bentobox.bentobox.database.objects.Island)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.LevelsManager#getInitialLevel(world.bentobox.bentobox.database.objects.Island)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testGetInitialLevel() {
|
public void testGetInitialLevel() {
|
||||||
@ -291,7 +287,8 @@ public class LevelsManagerTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.LevelsManager#getIslandLevel(org.bukkit.World, java.util.UUID)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.LevelsManager#getIslandLevel(org.bukkit.World, java.util.UUID)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testGetIslandLevel() {
|
public void testGetIslandLevel() {
|
||||||
@ -299,7 +296,8 @@ public class LevelsManagerTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.LevelsManager#getPointsToNextString(org.bukkit.World, java.util.UUID)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.LevelsManager#getPointsToNextString(org.bukkit.World, java.util.UUID)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testGetPointsToNextString() {
|
public void testGetPointsToNextString() {
|
||||||
@ -310,7 +308,8 @@ public class LevelsManagerTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.LevelsManager#getIslandLevelString(org.bukkit.World, java.util.UUID)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.LevelsManager#getIslandLevelString(org.bukkit.World, java.util.UUID)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testGetIslandLevelString() {
|
public void testGetIslandLevelString() {
|
||||||
@ -318,7 +317,8 @@ public class LevelsManagerTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.LevelsManager#getLevelsData(java.util.UUID)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.LevelsManager#getLevelsData(java.util.UUID)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testGetLevelsData() {
|
public void testGetLevelsData() {
|
||||||
@ -342,39 +342,44 @@ public class LevelsManagerTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.LevelsManager#getTopTen(org.bukkit.World, int)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.LevelsManager#getTopTen(org.bukkit.World, int)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testGetTopTenEmpty() {
|
public void testGetTopTenEmpty() {
|
||||||
Map<UUID, Long> tt = lm.getTopTen(world, Level.TEN);
|
Map<String, Long> tt = lm.getTopTen(world, Level.TEN);
|
||||||
assertTrue(tt.isEmpty());
|
assertTrue(tt.isEmpty());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.LevelsManager#getTopTen(org.bukkit.World, int)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.LevelsManager#getTopTen(org.bukkit.World, int)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testGetTopTen() {
|
public void testGetTopTen() {
|
||||||
testLoadTopTens();
|
testLoadTopTens();
|
||||||
Map<UUID, Long> tt = lm.getTopTen(world, Level.TEN);
|
Map<String, Long> tt = lm.getTopTen(world, Level.TEN);
|
||||||
assertFalse(tt.isEmpty());
|
assertFalse(tt.isEmpty());
|
||||||
assertEquals(1, tt.size());
|
assertEquals(1, tt.size());
|
||||||
assertEquals(1, lm.getTopTen(world, 1).size());
|
assertEquals(1, lm.getTopTen(world, 1).size());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.LevelsManager#getTopTen(org.bukkit.World, int)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.LevelsManager#getWeightedTopTen(org.bukkit.World, int)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testGetTopTenNoOwners() {
|
public void testGetWeightedTopTen() {
|
||||||
when(im.isOwner(eq(world), any())).thenReturn(false);
|
|
||||||
testLoadTopTens();
|
testLoadTopTens();
|
||||||
Map<UUID, Long> tt = lm.getTopTen(world, Level.TEN);
|
Map<Island, Long> tt = lm.getWeightedTopTen(world, Level.TEN);
|
||||||
assertTrue(tt.isEmpty());
|
assertFalse(tt.isEmpty());
|
||||||
|
assertEquals(1, tt.size());
|
||||||
|
assertEquals(1, lm.getTopTen(world, 1).size());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.LevelsManager#hasTopTenPerm(org.bukkit.World, java.util.UUID)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.LevelsManager#hasTopTenPerm(org.bukkit.World, java.util.UUID)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testHasTopTenPerm() {
|
public void testHasTopTenPerm() {
|
||||||
@ -398,20 +403,22 @@ public class LevelsManagerTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.LevelsManager#removeEntry(org.bukkit.World, java.util.UUID)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.LevelsManager#removeEntry(org.bukkit.World, java.util.UUID)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testRemoveEntry() {
|
public void testRemoveEntry() {
|
||||||
testLoadTopTens();
|
testLoadTopTens();
|
||||||
Map<UUID, Long> tt = lm.getTopTen(world, Level.TEN);
|
Map<String, Long> tt = lm.getTopTen(world, Level.TEN);
|
||||||
assertTrue(tt.containsKey(uuid));
|
assertTrue(tt.containsKey(uuid.toString()));
|
||||||
lm.removeEntry(world, uuid);
|
lm.removeEntry(world, uuid.toString());
|
||||||
tt = lm.getTopTen(world, Level.TEN);
|
tt = lm.getTopTen(world, Level.TEN);
|
||||||
assertFalse(tt.containsKey(uuid));
|
assertFalse(tt.containsKey(uuid.toString()));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.LevelsManager#setInitialIslandLevel(world.bentobox.bentobox.database.objects.Island, long)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.LevelsManager#setInitialIslandLevel(world.bentobox.bentobox.database.objects.Island, long)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testSetInitialIslandLevel() {
|
public void testSetInitialIslandLevel() {
|
||||||
@ -420,7 +427,8 @@ public class LevelsManagerTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.LevelsManager#setIslandLevel(org.bukkit.World, java.util.UUID, long)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.LevelsManager#setIslandLevel(org.bukkit.World, java.util.UUID, long)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testSetIslandLevel() {
|
public void testSetIslandLevel() {
|
||||||
@ -429,23 +437,23 @@ public class LevelsManagerTest {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.LevelsManager#getRank(World, UUID)}
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.LevelsManager#getRank(World, UUID)}
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testGetRank() {
|
public void testGetRank() {
|
||||||
lm.createAndCleanRankings(world);
|
lm.createAndCleanRankings(world);
|
||||||
Map<World, TopTenData> ttl = lm.getTopTenLists();
|
Map<World, TopTenData> ttl = lm.getTopTenLists();
|
||||||
Map<UUID, Long> tt = ttl.get(world).getTopTen();
|
Map<String, Long> tt = ttl.get(world).getTopTen();
|
||||||
for (long i = 100; i < 150; i++) {
|
for (long i = 100; i < 150; i++) {
|
||||||
tt.put(UUID.randomUUID(), i);
|
tt.put(UUID.randomUUID().toString(), i);
|
||||||
}
|
}
|
||||||
// Put player as lowest rank
|
// Put island as lowest rank
|
||||||
tt.put(uuid, 10L);
|
tt.put(uuid.toString(), 10L);
|
||||||
assertEquals(51, lm.getRank(world, uuid));
|
assertEquals(51, lm.getRank(world, uuid));
|
||||||
// Put player as highest rank
|
// Put island as highest rank
|
||||||
tt.put(uuid, 1000L);
|
tt.put(uuid.toString(), 1000L);
|
||||||
assertEquals(1, lm.getRank(world, uuid));
|
assertEquals(1, lm.getRank(world, uuid));
|
||||||
// Unknown UUID - lowest rank + 1
|
// Unknown UUID - lowest rank + 1
|
||||||
assertEquals(52, lm.getRank(world, UUID.randomUUID()));
|
assertEquals(52, lm.getRank(world, UUID.randomUUID()));
|
||||||
|
@ -3,6 +3,7 @@ package world.bentobox.level;
|
|||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.Assert.assertEquals;
|
||||||
import static org.junit.Assert.assertTrue;
|
import static org.junit.Assert.assertTrue;
|
||||||
import static org.mockito.ArgumentMatchers.any;
|
import static org.mockito.ArgumentMatchers.any;
|
||||||
|
import static org.mockito.ArgumentMatchers.anyString;
|
||||||
import static org.mockito.ArgumentMatchers.eq;
|
import static org.mockito.ArgumentMatchers.eq;
|
||||||
import static org.mockito.Mockito.mock;
|
import static org.mockito.Mockito.mock;
|
||||||
import static org.mockito.Mockito.verify;
|
import static org.mockito.Mockito.verify;
|
||||||
@ -10,9 +11,10 @@ import static org.mockito.Mockito.when;
|
|||||||
|
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
|
import java.util.HashSet;
|
||||||
import java.util.LinkedHashMap;
|
import java.util.LinkedHashMap;
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
import java.util.Map.Entry;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
import java.util.stream.Collectors;
|
import java.util.stream.Collectors;
|
||||||
@ -54,7 +56,7 @@ public class PlaceholderManagerTest {
|
|||||||
@Mock
|
@Mock
|
||||||
private BentoBox plugin;
|
private BentoBox plugin;
|
||||||
|
|
||||||
private PlaceholderManager pm;
|
private PlaceholderManager phm;
|
||||||
@Mock
|
@Mock
|
||||||
private PlaceholdersManager bpm;
|
private PlaceholdersManager bpm;
|
||||||
@Mock
|
@Mock
|
||||||
@ -67,13 +69,25 @@ public class PlaceholderManagerTest {
|
|||||||
private Island island;
|
private Island island;
|
||||||
@Mock
|
@Mock
|
||||||
private User user;
|
private User user;
|
||||||
private Map<UUID, String> names = new HashMap<>();
|
private static final Map<UUID, String> names = new LinkedHashMap<>();
|
||||||
private static final List<String> NAMES = List.of("tasty", "bento", "fred", "bonne", "cyprien", "mael", "joe", "horacio", "steph", "vicky");
|
static {
|
||||||
private Map<UUID, Island> islands = new HashMap<>();
|
names.put(UUID.randomUUID(), "tasty");
|
||||||
private Map<UUID, Long> map = new HashMap<>();
|
names.put(UUID.randomUUID(), "bento");
|
||||||
|
names.put(UUID.randomUUID(), "fred");
|
||||||
|
names.put(UUID.randomUUID(), "bonne");
|
||||||
|
names.put(UUID.randomUUID(), "cyprien");
|
||||||
|
names.put(UUID.randomUUID(), "mael");
|
||||||
|
names.put(UUID.randomUUID(), "joe");
|
||||||
|
names.put(UUID.randomUUID(), "horacio");
|
||||||
|
names.put(UUID.randomUUID(), "steph");
|
||||||
|
names.put(UUID.randomUUID(), "vicky");
|
||||||
|
}
|
||||||
|
private Map<String, Island> islands = new HashMap<>();
|
||||||
|
private Map<String, Long> map = new LinkedHashMap<>();
|
||||||
|
private Map<Island, Long> map2 = new LinkedHashMap<>();
|
||||||
private @NonNull IslandLevels data;
|
private @NonNull IslandLevels data;
|
||||||
@Mock
|
@Mock
|
||||||
private PlayersManager players;
|
private PlayersManager pm;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @throws java.lang.Exception
|
* @throws java.lang.Exception
|
||||||
@ -83,29 +97,32 @@ public class PlaceholderManagerTest {
|
|||||||
when(addon.getPlugin()).thenReturn(plugin);
|
when(addon.getPlugin()).thenReturn(plugin);
|
||||||
|
|
||||||
// Users
|
// Users
|
||||||
when(addon.getPlayers()).thenReturn(players);
|
when(addon.getPlayers()).thenReturn(pm);
|
||||||
|
|
||||||
// Users
|
// Users
|
||||||
when(user.getWorld()).thenReturn(world);
|
when(user.getWorld()).thenReturn(world);
|
||||||
when(user.getLocation()).thenReturn(mock(Location.class));
|
when(user.getLocation()).thenReturn(mock(Location.class));
|
||||||
|
|
||||||
for (int i = 0; i < Level.TEN; i++) {
|
int i = 0;
|
||||||
UUID uuid = UUID.randomUUID();
|
for (Entry<UUID, String> n : names.entrySet()) {
|
||||||
names.put(uuid, NAMES.get(i));
|
UUID uuid = UUID.randomUUID(); // Random island ID
|
||||||
map.put(uuid, (long)(100 - i));
|
Long value = (long)(100 - i++);
|
||||||
|
map.put(uuid.toString(), value); // level
|
||||||
Island is = new Island();
|
Island is = new Island();
|
||||||
is.setOwner(uuid);
|
is.setUniqueId(uuid.toString());
|
||||||
is.setName(NAMES.get(i) + "'s island");
|
is.setOwner(n.getKey());
|
||||||
islands.put(uuid, is);
|
is.setName(n.getValue() + "'s island");
|
||||||
|
islands.put(uuid.toString(), is);
|
||||||
|
map2.put(is, value);
|
||||||
}
|
}
|
||||||
// Sort
|
// Sort
|
||||||
map = map.entrySet().stream()
|
map = map.entrySet().stream()
|
||||||
.sorted(Collections.reverseOrder(Map.Entry.comparingByValue()))
|
.sorted(Collections.reverseOrder(Map.Entry.comparingByValue()))
|
||||||
.collect(Collectors.toMap( Map.Entry::getKey, Map.Entry::getValue, (e1, e2) -> e1, LinkedHashMap::new));
|
.collect(Collectors.toMap( Map.Entry::getKey, Map.Entry::getValue, (e1, e2) -> e1, LinkedHashMap::new));
|
||||||
when(players.getName(any())).thenAnswer((Answer<String>) invocation -> names.getOrDefault(invocation.getArgument(0, UUID.class), "unknown"));
|
when(pm.getName(any())).thenAnswer((Answer<String>) invocation -> names.getOrDefault(invocation.getArgument(0, UUID.class), "unknown"));
|
||||||
Map<UUID, Integer> members = new HashMap<>();
|
Map<UUID, Integer> members = new HashMap<>();
|
||||||
map.forEach((uuid, l) -> members.put(uuid, RanksManager.MEMBER_RANK));
|
names.forEach((uuid, l) -> members.put(uuid, RanksManager.MEMBER_RANK));
|
||||||
islands.values().forEach(i -> i.setMembers(members));
|
islands.values().forEach(is -> is.setMembers(members));
|
||||||
|
|
||||||
|
|
||||||
// Placeholders manager for plugin
|
// Placeholders manager for plugin
|
||||||
@ -120,7 +137,8 @@ public class PlaceholderManagerTest {
|
|||||||
// Islands
|
// Islands
|
||||||
when(im.getIsland(any(World.class), any(User.class))).thenReturn(island);
|
when(im.getIsland(any(World.class), any(User.class))).thenReturn(island);
|
||||||
when(im.getIslandAt(any(Location.class))).thenReturn(Optional.of(island));
|
when(im.getIslandAt(any(Location.class))).thenReturn(Optional.of(island));
|
||||||
when(im.getIsland(any(World.class), any(UUID.class))).thenAnswer((Answer<Island>) invocation -> islands.get(invocation.getArgument(1, UUID.class)));
|
when(im.getIslandById(anyString())).thenAnswer((Answer<Optional<Island>>) invocation -> Optional.of(islands.get(invocation.getArgument(0, String.class))));
|
||||||
|
when(im.getIslands(any(), any(UUID.class))).thenReturn(new HashSet<>(islands.values()));
|
||||||
when(addon.getIslands()).thenReturn(im);
|
when(addon.getIslands()).thenReturn(im);
|
||||||
|
|
||||||
// Levels Manager
|
// Levels Manager
|
||||||
@ -129,6 +147,7 @@ public class PlaceholderManagerTest {
|
|||||||
when(lm.getPointsToNextString(any(), any())).thenReturn("1234567");
|
when(lm.getPointsToNextString(any(), any())).thenReturn("1234567");
|
||||||
when(lm.getIslandMaxLevel(any(), any())).thenReturn(987654L);
|
when(lm.getIslandMaxLevel(any(), any())).thenReturn(987654L);
|
||||||
when(lm.getTopTen(world, Level.TEN)).thenReturn(map);
|
when(lm.getTopTen(world, Level.TEN)).thenReturn(map);
|
||||||
|
when(lm.getWeightedTopTen(world, Level.TEN)).thenReturn(map2);
|
||||||
when(lm.formatLevel(any())).thenAnswer((Answer<String>) invocation -> invocation.getArgument(0, Long.class).toString());
|
when(lm.formatLevel(any())).thenAnswer((Answer<String>) invocation -> invocation.getArgument(0, Long.class).toString());
|
||||||
|
|
||||||
data = new IslandLevels("uniqueId");
|
data = new IslandLevels("uniqueId");
|
||||||
@ -136,11 +155,12 @@ public class PlaceholderManagerTest {
|
|||||||
when(lm.getLevelsData(island)).thenReturn(data);
|
when(lm.getLevelsData(island)).thenReturn(data);
|
||||||
when(addon.getManager()).thenReturn(lm);
|
when(addon.getManager()).thenReturn(lm);
|
||||||
|
|
||||||
pm = new PlaceholderManager(addon);
|
phm = new PlaceholderManager(addon);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.PlaceholderManager#PlaceholderManager(world.bentobox.level.Level)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.PlaceholderManager#PlaceholderManager(world.bentobox.level.Level)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testPlaceholderManager() {
|
public void testPlaceholderManager() {
|
||||||
@ -148,11 +168,12 @@ public class PlaceholderManagerTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.PlaceholderManager#registerPlaceholders(world.bentobox.bentobox.api.addons.GameModeAddon)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.PlaceholderManager#registerPlaceholders(world.bentobox.bentobox.api.addons.GameModeAddon)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testRegisterPlaceholders() {
|
public void testRegisterPlaceholders() {
|
||||||
pm.registerPlaceholders(gm);
|
phm.registerPlaceholders(gm);
|
||||||
// Island Level
|
// Island Level
|
||||||
verify(bpm).registerPlaceholder(eq(addon), eq("aoneblock_island_level"), any());
|
verify(bpm).registerPlaceholder(eq(addon), eq("aoneblock_island_level"), any());
|
||||||
verify(bpm).registerPlaceholder(eq(addon), eq("aoneblock_island_level_raw"), any());
|
verify(bpm).registerPlaceholder(eq(addon), eq("aoneblock_island_level_raw"), any());
|
||||||
@ -182,78 +203,99 @@ public class PlaceholderManagerTest {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.PlaceholderManager#getRankName(org.bukkit.World, int)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.PlaceholderManager#getRankName(org.bukkit.World, int)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testGetRankName() {
|
public void testGetRankName() {
|
||||||
// Test extremes
|
// Test extremes
|
||||||
assertEquals("tasty", pm.getRankName(world, 0));
|
assertEquals("tasty", phm.getRankName(world, 0, false));
|
||||||
assertEquals("vicky", pm.getRankName(world, 100));
|
assertEquals("vicky", phm.getRankName(world, 100, false));
|
||||||
// Test the ranks
|
// Test the ranks
|
||||||
int rank = 1;
|
int rank = 1;
|
||||||
for (String name : NAMES) {
|
for (String name : names.values()) {
|
||||||
assertEquals(name, pm.getRankName(world, rank++));
|
assertEquals(name, phm.getRankName(world, rank++, false));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.PlaceholderManager#getRankIslandName(org.bukkit.World, int)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.PlaceholderManager#getRankIslandName(org.bukkit.World, int)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testGetRankIslandName() {
|
public void testGetRankIslandName() {
|
||||||
// Test extremes
|
// Test extremes
|
||||||
assertEquals("tasty's island", pm.getRankIslandName(world, 0));
|
assertEquals("tasty's island", phm.getRankIslandName(world, 0, false));
|
||||||
assertEquals("vicky's island", pm.getRankIslandName(world, 100));
|
assertEquals("vicky's island", phm.getRankIslandName(world, 100, false));
|
||||||
// Test the ranks
|
// Test the ranks
|
||||||
int rank = 1;
|
int rank = 1;
|
||||||
for (String name : NAMES) {
|
for (String name : names.values()) {
|
||||||
assertEquals(name + "'s island", pm.getRankIslandName(world, rank++));
|
assertEquals(name + "'s island", phm.getRankIslandName(world, rank++, false));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.PlaceholderManager#getRankMembers(org.bukkit.World, int)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.PlaceholderManager#getRankMembers(org.bukkit.World, int)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testGetRankMembers() {
|
public void testGetRankMembers() {
|
||||||
// Test extremes
|
// Test extremes
|
||||||
check(1, pm.getRankMembers(world, 0));
|
check(1, phm.getRankMembers(world, 0, false));
|
||||||
check(2, pm.getRankMembers(world, 100));
|
check(2, phm.getRankMembers(world, 100, false));
|
||||||
// Test the ranks
|
// Test the ranks
|
||||||
for (int rank = 1; rank < 11; rank++) {
|
for (int rank = 1; rank < 11; rank++) {
|
||||||
check(3, pm.getRankMembers(world, rank));
|
check(3, phm.getRankMembers(world, rank, false));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void check(int indicator, String list) {
|
void check(int indicator, String list) {
|
||||||
for (String n : NAMES) {
|
for (String n : names.values()) {
|
||||||
assertTrue(n + " is missing for twst " + indicator, list.contains(n));
|
assertTrue(n + " is missing for test " + indicator, list.contains(n));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.PlaceholderManager#getRankLevel(org.bukkit.World, int)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.PlaceholderManager#getRankLevel(org.bukkit.World, int)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testGetRankLevel() {
|
public void testGetRankLevel() {
|
||||||
// Test extremes
|
// Test extremes
|
||||||
assertEquals("100", pm.getRankLevel(world, 0));
|
assertEquals("100", phm.getRankLevel(world, 0, false));
|
||||||
assertEquals("91", pm.getRankLevel(world, 100));
|
assertEquals("91", phm.getRankLevel(world, 100, false));
|
||||||
// Test the ranks
|
// Test the ranks
|
||||||
for (int rank = 1; rank < 11; rank++) {
|
for (int rank = 1; rank < 11; rank++) {
|
||||||
assertEquals(String.valueOf(101 - rank), pm.getRankLevel(world, rank));
|
assertEquals(String.valueOf(101 - rank), phm.getRankLevel(world, rank, false));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.PlaceholderManager#getVisitedIslandLevel(world.bentobox.bentobox.api.addons.GameModeAddon, world.bentobox.bentobox.api.user.User)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.PlaceholderManager#getRankLevel(org.bukkit.World, int)}.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testGetWeightedRankLevel() {
|
||||||
|
// Test extremes
|
||||||
|
assertEquals("100", phm.getRankLevel(world, 0, true));
|
||||||
|
assertEquals("91", phm.getRankLevel(world, 100, true));
|
||||||
|
// Test the ranks
|
||||||
|
for (int rank = 1; rank < 11; rank++) {
|
||||||
|
assertEquals(String.valueOf(101 - rank), phm.getRankLevel(world, rank, true));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.PlaceholderManager#getVisitedIslandLevel(world.bentobox.bentobox.api.addons.GameModeAddon, world.bentobox.bentobox.api.user.User)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testGetVisitedIslandLevelNullUser() {
|
public void testGetVisitedIslandLevelNullUser() {
|
||||||
assertEquals("", pm.getVisitedIslandLevel(gm, null));
|
assertEquals("", phm.getVisitedIslandLevel(gm, null));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -264,16 +306,17 @@ public class PlaceholderManagerTest {
|
|||||||
public void testGetVisitedIslandLevelUserNotInWorld() {
|
public void testGetVisitedIslandLevelUserNotInWorld() {
|
||||||
// Another world
|
// Another world
|
||||||
when(user.getWorld()).thenReturn(mock(World.class));
|
when(user.getWorld()).thenReturn(mock(World.class));
|
||||||
assertEquals("", pm.getVisitedIslandLevel(gm, user));
|
assertEquals("", phm.getVisitedIslandLevel(gm, user));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Test method for {@link world.bentobox.level.PlaceholderManager#getVisitedIslandLevel(world.bentobox.bentobox.api.addons.GameModeAddon, world.bentobox.bentobox.api.user.User)}.
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.PlaceholderManager#getVisitedIslandLevel(world.bentobox.bentobox.api.addons.GameModeAddon, world.bentobox.bentobox.api.user.User)}.
|
||||||
*/
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testGetVisitedIslandLevel() {
|
public void testGetVisitedIslandLevel() {
|
||||||
assertEquals("1234567", pm.getVisitedIslandLevel(gm, user));
|
assertEquals("1234567", phm.getVisitedIslandLevel(gm, user));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -283,7 +326,7 @@ public class PlaceholderManagerTest {
|
|||||||
@Test
|
@Test
|
||||||
public void testGetVisitedIslandLevelNoIsland() {
|
public void testGetVisitedIslandLevelNoIsland() {
|
||||||
when(im.getIslandAt(any(Location.class))).thenReturn(Optional.empty());
|
when(im.getIslandAt(any(Location.class))).thenReturn(Optional.empty());
|
||||||
assertEquals("0", pm.getVisitedIslandLevel(gm, user));
|
assertEquals("0", phm.getVisitedIslandLevel(gm, user));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -0,0 +1,37 @@
|
|||||||
|
package world.bentobox.level.calculators;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
|
||||||
|
import java.text.ParseException;
|
||||||
|
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test the equation evaluation
|
||||||
|
*/
|
||||||
|
public class EquationEvaluatorTest {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test method for {@link world.bentobox.level.calculators.EquationEvaluator#eval(java.lang.String)}.
|
||||||
|
* @throws ParseException
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testEval() throws ParseException {
|
||||||
|
assertEquals(4D, EquationEvaluator.eval("2+2"), 0D);
|
||||||
|
assertEquals(0D, EquationEvaluator.eval("2-2"), 0D);
|
||||||
|
assertEquals(1D, EquationEvaluator.eval("2/2"), 0D);
|
||||||
|
assertEquals(4D, EquationEvaluator.eval("2*2"), 0D);
|
||||||
|
assertEquals(8D, EquationEvaluator.eval("2+2+2+2"), 0D);
|
||||||
|
assertEquals(5D, EquationEvaluator.eval("2.5+2.5"), 0D);
|
||||||
|
assertEquals(1.414, EquationEvaluator.eval("sqrt(2)"), 0.001D);
|
||||||
|
assertEquals(3.414, EquationEvaluator.eval("2 + sqrt(2)"), 0.001D);
|
||||||
|
assertEquals(0D, EquationEvaluator.eval("sin(0)"), 0.1D);
|
||||||
|
assertEquals(1D, EquationEvaluator.eval("cos(0)"), 0.1D);
|
||||||
|
assertEquals(0D, EquationEvaluator.eval("tan(0)"), 0.1D);
|
||||||
|
assertEquals(0D, EquationEvaluator.eval("log(1)"), 0.1D);
|
||||||
|
assertEquals(27D, EquationEvaluator.eval("3^3"), 0.D);
|
||||||
|
assertEquals(84.70332D, EquationEvaluator.eval("3^3 + 2 + 2.65 * (3 / 4) - sin(45) * log(10) + 55.344"),
|
||||||
|
0.0001D);
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,183 @@
|
|||||||
|
package world.bentobox.level.commands;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertFalse;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
import static org.mockito.ArgumentMatchers.any;
|
||||||
|
import static org.mockito.ArgumentMatchers.anyString;
|
||||||
|
import static org.mockito.Mockito.mock;
|
||||||
|
import static org.mockito.Mockito.never;
|
||||||
|
import static org.mockito.Mockito.verify;
|
||||||
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Random;
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
import org.bukkit.Bukkit;
|
||||||
|
import org.bukkit.Server;
|
||||||
|
import org.bukkit.World;
|
||||||
|
import org.bukkit.entity.Player;
|
||||||
|
import org.bukkit.inventory.ItemFactory;
|
||||||
|
import org.bukkit.inventory.meta.ItemMeta;
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.mockito.Mock;
|
||||||
|
import org.powermock.api.mockito.PowerMockito;
|
||||||
|
import org.powermock.core.classloader.annotations.PrepareForTest;
|
||||||
|
import org.powermock.modules.junit4.PowerMockRunner;
|
||||||
|
import org.powermock.reflect.Whitebox;
|
||||||
|
|
||||||
|
import world.bentobox.bentobox.BentoBox;
|
||||||
|
import world.bentobox.bentobox.api.addons.GameModeAddon;
|
||||||
|
import world.bentobox.bentobox.api.commands.CompositeCommand;
|
||||||
|
import world.bentobox.bentobox.api.user.User;
|
||||||
|
import world.bentobox.bentobox.database.objects.Island;
|
||||||
|
import world.bentobox.bentobox.managers.IslandWorldManager;
|
||||||
|
import world.bentobox.bentobox.managers.IslandsManager;
|
||||||
|
import world.bentobox.bentobox.managers.LocalesManager;
|
||||||
|
import world.bentobox.bentobox.managers.PlayersManager;
|
||||||
|
import world.bentobox.level.Level;
|
||||||
|
import world.bentobox.level.LevelsManager;
|
||||||
|
import world.bentobox.level.objects.TopTenData;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author tastybento
|
||||||
|
*/
|
||||||
|
@RunWith(PowerMockRunner.class)
|
||||||
|
@PrepareForTest({ Bukkit.class, BentoBox.class })
|
||||||
|
public class AdminStatsCommandTest {
|
||||||
|
|
||||||
|
@Mock
|
||||||
|
private CompositeCommand ic;
|
||||||
|
private UUID uuid;
|
||||||
|
@Mock
|
||||||
|
private User user;
|
||||||
|
@Mock
|
||||||
|
private IslandsManager im;
|
||||||
|
@Mock
|
||||||
|
private Island island;
|
||||||
|
@Mock
|
||||||
|
private Level addon;
|
||||||
|
@Mock
|
||||||
|
private World world;
|
||||||
|
@Mock
|
||||||
|
private IslandWorldManager iwm;
|
||||||
|
@Mock
|
||||||
|
private GameModeAddon gameModeAddon;
|
||||||
|
@Mock
|
||||||
|
private Player p;
|
||||||
|
@Mock
|
||||||
|
private LocalesManager lm;
|
||||||
|
@Mock
|
||||||
|
private PlayersManager pm;
|
||||||
|
|
||||||
|
private AdminStatsCommand asc;
|
||||||
|
private TopTenData ttd;
|
||||||
|
@Mock
|
||||||
|
private LevelsManager manager;
|
||||||
|
@Mock
|
||||||
|
private Server server;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setUp() {
|
||||||
|
// Set up plugin
|
||||||
|
BentoBox plugin = mock(BentoBox.class);
|
||||||
|
Whitebox.setInternalState(BentoBox.class, "instance", plugin);
|
||||||
|
User.setPlugin(plugin);
|
||||||
|
when(addon.getPlugin()).thenReturn(plugin);
|
||||||
|
|
||||||
|
// Addon
|
||||||
|
when(ic.getAddon()).thenReturn(addon);
|
||||||
|
when(ic.getPermissionPrefix()).thenReturn("bskyblock.");
|
||||||
|
when(ic.getLabel()).thenReturn("island");
|
||||||
|
when(ic.getTopLabel()).thenReturn("island");
|
||||||
|
when(ic.getWorld()).thenReturn(world);
|
||||||
|
when(ic.getTopLabel()).thenReturn("bsb");
|
||||||
|
|
||||||
|
// IWM friendly name
|
||||||
|
when(plugin.getIWM()).thenReturn(iwm);
|
||||||
|
when(iwm.getFriendlyName(any())).thenReturn("BSkyBlock");
|
||||||
|
|
||||||
|
// World
|
||||||
|
when(world.toString()).thenReturn("world");
|
||||||
|
when(world.getName()).thenReturn("BSkyBlock_world");
|
||||||
|
|
||||||
|
// Player manager
|
||||||
|
when(plugin.getPlayers()).thenReturn(pm);
|
||||||
|
when(pm.getUser(anyString())).thenReturn(user);
|
||||||
|
// topTen
|
||||||
|
when(addon.getManager()).thenReturn(manager);
|
||||||
|
// User
|
||||||
|
uuid = UUID.randomUUID();
|
||||||
|
when(user.getUniqueId()).thenReturn(uuid);
|
||||||
|
when(user.getTranslation(any())).thenAnswer(invocation -> invocation.getArgument(0, String.class));
|
||||||
|
|
||||||
|
// Bukkit
|
||||||
|
PowerMockito.mockStatic(Bukkit.class);
|
||||||
|
when(Bukkit.getServer()).thenReturn(server);
|
||||||
|
// Mock item factory (for itemstacks)
|
||||||
|
ItemFactory itemFactory = mock(ItemFactory.class);
|
||||||
|
ItemMeta itemMeta = mock(ItemMeta.class);
|
||||||
|
when(itemFactory.getItemMeta(any())).thenReturn(itemMeta);
|
||||||
|
when(server.getItemFactory()).thenReturn(itemFactory);
|
||||||
|
when(Bukkit.getItemFactory()).thenReturn(itemFactory);
|
||||||
|
|
||||||
|
// Top ten
|
||||||
|
ttd = new TopTenData(world);
|
||||||
|
Map<String, Long> topten = new HashMap<>();
|
||||||
|
Random r = new Random();
|
||||||
|
for (int i = 0; i < 1000; i++) {
|
||||||
|
topten.put(UUID.randomUUID().toString(), r.nextLong(20000));
|
||||||
|
}
|
||||||
|
ttd.setTopTen(topten);
|
||||||
|
asc = new AdminStatsCommand(addon, ic);
|
||||||
|
}
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void tearDown() {
|
||||||
|
User.clearUsers();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.commands.AdminStatsCommand#setup()}.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testSetup() {
|
||||||
|
assertEquals("bskyblock.admin.stats", asc.getPermission());
|
||||||
|
assertFalse(asc.isOnlyPlayer());
|
||||||
|
assertEquals("admin.stats.description", asc.getDescription());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.commands.AdminStatsCommand#execute(world.bentobox.bentobox.api.user.User, java.lang.String, java.util.List)}.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testExecuteUserStringListOfString() {
|
||||||
|
assertFalse(asc.execute(user, "", List.of()));
|
||||||
|
verify(user).sendMessage("admin.stats.title");
|
||||||
|
verify(user).sendMessage("admin.stats.no-data");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.commands.AdminStatsCommand#execute(world.bentobox.bentobox.api.user.User, java.lang.String, java.util.List)}.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testExecuteUserStringListOfStringLevels() {
|
||||||
|
Map<World, TopTenData> map = new HashMap<>();
|
||||||
|
map.put(world, ttd);
|
||||||
|
when(manager.getTopTenLists()).thenReturn(map);
|
||||||
|
assertTrue(asc.execute(user, "", List.of()));
|
||||||
|
verify(user).sendMessage("admin.stats.title");
|
||||||
|
verify(user, never()).sendMessage("admin.stats.no-data");
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,208 @@
|
|||||||
|
package world.bentobox.level.commands;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertFalse;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
import static org.mockito.ArgumentMatchers.any;
|
||||||
|
import static org.mockito.ArgumentMatchers.anyString;
|
||||||
|
import static org.mockito.Mockito.mock;
|
||||||
|
import static org.mockito.Mockito.verify;
|
||||||
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
import org.bukkit.Bukkit;
|
||||||
|
import org.bukkit.Server;
|
||||||
|
import org.bukkit.World;
|
||||||
|
import org.bukkit.entity.Player;
|
||||||
|
import org.bukkit.inventory.ItemFactory;
|
||||||
|
import org.bukkit.inventory.meta.ItemMeta;
|
||||||
|
import org.junit.After;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.mockito.Mock;
|
||||||
|
import org.powermock.api.mockito.PowerMockito;
|
||||||
|
import org.powermock.core.classloader.annotations.PrepareForTest;
|
||||||
|
import org.powermock.modules.junit4.PowerMockRunner;
|
||||||
|
import org.powermock.reflect.Whitebox;
|
||||||
|
|
||||||
|
import world.bentobox.bentobox.BentoBox;
|
||||||
|
import world.bentobox.bentobox.api.addons.GameModeAddon;
|
||||||
|
import world.bentobox.bentobox.api.commands.CompositeCommand;
|
||||||
|
import world.bentobox.bentobox.api.localization.TextVariables;
|
||||||
|
import world.bentobox.bentobox.api.user.User;
|
||||||
|
import world.bentobox.bentobox.database.objects.Island;
|
||||||
|
import world.bentobox.bentobox.managers.IslandWorldManager;
|
||||||
|
import world.bentobox.bentobox.managers.IslandsManager;
|
||||||
|
import world.bentobox.bentobox.managers.LocalesManager;
|
||||||
|
import world.bentobox.bentobox.managers.PlayersManager;
|
||||||
|
import world.bentobox.level.Level;
|
||||||
|
import world.bentobox.level.LevelsManager;
|
||||||
|
import world.bentobox.level.objects.TopTenData;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author tastybento
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
@RunWith(PowerMockRunner.class)
|
||||||
|
@PrepareForTest({ Bukkit.class, BentoBox.class })
|
||||||
|
public class AdminTopRemoveCommandTest {
|
||||||
|
|
||||||
|
@Mock
|
||||||
|
private CompositeCommand ic;
|
||||||
|
private UUID uuid;
|
||||||
|
@Mock
|
||||||
|
private User user;
|
||||||
|
@Mock
|
||||||
|
private IslandsManager im;
|
||||||
|
@Mock
|
||||||
|
private Island island;
|
||||||
|
@Mock
|
||||||
|
private Level addon;
|
||||||
|
@Mock
|
||||||
|
private World world;
|
||||||
|
@Mock
|
||||||
|
private IslandWorldManager iwm;
|
||||||
|
@Mock
|
||||||
|
private GameModeAddon gameModeAddon;
|
||||||
|
@Mock
|
||||||
|
private Player p;
|
||||||
|
@Mock
|
||||||
|
private LocalesManager lm;
|
||||||
|
@Mock
|
||||||
|
private PlayersManager pm;
|
||||||
|
|
||||||
|
private AdminTopRemoveCommand atrc;
|
||||||
|
@Mock
|
||||||
|
private TopTenData ttd;
|
||||||
|
@Mock
|
||||||
|
private LevelsManager manager;
|
||||||
|
@Mock
|
||||||
|
private Server server;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setUp() {
|
||||||
|
// Set up plugin
|
||||||
|
BentoBox plugin = mock(BentoBox.class);
|
||||||
|
Whitebox.setInternalState(BentoBox.class, "instance", plugin);
|
||||||
|
User.setPlugin(plugin);
|
||||||
|
|
||||||
|
// Addon
|
||||||
|
when(ic.getAddon()).thenReturn(addon);
|
||||||
|
when(ic.getPermissionPrefix()).thenReturn("bskyblock.");
|
||||||
|
when(ic.getLabel()).thenReturn("island");
|
||||||
|
when(ic.getTopLabel()).thenReturn("island");
|
||||||
|
when(ic.getWorld()).thenReturn(world);
|
||||||
|
when(ic.getTopLabel()).thenReturn("bsb");
|
||||||
|
|
||||||
|
// IWM friendly name
|
||||||
|
when(plugin.getIWM()).thenReturn(iwm);
|
||||||
|
when(iwm.getFriendlyName(any())).thenReturn("BSkyBlock");
|
||||||
|
|
||||||
|
// World
|
||||||
|
when(world.toString()).thenReturn("world");
|
||||||
|
when(world.getName()).thenReturn("BSkyBlock_world");
|
||||||
|
|
||||||
|
// Player manager
|
||||||
|
when(plugin.getPlayers()).thenReturn(pm);
|
||||||
|
when(pm.getUser(anyString())).thenReturn(user);
|
||||||
|
// topTen
|
||||||
|
when(addon.getManager()).thenReturn(manager);
|
||||||
|
// User
|
||||||
|
uuid = UUID.randomUUID();
|
||||||
|
when(user.getUniqueId()).thenReturn(uuid);
|
||||||
|
when(user.getTranslation(any())).thenAnswer(invocation -> invocation.getArgument(0, String.class));
|
||||||
|
// Island
|
||||||
|
when(island.getUniqueId()).thenReturn(uuid.toString());
|
||||||
|
when(island.getOwner()).thenReturn(uuid);
|
||||||
|
// Island Manager
|
||||||
|
when(plugin.getIslands()).thenReturn(im);
|
||||||
|
when(im.getIslands(any(), any(User.class))).thenReturn(Set.of(island));
|
||||||
|
when(im.getIslands(any(), any(UUID.class))).thenReturn(Set.of(island));
|
||||||
|
|
||||||
|
// Bukkit
|
||||||
|
PowerMockito.mockStatic(Bukkit.class);
|
||||||
|
when(Bukkit.getServer()).thenReturn(server);
|
||||||
|
// Mock item factory (for itemstacks)
|
||||||
|
ItemFactory itemFactory = mock(ItemFactory.class);
|
||||||
|
ItemMeta itemMeta = mock(ItemMeta.class);
|
||||||
|
when(itemFactory.getItemMeta(any())).thenReturn(itemMeta);
|
||||||
|
when(server.getItemFactory()).thenReturn(itemFactory);
|
||||||
|
when(Bukkit.getItemFactory()).thenReturn(itemFactory);
|
||||||
|
|
||||||
|
atrc = new AdminTopRemoveCommand(addon, ic);
|
||||||
|
}
|
||||||
|
|
||||||
|
@After
|
||||||
|
public void tearDown() {
|
||||||
|
User.clearUsers();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.commands.admin.AdminTopRemoveCommand#AdminTopRemoveCommand(world.bentobox.level.Level, world.bentobox.bentobox.api.commands.CompositeCommand)}.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testAdminTopRemoveCommand() {
|
||||||
|
assertEquals("remove", atrc.getLabel());
|
||||||
|
assertEquals("delete", atrc.getAliases().get(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.commands.admin.AdminTopRemoveCommand#setup()}.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testSetup() {
|
||||||
|
assertEquals("bskyblock.admin.top.remove", atrc.getPermission());
|
||||||
|
assertEquals("admin.top.remove.parameters", atrc.getParameters());
|
||||||
|
assertEquals("admin.top.remove.description", atrc.getDescription());
|
||||||
|
assertFalse(atrc.isOnlyPlayer());
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.commands.admin.AdminTopRemoveCommand#canExecute(world.bentobox.bentobox.api.user.User, java.lang.String, java.util.List)}.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testCanExecuteWrongArgs() {
|
||||||
|
assertFalse(atrc.canExecute(user, "delete", Collections.emptyList()));
|
||||||
|
verify(user).sendMessage("commands.help.header", TextVariables.LABEL, "BSkyBlock");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test method for {@link world.bentobox.level.commands.admin.AdminTopRemoveCommand#canExecute(world.bentobox.bentobox.api.user.User, java.lang.String, java.util.List)}.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testCanExecuteUnknown() {
|
||||||
|
when(pm.getUser(anyString())).thenReturn(null);
|
||||||
|
assertFalse(atrc.canExecute(user, "delete", Collections.singletonList("tastybento")));
|
||||||
|
verify(user).sendMessage("general.errors.unknown-player", TextVariables.NAME, "tastybento");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.commands.admin.AdminTopRemoveCommand#canExecute(world.bentobox.bentobox.api.user.User, java.lang.String, java.util.List)}.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testCanExecuteKnown() {
|
||||||
|
assertTrue(atrc.canExecute(user, "delete", Collections.singletonList("tastybento")));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test method for
|
||||||
|
* {@link world.bentobox.level.commands.admin.AdminTopRemoveCommand#execute(world.bentobox.bentobox.api.user.User, java.lang.String, java.util.List)}.
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testExecuteUserStringListOfString() {
|
||||||
|
testCanExecuteKnown();
|
||||||
|
assertTrue(atrc.execute(user, "delete", Collections.singletonList("tastybento")));
|
||||||
|
verify(manager).removeEntry(world, uuid.toString());
|
||||||
|
verify(user).sendMessage("general.success");
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -1,199 +0,0 @@
|
|||||||
package world.bentobox.level.commands.admin;
|
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
|
||||||
import static org.junit.Assert.assertFalse;
|
|
||||||
import static org.junit.Assert.assertTrue;
|
|
||||||
import static org.mockito.ArgumentMatchers.any;
|
|
||||||
import static org.mockito.ArgumentMatchers.anyString;
|
|
||||||
import static org.mockito.ArgumentMatchers.eq;
|
|
||||||
import static org.mockito.Mockito.mock;
|
|
||||||
import static org.mockito.Mockito.verify;
|
|
||||||
import static org.mockito.Mockito.when;
|
|
||||||
|
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.UUID;
|
|
||||||
|
|
||||||
import org.bukkit.Bukkit;
|
|
||||||
import org.bukkit.Server;
|
|
||||||
import org.bukkit.World;
|
|
||||||
import org.bukkit.entity.Player;
|
|
||||||
import org.bukkit.inventory.ItemFactory;
|
|
||||||
import org.bukkit.inventory.meta.ItemMeta;
|
|
||||||
import org.junit.After;
|
|
||||||
import org.junit.Before;
|
|
||||||
import org.junit.Test;
|
|
||||||
import org.junit.runner.RunWith;
|
|
||||||
import org.mockito.Mock;
|
|
||||||
import org.powermock.api.mockito.PowerMockito;
|
|
||||||
import org.powermock.core.classloader.annotations.PrepareForTest;
|
|
||||||
import org.powermock.modules.junit4.PowerMockRunner;
|
|
||||||
import org.powermock.reflect.Whitebox;
|
|
||||||
|
|
||||||
import world.bentobox.bentobox.BentoBox;
|
|
||||||
import world.bentobox.bentobox.api.addons.GameModeAddon;
|
|
||||||
import world.bentobox.bentobox.api.commands.CompositeCommand;
|
|
||||||
import world.bentobox.bentobox.api.localization.TextVariables;
|
|
||||||
import world.bentobox.bentobox.api.user.User;
|
|
||||||
import world.bentobox.bentobox.database.objects.Island;
|
|
||||||
import world.bentobox.bentobox.managers.IslandWorldManager;
|
|
||||||
import world.bentobox.bentobox.managers.IslandsManager;
|
|
||||||
import world.bentobox.bentobox.managers.LocalesManager;
|
|
||||||
import world.bentobox.bentobox.managers.PlayersManager;
|
|
||||||
import world.bentobox.level.Level;
|
|
||||||
import world.bentobox.level.LevelsManager;
|
|
||||||
import world.bentobox.level.commands.AdminTopRemoveCommand;
|
|
||||||
import world.bentobox.level.objects.TopTenData;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @author tastybento
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
@RunWith(PowerMockRunner.class)
|
|
||||||
@PrepareForTest({Bukkit.class, BentoBox.class})
|
|
||||||
public class AdminTopRemoveCommandTest {
|
|
||||||
|
|
||||||
@Mock
|
|
||||||
private CompositeCommand ic;
|
|
||||||
private UUID uuid;
|
|
||||||
@Mock
|
|
||||||
private User user;
|
|
||||||
@Mock
|
|
||||||
private IslandsManager im;
|
|
||||||
@Mock
|
|
||||||
private Island island;
|
|
||||||
@Mock
|
|
||||||
private Level addon;
|
|
||||||
@Mock
|
|
||||||
private World world;
|
|
||||||
@Mock
|
|
||||||
private IslandWorldManager iwm;
|
|
||||||
@Mock
|
|
||||||
private GameModeAddon gameModeAddon;
|
|
||||||
@Mock
|
|
||||||
private Player p;
|
|
||||||
@Mock
|
|
||||||
private LocalesManager lm;
|
|
||||||
@Mock
|
|
||||||
private PlayersManager pm;
|
|
||||||
|
|
||||||
private AdminTopRemoveCommand atrc;
|
|
||||||
@Mock
|
|
||||||
private TopTenData ttd;
|
|
||||||
@Mock
|
|
||||||
private LevelsManager manager;
|
|
||||||
@Mock
|
|
||||||
private Server server;
|
|
||||||
|
|
||||||
@Before
|
|
||||||
public void setUp() {
|
|
||||||
// Set up plugin
|
|
||||||
BentoBox plugin = mock(BentoBox.class);
|
|
||||||
Whitebox.setInternalState(BentoBox.class, "instance", plugin);
|
|
||||||
User.setPlugin(plugin);
|
|
||||||
// Addon
|
|
||||||
when(ic.getAddon()).thenReturn(addon);
|
|
||||||
when(ic.getPermissionPrefix()).thenReturn("bskyblock.");
|
|
||||||
when(ic.getLabel()).thenReturn("island");
|
|
||||||
when(ic.getTopLabel()).thenReturn("island");
|
|
||||||
when(ic.getWorld()).thenReturn(world);
|
|
||||||
when(ic.getTopLabel()).thenReturn("bsb");
|
|
||||||
|
|
||||||
|
|
||||||
// IWM friendly name
|
|
||||||
when(plugin.getIWM()).thenReturn(iwm);
|
|
||||||
when(iwm.getFriendlyName(any())).thenReturn("BSkyBlock");
|
|
||||||
|
|
||||||
// World
|
|
||||||
when(world.toString()).thenReturn("world");
|
|
||||||
when(world.getName()).thenReturn("BSkyBlock_world");
|
|
||||||
|
|
||||||
|
|
||||||
// Player manager
|
|
||||||
when(plugin.getPlayers()).thenReturn(pm);
|
|
||||||
when(pm.getUser(anyString())).thenReturn(user);
|
|
||||||
// topTen
|
|
||||||
when(addon.getManager()).thenReturn(manager);
|
|
||||||
// User
|
|
||||||
uuid = UUID.randomUUID();
|
|
||||||
when(user.getUniqueId()).thenReturn(uuid);
|
|
||||||
when(user.getTranslation(any())).thenAnswer(invocation -> invocation.getArgument(0, String.class));
|
|
||||||
|
|
||||||
// Bukkit
|
|
||||||
PowerMockito.mockStatic(Bukkit.class);
|
|
||||||
when(Bukkit.getServer()).thenReturn(server);
|
|
||||||
// Mock item factory (for itemstacks)
|
|
||||||
ItemFactory itemFactory = mock(ItemFactory.class);
|
|
||||||
ItemMeta itemMeta = mock(ItemMeta.class);
|
|
||||||
when(itemFactory.getItemMeta(any())).thenReturn(itemMeta);
|
|
||||||
when(server.getItemFactory()).thenReturn(itemFactory);
|
|
||||||
when(Bukkit.getItemFactory()).thenReturn(itemFactory);
|
|
||||||
|
|
||||||
|
|
||||||
atrc = new AdminTopRemoveCommand(addon, ic);
|
|
||||||
}
|
|
||||||
|
|
||||||
@After
|
|
||||||
public void tearDown() {
|
|
||||||
User.clearUsers();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test method for {@link world.bentobox.level.commands.admin.AdminTopRemoveCommand#AdminTopRemoveCommand(world.bentobox.level.Level, world.bentobox.bentobox.api.commands.CompositeCommand)}.
|
|
||||||
*/
|
|
||||||
@Test
|
|
||||||
public void testAdminTopRemoveCommand() {
|
|
||||||
assertEquals("remove", atrc.getLabel());
|
|
||||||
assertEquals("delete", atrc.getAliases().get(0));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test method for {@link world.bentobox.level.commands.admin.AdminTopRemoveCommand#setup()}.
|
|
||||||
*/
|
|
||||||
@Test
|
|
||||||
public void testSetup() {
|
|
||||||
assertEquals("bskyblock.admin.top.remove", atrc.getPermission());
|
|
||||||
assertEquals("admin.top.remove.parameters", atrc.getParameters());
|
|
||||||
assertEquals("admin.top.remove.description", atrc.getDescription());
|
|
||||||
assertFalse(atrc.isOnlyPlayer());
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test method for {@link world.bentobox.level.commands.admin.AdminTopRemoveCommand#canExecute(world.bentobox.bentobox.api.user.User, java.lang.String, java.util.List)}.
|
|
||||||
*/
|
|
||||||
@Test
|
|
||||||
public void testCanExecuteWrongArgs() {
|
|
||||||
assertFalse(atrc.canExecute(user, "delete", Collections.emptyList()));
|
|
||||||
verify(user).sendMessage("commands.help.header", TextVariables.LABEL, "BSkyBlock");
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test method for {@link world.bentobox.level.commands.admin.AdminTopRemoveCommand#canExecute(world.bentobox.bentobox.api.user.User, java.lang.String, java.util.List)}.
|
|
||||||
*/
|
|
||||||
@Test
|
|
||||||
public void testCanExecuteUnknown() {
|
|
||||||
when(pm.getUser(anyString())).thenReturn(null);
|
|
||||||
assertFalse(atrc.canExecute(user, "delete", Collections.singletonList("tastybento")));
|
|
||||||
verify(user).sendMessage("general.errors.unknown-player", TextVariables.NAME, "tastybento");
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test method for {@link world.bentobox.level.commands.admin.AdminTopRemoveCommand#canExecute(world.bentobox.bentobox.api.user.User, java.lang.String, java.util.List)}.
|
|
||||||
*/
|
|
||||||
@Test
|
|
||||||
public void testCanExecuteKnown() {
|
|
||||||
assertTrue(atrc.canExecute(user, "delete", Collections.singletonList("tastybento")));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test method for {@link world.bentobox.level.commands.admin.AdminTopRemoveCommand#execute(world.bentobox.bentobox.api.user.User, java.lang.String, java.util.List)}.
|
|
||||||
*/
|
|
||||||
@Test
|
|
||||||
public void testExecuteUserStringListOfString() {
|
|
||||||
testCanExecuteKnown();
|
|
||||||
assertTrue(atrc.execute(user, "delete", Collections.singletonList("tastybento")));
|
|
||||||
verify(manager).removeEntry(any(World.class), eq(uuid));
|
|
||||||
verify(user).sendMessage("general.success");
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user