Compare commits
	
		
			171 Commits
		
	
	
		
			1572096020
			...
			develop-bl
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 0565f5de9e | |||
| ff81c91661 | |||
| 1182cb2ed6 | |||
| 045f31d4ca | |||
| 263fd85df7 | |||
| d70b025502 | |||
| e14c87c2fb | |||
| 9433495a52 | |||
| 4955e94306 | |||
| 09abfefe33 | |||
| bd3546abc8 | |||
| 8a7a0453ce | |||
| 64d0d817c0 | |||
| 713561bf07 | |||
| 4be3e528b1 | |||
| 53fca580f3 | |||
| 20fb4bf9fb | |||
| c42d259909 | |||
| 5c82c8d6da | |||
| 5910847172 | |||
| 7c254707c1 | |||
| 9ee5f6e419 | |||
| e49c3b1987 | |||
| 5ca4c70a41 | |||
| 040cae6cd1 | |||
| 324defc4a8 | |||
| dc0003b91e | |||
| d4a3c798f8 | |||
| c88c2ab6aa | |||
| 32a20cd4c5 | |||
| d7cc141b94 | |||
| 16d7347fd0 | |||
| fdf3b5c73f | |||
| 74f17e1b6d | |||
| 0d18b81399 | |||
| 1fa5fdfeb7 | |||
| 3bec5f4cbd | |||
| d38871eac9 | |||
| 238df2feff | |||
| e752d7f73b | |||
| b0df982be3 | |||
| dc1b5957f6 | |||
| 4068eae5bb | |||
| b1e3e99cb8 | |||
| b3787983d5 | |||
| 43ef28499b | |||
| 6e1ef4fd7c | |||
| ec2d243b7b | |||
| ef6f34c2b2 | |||
| 977f4ff4ec | |||
| 337727b0f0 | |||
| 44dae51e1c | |||
| 035864631d | |||
| f3b884058e | |||
| 03d4f4e6d8 | |||
| 7422a89d98 | |||
| 3590a5d278 | |||
| 15ac47b314 | |||
| af644a71ee | |||
| 0ce69f207f | |||
| 76297bb3af | |||
| 1aad8f07c4 | |||
| f26f4ed56a | |||
| 831eacaf47 | |||
| c71a2567bd | |||
| 72e88ce491 | |||
| 66d84f4677 | |||
| 427aed9a7e | |||
| 0d1e6070ce | |||
| 220fb9e229 | |||
| 9acac488f2 | |||
| d71c0d768e | |||
| 9ef4c2e96b | |||
| 5d33d2aff7 | |||
| 3f1065fd3a | |||
| aa868deeca | |||
| b6c298cec3 | |||
| 8f5a96dc31 | |||
| 2824c1053b | |||
| ccf383cdb5 | |||
| fce9449b7e | |||
| 69e971f618 | |||
| b1f188dece | |||
| a4289d5ac9 | |||
| 1fef363c50 | |||
| 558e6f84f1 | |||
| bdbb8b5824 | |||
| 8093a4a644 | |||
| 50147a06e2 | |||
| a52476650e | |||
| 0e5e841527 | |||
| ea5279dd82 | |||
| 32cbbe6c51 | |||
| 9544c953a2 | |||
| 34df173940 | |||
| ca99e6cfef | |||
| b0414ae6ab | |||
| c28d34ab88 | |||
| 9bae26044a | |||
| d1b5d81fa7 | |||
| e37e410542 | |||
| 956d2717d8 | |||
| ef7232e687 | |||
| ff31215295 | |||
| a4a254ebbe | |||
| 71d9faa9f4 | |||
| 859733e3dd | |||
| d94bbb7417 | |||
| 153a968776 | |||
| 639d06b01d | |||
| fdbb525870 | |||
| fcc2abdc49 | |||
| a3729734cb | |||
| 90b623ea07 | |||
| 9f49f44075 | |||
| e9a8e83019 | |||
| 7c81286feb | |||
| e7cf3caae8 | |||
| 8742f5f631 | |||
| 2c0e264ece | |||
| 4592d53d22 | |||
| 6d0913fa0c | |||
| 092d33beb3 | |||
| 71d5d8303d | |||
| 49eeb646ea | |||
| ceca038b27 | |||
| 76ceb9ef79 | |||
| 219879974c | |||
| bd630ebb7a | |||
| c56f318f1c | |||
| 4d98d7aa75 | |||
| 619190d0ae | |||
| 06641c5d84 | |||
| 2a52177043 | |||
| fc067a2ae0 | |||
| 116a9c11a2 | |||
| 3f29ceb08f | |||
| a33ee357e8 | |||
| e36256d5be | |||
| 0e3a54a1b9 | |||
| 2e67b41b44 | |||
| e4ac8f7a63 | |||
| e89e9d2181 | |||
| 8faf0efd60 | |||
| 2f1aeb71ee | |||
| 6475a7b825 | |||
| 193d8d778f | |||
| 38da5b1d34 | |||
| 04e3ddb09f | |||
| 47db27a86e | |||
| f13534da3f | |||
| e45698c88a | |||
| 9197840873 | |||
| 63d8335b3a | |||
| 184617e9c3 | |||
| 696c4bc260 | |||
| 9004609c1b | |||
| ddedcea8ea | |||
| 318a30fe54 | |||
| 4808d22f6a | |||
| 6b2a323a9c | |||
| 694ca0efba | |||
| 0276763a8d | |||
| 8811328571 | |||
| b3c43f1763 | |||
| 86677c942f | |||
| 31581fc643 | |||
| a412f5c24c | |||
| eae979ee65 | |||
| 28b9b84e07 | |||
| a5cdb93f1b | 
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -118,3 +118,5 @@ run/ | ||||
| !gradle-wrapper.jar | ||||
| /gradlew | ||||
| /gradlew.bat | ||||
|  | ||||
| local.gradle | ||||
|   | ||||
							
								
								
									
										57
									
								
								build.gradle
									
									
									
									
									
								
							
							
						
						
									
										57
									
								
								build.gradle
									
									
									
									
									
								
							| @@ -1,12 +1,13 @@ | ||||
| plugins { | ||||
|     id 'com.github.johnrengelman.shadow' version '8.1.1' | ||||
|     id 'java' | ||||
|     id 'com.gradleup.shadow' version "8.3.5" | ||||
| } | ||||
|  | ||||
| group = 'eu.mhsl.craftattack' | ||||
| version = '1.0' | ||||
| allprojects { | ||||
|     group = 'de.mhsl.craftattack' | ||||
|     version = '1.0.0' | ||||
|  | ||||
| repositories { | ||||
|     repositories { | ||||
|         mavenCentral() | ||||
|         maven { | ||||
|             name = "papermc-repo" | ||||
| @@ -19,29 +20,17 @@ repositories { | ||||
|         maven { | ||||
|             url = uri("https://repo.opencollab.dev/main/") | ||||
|         } | ||||
| } | ||||
|  | ||||
| dependencies { | ||||
|     compileOnly 'io.papermc.paper:paper-api:1.21.1-R0.1-SNAPSHOT' | ||||
|     compileOnly 'org.geysermc.floodgate:api:2.2.2-SNAPSHOT' | ||||
|     implementation 'org.apache.httpcomponents:httpclient:4.5.14' | ||||
|     implementation 'com.sparkjava:spark-core:2.9.4' | ||||
|     implementation 'org.reflections:reflections:0.10.2' | ||||
| } | ||||
|  | ||||
| def targetJavaVersion = 21 | ||||
| java { | ||||
|     def javaVersion = JavaVersion.toVersion(targetJavaVersion) | ||||
|     sourceCompatibility = javaVersion | ||||
|     targetCompatibility = javaVersion | ||||
|     if (JavaVersion.current() < javaVersion) { | ||||
|         toolchain.languageVersion = JavaLanguageVersion.of(targetJavaVersion) | ||||
|     } | ||||
| } | ||||
|  | ||||
| tasks.withType(JavaCompile).configureEach { | ||||
|     if (targetJavaVersion >= 10 || JavaVersion.current().isJava10Compatible()) { | ||||
|         options.release = targetJavaVersion | ||||
| subprojects { | ||||
|     apply plugin: 'java' | ||||
|     apply plugin: 'com.gradleup.shadow' | ||||
|  | ||||
|     java { | ||||
|         toolchain { | ||||
|             languageVersion = JavaLanguageVersion.of(21) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -51,18 +40,14 @@ configurations { | ||||
|  | ||||
| shadowJar { | ||||
|     configurations = [project.configurations.shadowImplementation] | ||||
|     archiveClassifier.set('') | ||||
|  | ||||
|     relocate 'org.apache.httpcomponents', 'eu.mhsl.lib.shadow.httpclient' | ||||
|     relocate 'com.sparkjava', 'eu.mhsl.lib.shadow.spark-core' | ||||
|  | ||||
|     mergeServiceFiles() | ||||
| } | ||||
|  | ||||
| tasks.register('copyJarToServer', Exec) { | ||||
|     dependsOn shadowJar | ||||
|     mustRunAfter shadowJar | ||||
|  | ||||
|     commandLine 'scp', 'build/libs/spawn-1.0-all.jar', 'root@10.20.6.1:/home/minecraft/server/plugins' | ||||
| } | ||||
|  | ||||
| tasks.register('copyJarToTestServer', Exec) { | ||||
|     dependsOn shadowJar | ||||
|     mustRunAfter shadowJar | ||||
|  | ||||
|     commandLine 'cp', 'build/libs/spawn-1.0-all.jar', '/home/elias/Dokumente/mcTestServer/plugins/spawn-1.0-all.jar' | ||||
| if (file("local.gradle").exists()) { | ||||
|     apply from: "local.gradle" | ||||
| } | ||||
|   | ||||
							
								
								
									
										8
									
								
								common/build.gradle
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								common/build.gradle
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| dependencies { | ||||
|     implementation project(':core') | ||||
|  | ||||
|     compileOnly 'io.papermc.paper:paper-api:1.21.8-R0.1-SNAPSHOT' | ||||
|     compileOnly 'org.geysermc.floodgate:api:2.2.4-SNAPSHOT' | ||||
|     implementation 'org.apache.httpcomponents:httpclient:4.5.14' | ||||
|     implementation 'com.sparkjava:spark-core:2.9.4' | ||||
| } | ||||
| @@ -0,0 +1,33 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.api; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.config.Configuration; | ||||
| import org.apache.http.client.utils.URIBuilder; | ||||
| import org.bukkit.configuration.ConfigurationSection; | ||||
|  | ||||
| import java.net.URI; | ||||
| import java.net.URISyntaxException; | ||||
| import java.net.http.HttpRequest; | ||||
| import java.util.Objects; | ||||
|  | ||||
| public class CraftAttackApi { | ||||
|     private final static ConfigurationSection apiConfig = Objects.requireNonNull(Configuration.cfg.getConfigurationSection("api")); | ||||
|     public final static String basePath = apiConfig.getString("baseurl"); | ||||
|     public final static String apiSecret = apiConfig.getString("secret"); | ||||
|  | ||||
|     public static URI getBaseUri() { | ||||
|         Objects.requireNonNull(basePath); | ||||
|         try { | ||||
|             return new URI(basePath); | ||||
|         } catch(URISyntaxException e) { | ||||
|             throw new RuntimeException(e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public static void withAuthorizationSecret(URIBuilder builder) { | ||||
|         builder.addParameter("secret", apiSecret); | ||||
|     } | ||||
|  | ||||
|     public static void withAuthorizationHeader(HttpRequest.Builder builder) { | ||||
|         builder.header("Authorization", apiSecret); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,28 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.api; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.config.Configuration; | ||||
| import org.bukkit.configuration.ConfigurationSection; | ||||
|  | ||||
| import java.net.URI; | ||||
| import java.net.URISyntaxException; | ||||
| import java.net.http.HttpRequest; | ||||
| import java.util.Objects; | ||||
|  | ||||
| public class VaroApi { | ||||
|     private final static ConfigurationSection apiConfig = Objects.requireNonNull(Configuration.cfg.getConfigurationSection("varoApi")); | ||||
|     public final static String basePath = apiConfig.getString("endpoint"); | ||||
|     public final static String apiSecret = apiConfig.getString("auth"); | ||||
|  | ||||
|     public static URI getBaseUri() { | ||||
|         Objects.requireNonNull(basePath); | ||||
|         try { | ||||
|             return new URI(basePath); | ||||
|         } catch(URISyntaxException e) { | ||||
|             throw new RuntimeException(e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public static void authorizationHeader(HttpRequest.Builder builder) { | ||||
|         builder.header("Authorization", apiSecret); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,27 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.api.repositories; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.api.client.ReqResp; | ||||
| import eu.mhsl.craftattack.spawn.common.api.CraftAttackApi; | ||||
| import java.util.UUID; | ||||
|  | ||||
| public class CraftAttackReportRepository extends ReportRepository { | ||||
|     public CraftAttackReportRepository() { | ||||
|         super(CraftAttackApi.getBaseUri(), new RequestModifier(null, CraftAttackApi::withAuthorizationHeader)); | ||||
|     } | ||||
|  | ||||
|     public ReqResp<PlayerReports> queryReports(UUID player) { | ||||
|         return this.get( | ||||
|             "report", | ||||
|             (parameters) -> parameters.addParameter("uuid", player.toString()), | ||||
|             PlayerReports.class | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     public ReqResp<ReportUrl> createReport(ReportCreationInfo data) { | ||||
|         return this.post( | ||||
|             "report", | ||||
|             data, | ||||
|             ReportUrl.class | ||||
|         ); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,42 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.api.repositories; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.api.client.HttpRepository; | ||||
| import eu.mhsl.craftattack.spawn.core.api.client.RepositoryLoader; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
|  | ||||
| import java.net.URI; | ||||
| import java.util.List; | ||||
| import java.util.UUID; | ||||
|  | ||||
| @RepositoryLoader.Abstraction | ||||
| public abstract class ReportRepository extends HttpRepository { | ||||
|     public ReportRepository(URI basePath, RequestModifier... baseRequestModifier) { | ||||
|         super(basePath, baseRequestModifier); | ||||
|     } | ||||
|  | ||||
|     public record ReportCreationInfo(@NotNull UUID reporter, @Nullable UUID reported, String reason) { | ||||
|     } | ||||
|  | ||||
|     public record ReportUrl(@NotNull String url) { | ||||
|     } | ||||
|  | ||||
|     public record PlayerReports( | ||||
|         List<Report> from_self, | ||||
|         Object to_self | ||||
|     ) { | ||||
|         public record Report( | ||||
|             @Nullable Reporter reported, | ||||
|             @NotNull String subject, | ||||
|             boolean draft, | ||||
|             @NotNull String status, | ||||
|             @NotNull String url | ||||
|         ) { | ||||
|             public record Reporter( | ||||
|                 @NotNull String username, | ||||
|                 @NotNull String uuid | ||||
|             ) { | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,45 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.api.repositories; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.common.api.VaroApi; | ||||
| import eu.mhsl.craftattack.spawn.core.api.client.ReqResp; | ||||
| import org.apache.commons.lang3.NotImplementedException; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
|  | ||||
| import java.util.UUID; | ||||
|  | ||||
| public class VaroReportRepository extends ReportRepository { | ||||
|     public VaroReportRepository() { | ||||
|         super(VaroApi.getBaseUri(), new RequestModifier(null, VaroApi::authorizationHeader)); | ||||
|     } | ||||
|  | ||||
|     public ReqResp<PlayerReports> queryReports(UUID player) { | ||||
|         throw new NotImplementedException("Report querying is not supported in Varo!"); | ||||
|     } | ||||
|  | ||||
|     public ReqResp<ReportUrl> createReport(ReportCreationInfo data) { | ||||
|         return this.post( | ||||
|             "report", | ||||
|             data, | ||||
|             ReportUrl.class | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     public record StrikeCreationInfo( | ||||
|         @Nullable UUID reporter, // null for automatic creations | ||||
|         @NotNull UUID reported, | ||||
|         @NotNull String reason, | ||||
|         @Nullable String body, | ||||
|         @Nullable String notice, | ||||
|         @Nullable String statement, | ||||
|         int strike_reason_id // internal strike mapping | ||||
|     ) { | ||||
|     } | ||||
|     public ReqResp<Void> createStrike(StrikeCreationInfo data) { | ||||
|         return this.put( | ||||
|             "report", | ||||
|             data, | ||||
|             Void.class | ||||
|         ); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,21 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.gameplay.cordinateDisplay; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceListener; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.player.PlayerJoinEvent; | ||||
| import org.bukkit.event.player.PlayerMoveEvent; | ||||
|  | ||||
| public class CoordinateChangedListener extends ApplianceListener<CoordinateDisplay> { | ||||
|     @EventHandler | ||||
|     public void onJoin(PlayerJoinEvent event) { | ||||
|         this.getAppliance().updateEnabled(event.getPlayer()); | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
|     public void onMove(PlayerMoveEvent event) { | ||||
|         if(!this.getAppliance().isEnabled(event.getPlayer())) return; | ||||
|         boolean hasChangedOrientation = this.getAppliance().hasChangedDirection(event.getFrom(), event.getTo()); | ||||
|         if(!event.hasChangedBlock() && !hasChangedOrientation) return; | ||||
|         this.getAppliance().sendCoordinates(event.getPlayer()); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,93 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.gameplay.cordinateDisplay; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.DataSizeConverter; | ||||
| import eu.mhsl.craftattack.spawn.core.util.world.WorldUtils; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.Location; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
|  | ||||
| import java.util.*; | ||||
|  | ||||
| public class CoordinateDisplay extends Appliance { | ||||
|     Map<Player, CoordinateDisplaySetting.CoordinateDisplayConfiguration> enabledPlayers = new WeakHashMap<>(); | ||||
|  | ||||
|     @Override | ||||
|     public void onEnable() { | ||||
|         Settings.instance().declareSetting(CoordinateDisplaySetting.class); | ||||
|         Settings.instance().addChangeListener(CoordinateDisplaySetting.class, this::updateEnabled); | ||||
|     } | ||||
|  | ||||
|     public void updateEnabled(Player player) { | ||||
|         CoordinateDisplaySetting.CoordinateDisplayConfiguration configuration = Settings.instance().getSetting( | ||||
|             player, | ||||
|             Settings.Key.CoordinateDisplay, | ||||
|             CoordinateDisplaySetting.CoordinateDisplayConfiguration.class | ||||
|         ); | ||||
|         this.enabledPlayers.put(player, configuration); | ||||
|     } | ||||
|  | ||||
|     public boolean isEnabled(Player player) { | ||||
|         return Optional.ofNullable(this.enabledPlayers.get(player)) | ||||
|             .map(CoordinateDisplaySetting.CoordinateDisplayConfiguration::anyEnabled) | ||||
|             .orElse(false); | ||||
|     } | ||||
|  | ||||
|     public void sendCoordinates(Player player) { | ||||
|         CoordinateDisplaySetting.CoordinateDisplayConfiguration config = this.enabledPlayers.get(player); | ||||
|         List<Component> components = new ArrayList<>(); | ||||
|  | ||||
|         if (config.coordinates()) { | ||||
|             components.add(Component.text("\uD83C\uDF0E ", NamedTextColor.GOLD)); | ||||
|             components.add(Component.text(String.format( | ||||
|                 "%d %d %d", | ||||
|                 player.getLocation().getBlockX(), | ||||
|                 player.getLocation().getBlockY(), | ||||
|                 player.getLocation().getBlockZ() | ||||
|             ))); | ||||
|         } | ||||
|  | ||||
|         if (config.direction()) { | ||||
|             if (!components.isEmpty()) { | ||||
|                 components.add(Component.text(" | ", NamedTextColor.GRAY)); | ||||
|             } | ||||
|             components.add(Component.text("\uD83E\uDDED ", NamedTextColor.GOLD)); | ||||
|             components.add(Component.text(DataSizeConverter.getCardinalDirection(player.getLocation()))); | ||||
|         } | ||||
|  | ||||
|         if (config.time()) { | ||||
|             if (!components.isEmpty()) { | ||||
|                 components.add(Component.text(" | ", NamedTextColor.GRAY)); | ||||
|             } | ||||
|             components.add(Component.text("⏱ ", NamedTextColor.GOLD)); | ||||
|             components.add(Component.text(WorldUtils.getGameTime(player.getWorld()))); | ||||
|         } | ||||
|  | ||||
|         if (!components.isEmpty()) { | ||||
|             Component actionBar = Component.empty(); | ||||
|             for (Component component : components) { | ||||
|                 actionBar = actionBar.append(component); | ||||
|             } | ||||
|             player.sendActionBar(actionBar); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     public boolean hasChangedDirection(Location previous, Location next) { | ||||
|         return !Objects.equals( | ||||
|             DataSizeConverter.getCardinalDirection(previous), | ||||
|             DataSizeConverter.getCardinalDirection(next) | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected @NotNull List<Listener> listeners() { | ||||
|         return List.of( | ||||
|             new CoordinateChangedListener() | ||||
|         ); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,53 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.gameplay.cordinateDisplay; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.CategorizedSetting; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.SettingCategory; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.datatypes.MultiBoolSetting; | ||||
| import org.bukkit.Material; | ||||
|  | ||||
| public class CoordinateDisplaySetting extends MultiBoolSetting<CoordinateDisplaySetting.CoordinateDisplayConfiguration> implements CategorizedSetting { | ||||
|     public CoordinateDisplaySetting() { | ||||
|         super(Settings.Key.CoordinateDisplay); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public SettingCategory category() { | ||||
|         return SettingCategory.Gameplay; | ||||
|     } | ||||
|  | ||||
|     public record CoordinateDisplayConfiguration( | ||||
|         @DisplayName("Koordinaten") boolean coordinates, | ||||
|         @DisplayName("Richtung") boolean direction, | ||||
|         @DisplayName("Zeit") boolean time | ||||
|     ) { | ||||
|         public boolean anyEnabled() { | ||||
|             return this.coordinates || this.direction || this.time; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected String title() { | ||||
|         return "Koordinatenanzeige"; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected String description() { | ||||
|         return "Zeige deine aktuelle Position über der Hotbar an"; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected Material icon() { | ||||
|         return Material.RECOVERY_COMPASS; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected CoordinateDisplayConfiguration defaultValue() { | ||||
|         return new CoordinateDisplayConfiguration(false, false, false); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public Class<?> dataType() { | ||||
|         return CoordinateDisplayConfiguration.class; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,20 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.internal.debug; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.internal.debug.command.AppliancesCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.internal.debug.command.UserInfoCommand; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| public class Debug extends Appliance { | ||||
|     @Override | ||||
|     @NotNull | ||||
|     protected List<ApplianceCommand<?>> commands() { | ||||
|         return List.of( | ||||
|             new UserInfoCommand(), | ||||
|             new AppliancesCommand() | ||||
|         ); | ||||
|     } | ||||
| } | ||||
| @@ -1,9 +1,9 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.debug.command; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.internal.debug.command; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.Main; | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.appliances.debug.Debug; | ||||
| import eu.mhsl.craftattack.spawn.util.text.ComponentUtil; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.internal.debug.Debug; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.ComponentUtil; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.ComponentBuilder; | ||||
| import net.kyori.adventure.text.TextComponent; | ||||
| @@ -1,7 +1,7 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.debug.command; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.internal.debug.command; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.appliances.debug.Debug; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.internal.debug.Debug; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.event.ClickEvent; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| @@ -50,13 +50,13 @@ public class UserInfoCommand extends ApplianceCommand<Debug> { | ||||
|                 .appendNewline() | ||||
|                 .append( | ||||
|                     Component | ||||
|                         .text("Erster Besuch: " + formatUnixTimestamp(player.getFirstPlayed()), NamedTextColor.GRAY) | ||||
|                         .text("Erster Besuch: " + this.formatUnixTimestamp(player.getFirstPlayed()), NamedTextColor.GRAY) | ||||
|                         .clickEvent(ClickEvent.copyToClipboard(String.valueOf(player.getFirstPlayed()))) | ||||
|                 ) | ||||
|                 .appendNewline() | ||||
|                 .append( | ||||
|                     Component | ||||
|                         .text("Letzter Besuch: " + formatUnixTimestamp(player.getLastSeen()), NamedTextColor.GRAY) | ||||
|                         .text("Letzter Besuch: " + this.formatUnixTimestamp(player.getLastSeen()), NamedTextColor.GRAY) | ||||
|                         .clickEvent(ClickEvent.copyToClipboard(String.valueOf(player.getLastSeen()))) | ||||
|                 ) | ||||
|                 .appendNewline() | ||||
| @@ -1,6 +1,6 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.titleClear; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.internal.titleClear; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| @@ -0,0 +1,12 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.internal.titleClear; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceListener; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.player.PlayerJoinEvent; | ||||
|  | ||||
| class TitleClearListener extends ApplianceListener<TitleClear> { | ||||
|     @EventHandler | ||||
|     public void onPlayerJoin(PlayerJoinEvent event) { | ||||
|         this.getAppliance().clearTitle(event.getPlayer()); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,34 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.adminMarker; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.api.server.HttpServer; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.displayName.DisplayName; | ||||
| import net.kyori.adventure.text.format.TextColor; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.Color; | ||||
| import org.bukkit.OfflinePlayer; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
|  | ||||
| import java.util.UUID; | ||||
|  | ||||
| public class AdminMarker extends Appliance implements DisplayName.Colored { | ||||
|     public final static String adminPermission = "admin"; | ||||
|  | ||||
|     @Override | ||||
|     public @Nullable TextColor getNameColor(Player player) { | ||||
|         if(player.hasPermission(adminPermission)) | ||||
|             return TextColor.color(Color.AQUA.asRGB()); // TODO read permission from config | ||||
|         return TextColor.color(Color.WHITE.asRGB()); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void httpApi(HttpServer.ApiBuilder apiBuilder) { | ||||
|         apiBuilder.get("isAdmin", request -> { | ||||
|             OfflinePlayer player = Bukkit.getOfflinePlayer(UUID.fromString(request.queryParams("player"))); | ||||
|             Main.logger().info(String.format("Adminstatus requested for %s, response: %s", player.getUniqueId(), player.isOp())); | ||||
|             return player.isOp(); | ||||
|         }); | ||||
|     } | ||||
| } | ||||
| @@ -1,8 +1,8 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.chatMention; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.chatMention; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.Main; | ||||
| import eu.mhsl.craftattack.spawn.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import net.kyori.adventure.sound.Sound; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.OfflinePlayer; | ||||
| @@ -56,7 +56,7 @@ public class ChatMention extends Appliance { | ||||
|     @Override | ||||
|     public void onEnable() { | ||||
|         Settings.instance().declareSetting(ChatMentionSetting.class); | ||||
|         refreshPlayers(); | ||||
|         this.refreshPlayers(); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @@ -1,10 +1,10 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.chatMention; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.chatMention; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.Main; | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceListener; | ||||
| import eu.mhsl.craftattack.spawn.appliances.chatMessages.ChatMessages; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.util.text.ComponentUtil; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceListener; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.chatMessages.ChatMessages; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.ComponentUtil; | ||||
| import io.papermc.paper.event.player.AsyncChatDecorateEvent; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| @@ -15,7 +15,7 @@ import org.bukkit.event.player.PlayerJoinEvent; | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| 
 | ||||
| public class ChatMentionListener extends ApplianceListener<ChatMention> { | ||||
| class ChatMentionListener extends ApplianceListener<ChatMention> { | ||||
|     @SuppressWarnings("UnstableApiUsage") | ||||
|     @EventHandler | ||||
|     public void coloringEvent(AsyncChatDecorateEvent event) { | ||||
| @@ -30,11 +30,11 @@ public class ChatMentionListener extends ApplianceListener<ChatMention> { | ||||
|         Component result = words.stream() | ||||
|             .map(word -> { | ||||
|                 String wordWithoutAnnotation = word.replace("@", ""); | ||||
|                 boolean isPlayer = getAppliance().getPlayerNames().contains(wordWithoutAnnotation); | ||||
|                 boolean isPlayer = this.getAppliance().getPlayerNames().contains(wordWithoutAnnotation); | ||||
|                 if(isPlayer && config.applyMentions()) { | ||||
|                     mentioned.add(wordWithoutAnnotation); | ||||
|                     Component mention = Component.text( | ||||
|                         getAppliance().formatPlayer(wordWithoutAnnotation), | ||||
|                         this.getAppliance().formatPlayer(wordWithoutAnnotation), | ||||
|                         NamedTextColor.GOLD | ||||
|                     ); | ||||
|                     return chatMessages.addReportActions(mention, wordWithoutAnnotation); | ||||
| @@ -45,12 +45,12 @@ public class ChatMentionListener extends ApplianceListener<ChatMention> { | ||||
|             .reduce(ComponentUtil::appendWithSpace) | ||||
|             .orElseThrow(); | ||||
| 
 | ||||
|         getAppliance().notifyPlayers(mentioned); | ||||
|         this.getAppliance().notifyPlayers(mentioned); | ||||
|         event.result(result); | ||||
|     } | ||||
| 
 | ||||
|     @EventHandler | ||||
|     public void onJoin(PlayerJoinEvent event) { | ||||
|         getAppliance().refreshPlayers(); | ||||
|         this.getAppliance().refreshPlayers(); | ||||
|     } | ||||
| } | ||||
| @@ -1,9 +1,9 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.chatMention; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.chatMention; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.CategorizedSetting; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.SettingCategory; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.datatypes.MultiBoolSetting; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.CategorizedSetting; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.SettingCategory; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.datatypes.MultiBoolSetting; | ||||
| import org.bukkit.Material; | ||||
| 
 | ||||
| public class ChatMentionSetting extends MultiBoolSetting<ChatMentionSetting.ChatMentionConfig> implements CategorizedSetting { | ||||
| @@ -15,7 +15,8 @@ public class ChatMentionSetting extends MultiBoolSetting<ChatMentionSetting.Chat | ||||
|     public record ChatMentionConfig( | ||||
|         @DisplayName("Spielernamen hervorheben") boolean applyMentions, | ||||
|         @DisplayName("Benachrichtigungston") boolean notifyOnMention | ||||
|     ) {} | ||||
|     ) { | ||||
|     } | ||||
| 
 | ||||
|     public ChatMentionSetting() { | ||||
|         super(Settings.Key.ChatMentions); | ||||
| @@ -1,7 +1,7 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.chatMessages; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.chatMessages; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.event.ClickEvent; | ||||
| import net.kyori.adventure.text.event.HoverEvent; | ||||
| @@ -19,7 +19,7 @@ public class ChatMessages extends Appliance { | ||||
|     } | ||||
| 
 | ||||
|     public Component getReportablePlayerName(Player player) { | ||||
|         return addReportActions(player.displayName(), player.getName()); | ||||
|         return this.addReportActions(player.displayName(), player.getName()); | ||||
|     } | ||||
| 
 | ||||
|     public Component addReportActions(Component message, String username) { | ||||
| @@ -1,54 +1,59 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.chatMessages; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.chatMessages; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceListener; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.util.IteratorUtil; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceListener; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.core.util.IteratorUtil; | ||||
| import io.papermc.paper.event.player.AsyncChatEvent; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import net.kyori.adventure.text.format.TextColor; | ||||
| import org.bukkit.Color; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.EventPriority; | ||||
| import org.bukkit.event.entity.PlayerDeathEvent; | ||||
| import org.bukkit.event.player.PlayerJoinEvent; | ||||
| import org.bukkit.event.player.PlayerQuitEvent; | ||||
| 
 | ||||
| import java.util.Optional; | ||||
| 
 | ||||
| public class ChatMessagesListener extends ApplianceListener<ChatMessages> { | ||||
| class ChatMessagesListener extends ApplianceListener<ChatMessages> { | ||||
|     @EventHandler | ||||
|     public void onPlayerChatEvent(AsyncChatEvent event) { | ||||
|         event.renderer( | ||||
|             (source, sourceDisplayName, message, viewer) -> | ||||
|                 Component.text("") | ||||
|                     .append(getAppliance().getReportablePlayerName(source)) | ||||
|                     .append(this.getAppliance().getReportablePlayerName(source)) | ||||
|                     .append(Component.text(" > ").color(TextColor.color(Color.GRAY.asRGB()))) | ||||
|                     .append(message).color(TextColor.color(Color.SILVER.asRGB())) | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     @EventHandler | ||||
|     @EventHandler(priority = EventPriority.HIGH) | ||||
|     public void onPlayerJoin(PlayerJoinEvent event) { | ||||
|         boolean wasHidden = event.joinMessage() == null; | ||||
|         event.joinMessage(null); | ||||
|         if(wasHidden) return; | ||||
|         IteratorUtil.onlinePlayers(player -> { | ||||
|             if(!Settings.instance().getSetting(player, Settings.Key.ShowJoinAndLeaveMessages, Boolean.class)) return; | ||||
|             player.sendMessage( | ||||
|                 Component | ||||
|                     .text(">>> ").color(NamedTextColor.GREEN) | ||||
|                     .append(getAppliance().getReportablePlayerName(event.getPlayer())) | ||||
|                     .append(this.getAppliance().getReportablePlayerName(event.getPlayer())) | ||||
|             ); | ||||
|         }); | ||||
|     } | ||||
| 
 | ||||
|     @EventHandler | ||||
|     public void onPlayerLeave(PlayerQuitEvent event) { | ||||
|         boolean wasHidden = event.quitMessage() == null; | ||||
|         event.quitMessage(null); | ||||
|         if(wasHidden) return; | ||||
|         IteratorUtil.onlinePlayers(player -> { | ||||
|             if(!Settings.instance().getSetting(player, Settings.Key.ShowJoinAndLeaveMessages, Boolean.class)) return; | ||||
|             player.sendMessage( | ||||
|                 Component | ||||
|                     .text("<<< ").color(NamedTextColor.RED) | ||||
|                     .append(getAppliance().getReportablePlayerName(event.getPlayer())) | ||||
|                     .append(this.getAppliance().getReportablePlayerName(event.getPlayer())) | ||||
|             ); | ||||
|         }); | ||||
|     } | ||||
| @@ -1,9 +1,9 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.chatMessages; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.chatMessages; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.CategorizedSetting; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.SettingCategory; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.datatypes.BoolSetting; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.CategorizedSetting; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.SettingCategory; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.datatypes.BoolSetting; | ||||
| import org.bukkit.Material; | ||||
| 
 | ||||
| public class ShowJoinAndLeaveMessagesSetting extends BoolSetting implements CategorizedSetting { | ||||
| @@ -0,0 +1,91 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.displayName; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.util.server.Floodgate; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.ComponentUtil; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.ComponentBuilder; | ||||
| import net.kyori.adventure.text.TextComponent; | ||||
| import net.kyori.adventure.text.event.HoverEvent; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import net.kyori.adventure.text.format.TextColor; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.Objects; | ||||
| import java.util.logging.Level; | ||||
|  | ||||
| public class DisplayName extends Appliance { | ||||
|     public interface Prefixed { | ||||
|         @Nullable | ||||
|         Component getNamePrefix(Player player); | ||||
|     } | ||||
|  | ||||
|     public interface Colored { | ||||
|         @Nullable | ||||
|         TextColor getNameColor(Player player); | ||||
|     } | ||||
|  | ||||
|     public void update(Player player) { | ||||
|         List<Colored> coloring = Main.instance().getAppliances().stream() | ||||
|             .filter(appliance -> appliance instanceof Colored) | ||||
|             .map(appliance -> (Colored) appliance) | ||||
|             .toList(); | ||||
|  | ||||
|         if(coloring.size() > 1) throw new IllegalStateException( | ||||
|             "There are two or more appliances which provide coloring for player names. This is currently not supported!" | ||||
|         ); | ||||
|  | ||||
|         TextColor playerColor = coloring.isEmpty() | ||||
|             ? NamedTextColor.WHITE | ||||
|             : coloring.getFirst().getNameColor(player); | ||||
|  | ||||
|         List<Prefixed> prefixes = Main.instance().getAppliances().stream() | ||||
|             .filter(appliance -> appliance instanceof Prefixed) | ||||
|             .map(appliance -> (Prefixed) appliance) | ||||
|             .toList(); | ||||
|  | ||||
|         ComponentBuilder<TextComponent, TextComponent.Builder> playerName = Component.text(); | ||||
|         prefixes.stream() | ||||
|             .map(prefixed -> prefixed.getNamePrefix(player)) | ||||
|             .filter(Objects::nonNull) | ||||
|             .forEach(prefix -> playerName | ||||
|                 .append(prefix) | ||||
|                 .append(ComponentUtil.clearedSpace()) | ||||
|         ); | ||||
|  | ||||
|         if(Floodgate.isBedrock(player)) { | ||||
|             playerName | ||||
|                 .append( | ||||
|                     Component.text("\uD83C\uDFAE", NamedTextColor.GRAY) | ||||
|                         .hoverEvent(HoverEvent.showText(Component.text( | ||||
|                             String.format("%s spielt die Minecraft: Bedrock Edition", player.getName()) | ||||
|                         ))) | ||||
|                 ) | ||||
|                 .append(ComponentUtil.clearedSpace()); | ||||
|         } | ||||
|  | ||||
|         playerName.append(Component.text(player.getName(), playerColor)); | ||||
|  | ||||
|         this.setGlobal(player, playerName.build()); | ||||
|     } | ||||
|  | ||||
|     private void setGlobal(Player player, Component component) { | ||||
|         try { | ||||
|             player.customName(component); | ||||
|             player.displayName(component); | ||||
|             player.playerListName(component); | ||||
|         } catch(Exception e) { | ||||
|             Main.instance().getLogger().log(Level.SEVERE, e, e::getMessage); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected @NotNull List<Listener> listeners() { | ||||
|         return List.of(new DisplayNameUpdateListener()); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,13 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.displayName; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceListener; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.EventPriority; | ||||
| import org.bukkit.event.player.PlayerJoinEvent; | ||||
|  | ||||
| class DisplayNameUpdateListener extends ApplianceListener<DisplayName> { | ||||
|     @EventHandler(priority = EventPriority.LOW) | ||||
|     public void onJoin(PlayerJoinEvent event) { | ||||
|         this.getAppliance().update(event.getPlayer()); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,28 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.help; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.help.command.DiscordCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.help.command.HelpCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.help.command.SpawnCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.help.command.TeamspeakCommand; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| public class Help extends Appliance { | ||||
|     public Help() { | ||||
|         super("help"); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     @NotNull | ||||
|     protected List<ApplianceCommand<?>> commands() { | ||||
|         return List.of( | ||||
|             new HelpCommand(), | ||||
|             new SpawnCommand(), | ||||
|             new TeamspeakCommand(), | ||||
|             new DiscordCommand() | ||||
|         ); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,27 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.help.command; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.help.Help; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.event.ClickEvent; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.command.Command; | ||||
| import org.bukkit.command.CommandSender; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
|  | ||||
| public class DiscordCommand extends ApplianceCommand<Help> { | ||||
|     public DiscordCommand() { | ||||
|         super("discord"); | ||||
|     } | ||||
|  | ||||
|     private final static String discordLink = "https://discord.gg/TXxspGVanq"; | ||||
|  | ||||
|     @Override | ||||
|     protected void execute(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) throws Exception { | ||||
|         sender.sendMessage( | ||||
|             Component.text("Offizieller Discord Server: ", NamedTextColor.GOLD) | ||||
|                 .append(Component.text(discordLink, NamedTextColor.AQUA)) | ||||
|                 .clickEvent(ClickEvent.openUrl(discordLink)) | ||||
|         ); | ||||
|     } | ||||
| } | ||||
| @@ -1,7 +1,7 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.help.command; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.help.command; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.appliances.help.Help; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.help.Help; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.command.Command; | ||||
| @@ -18,7 +18,8 @@ public class HelpCommand extends ApplianceCommand<Help> { | ||||
|         sender.sendMessage( | ||||
|             Component.text("Willkommen auf Craftattack!", NamedTextColor.GOLD) | ||||
|                 .appendNewline() | ||||
|                 .append(Component.text("Hier ist ein Hilfetext!", NamedTextColor.GRAY)) | ||||
|                 .append(Component.text("Wenn du hilfe benötigst kannst du dich jederzeit an einen Admin wenden." + | ||||
|                     " Weitere Informationen zu Funktionen und Befehlen erhältst du zudem im Turm am Spawn.", NamedTextColor.GRAY)) | ||||
|         ); | ||||
|     } | ||||
| } | ||||
| @@ -1,7 +1,7 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.help.command; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.help.command; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.appliances.help.Help; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.help.Help; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.command.Command; | ||||
| @@ -19,8 +19,8 @@ public class SpawnCommand extends ApplianceCommand<Help> { | ||||
| 
 | ||||
|     @Override | ||||
|     protected void execute(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) { | ||||
|         if(!getAppliance().localConfig().isString(spawnKey)) | ||||
|         if(!this.getAppliance().localConfig().isString(spawnKey)) | ||||
|             throw new ApplianceCommand.Error("Es wurde kein Spawnbereich hinterlegt!"); | ||||
|         sender.sendMessage(Component.text(Objects.requireNonNull(getAppliance().localConfig().getString(spawnKey)), NamedTextColor.GOLD)); | ||||
|         sender.sendMessage(Component.text(Objects.requireNonNull(this.getAppliance().localConfig().getString(spawnKey)), NamedTextColor.GOLD)); | ||||
|     } | ||||
| } | ||||
| @@ -1,7 +1,7 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.help.command; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.help.command; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.appliances.help.Help; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.help.Help; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.command.Command; | ||||
| @@ -17,12 +17,12 @@ public class TeamspeakCommand extends ApplianceCommand<Help> { | ||||
| 
 | ||||
|     @Override | ||||
|     protected void execute(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) { | ||||
|         if(!getAppliance().localConfig().isString(teamspeakKey)) | ||||
|         if(!this.getAppliance().localConfig().isString(teamspeakKey)) | ||||
|             throw new ApplianceCommand.Error("Es wurde kein Teamspeak hinterlegt!"); | ||||
|         sender.sendMessage( | ||||
|             Component.text() | ||||
|                 .append(Component.text("Joine unserem Teamspeak: ", NamedTextColor.GOLD)) | ||||
|                 .append(getTeamspeakIp(getAppliance().localConfig().getString(teamspeakKey))) | ||||
|                 .append(this.getTeamspeakIp(this.getAppliance().localConfig().getString(teamspeakKey))) | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
| @@ -1,6 +1,6 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.infoBars; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.infoBars; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import net.kyori.adventure.bossbar.BossBar; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.util.Ticks; | ||||
| @@ -13,6 +13,7 @@ import java.time.temporal.ChronoUnit; | ||||
| public abstract class Bar { | ||||
|     private BossBar bossBar; | ||||
|     private final BukkitTask updateTask; | ||||
|     public static String name; | ||||
| 
 | ||||
|     public Bar() { | ||||
|         long refreshRateInTicks = this.refresh().get(ChronoUnit.SECONDS) * Ticks.TICKS_PER_SECOND; | ||||
| @@ -32,7 +33,7 @@ public abstract class Bar { | ||||
|     private BossBar createBar() { | ||||
|         return BossBar.bossBar( | ||||
|             this.title(), | ||||
|             this.correctedProgress(), | ||||
|             this.clampedProgress(), | ||||
|             this.color(), | ||||
|             this.overlay() | ||||
|         ); | ||||
| @@ -43,7 +44,7 @@ public abstract class Bar { | ||||
| 
 | ||||
|         this.beforeRefresh(); | ||||
|         this.bossBar.name(this.title()); | ||||
|         this.bossBar.progress(this.correctedProgress()); | ||||
|         this.bossBar.progress(this.clampedProgress()); | ||||
|         this.bossBar.color(this.color()); | ||||
|         this.bossBar.overlay(this.overlay()); | ||||
|     } | ||||
| @@ -52,11 +53,13 @@ public abstract class Bar { | ||||
|         this.updateTask.cancel(); | ||||
|     } | ||||
| 
 | ||||
|     private float correctedProgress() { | ||||
|     private float clampedProgress() { | ||||
|         return Math.clamp(this.progress(), 0, 1); | ||||
|     } | ||||
| 
 | ||||
|     protected void beforeRefresh() {} | ||||
|     protected void beforeRefresh() { | ||||
|     } | ||||
| 
 | ||||
|     protected abstract Duration refresh(); | ||||
|     protected abstract String name(); | ||||
| 
 | ||||
| @@ -0,0 +1,51 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.infoBars; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.CategorizedSetting; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.SettingCategory; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.datatypes.MultiBoolSetting; | ||||
| import org.bukkit.Material; | ||||
|  | ||||
| public class InfoBarSetting extends MultiBoolSetting<InfoBarSetting.InfoBarConfiguration> implements CategorizedSetting { | ||||
|     public InfoBarSetting() { | ||||
|         super(Settings.Key.InfoBars); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public SettingCategory category() { | ||||
|         return SettingCategory.Misc; | ||||
|     } | ||||
|  | ||||
|     public record InfoBarConfiguration( | ||||
|         @DisplayName("Millisekunden pro Tick") boolean mspt, | ||||
|         @DisplayName("Spieler online") boolean playerCounter, | ||||
|         @DisplayName("Ticks pro Sekunde") boolean tps | ||||
|     ) {} | ||||
|  | ||||
|     @Override | ||||
|     protected String title() { | ||||
|         return "Informationsleisten"; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected String description() { | ||||
|         return "Wähle anzuzeigende Informationsleisten aus"; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected Material icon() { | ||||
|         return Material.COMMAND_BLOCK; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected InfoBarConfiguration defaultValue() { | ||||
|         return new InfoBarConfiguration(false, false, false); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public Class<?> dataType() { | ||||
|         return InfoBarConfiguration.class; | ||||
|     } | ||||
|  | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,100 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.infoBars; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.infoBars.bars.MsptBar; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.infoBars.bars.PlayerCounterBar; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.infoBars.bars.TpsBar; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.NamespacedKey; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.bukkit.persistence.PersistentDataContainer; | ||||
| import org.bukkit.persistence.PersistentDataType; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
|  | ||||
| public class InfoBars extends Appliance { | ||||
|     private final NamespacedKey infoBarKey = new NamespacedKey(Main.instance(), "infobars"); | ||||
|     private final List<Bar> infoBars = List.of( | ||||
|         new TpsBar(), | ||||
|         new MsptBar(), | ||||
|         new PlayerCounterBar() | ||||
|     ); | ||||
|  | ||||
|     public void showAllEnabled(Player player) { | ||||
|         this.getEnabledBars(player).forEach(bar -> this.show(player, bar)); | ||||
|     } | ||||
|  | ||||
|     public void hideAllEnabled(Player player) { | ||||
|         this.getEnabledBars(player).forEach(bar -> this.hide(player, bar)); | ||||
|         this.setEnabledBars(player, List.of()); | ||||
|     } | ||||
|  | ||||
|     public void show(Player player, String bar) { | ||||
|         this.validateBarName(bar); | ||||
|         List<String> existingBars = new ArrayList<>(this.getEnabledBars(player)); | ||||
|         existingBars.add(bar); | ||||
|         player.showBossBar(this.getBarByName(bar).getBossBar()); | ||||
|         this.setEnabledBars(player, existingBars); | ||||
|     } | ||||
|  | ||||
|     public void hide(Player player, String bar) { | ||||
|         this.validateBarName(bar); | ||||
|         List<String> existingBars = new ArrayList<>(this.getEnabledBars(player)); | ||||
|         existingBars.remove(bar); | ||||
|         player.hideBossBar(this.getBarByName(bar).getBossBar()); | ||||
|         this.setEnabledBars(player, existingBars); | ||||
|     } | ||||
|  | ||||
|     private List<String> getEnabledBars(Player player) { | ||||
|         PersistentDataContainer container = player.getPersistentDataContainer(); | ||||
|         if(!container.has(this.infoBarKey)) return List.of(); | ||||
|         return container.get(this.infoBarKey, PersistentDataType.LIST.strings()); | ||||
|     } | ||||
|  | ||||
|     private void setEnabledBars(Player player, List<String> bars) { | ||||
|         Bukkit.getScheduler().runTask( | ||||
|             Main.instance(), | ||||
|             () -> player.getPersistentDataContainer().set(this.infoBarKey, PersistentDataType.LIST.strings(), bars) | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     private Bar getBarByName(String name) { | ||||
|         return this.infoBars.stream() | ||||
|             .filter(bar -> bar.name().equalsIgnoreCase(name)) | ||||
|             .findFirst() | ||||
|             .orElse(null); | ||||
|     } | ||||
|  | ||||
|     private void validateBarName(String name) { | ||||
|         if(this.getBarByName(name) == null) | ||||
|             throw new ApplianceCommand.Error(String.format("Ungültiger infobar name '%s'", name)); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onEnable() { | ||||
|         Settings.instance().declareSetting(InfoBarSetting.class); | ||||
|         Settings.instance().addChangeListener(InfoBarSetting.class, player -> { | ||||
|             this.hideAllEnabled(player); | ||||
|             InfoBarSetting.InfoBarConfiguration config = Settings.instance().getSetting(player, Settings.Key.InfoBars, InfoBarSetting.InfoBarConfiguration.class); | ||||
|             if(config.mspt()) this.show(player, MsptBar.name); | ||||
|             if(config.playerCounter()) this.show(player, PlayerCounterBar.name); | ||||
|             if(config.tps()) this.show(player, TpsBar.name); | ||||
|         }); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public void onDisable() { | ||||
|         this.infoBars.forEach(Bar::stopUpdate); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected @NotNull List<Listener> listeners() { | ||||
|         return List.of(new ShowPreviousBarsListener()); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,12 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.infoBars; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceListener; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.player.PlayerJoinEvent; | ||||
|  | ||||
| class ShowPreviousBarsListener extends ApplianceListener<InfoBars> { | ||||
|     @EventHandler | ||||
|     public void onJoin(PlayerJoinEvent event) { | ||||
|         this.getAppliance().showAllEnabled(event.getPlayer()); | ||||
|     } | ||||
| } | ||||
| @@ -1,8 +1,8 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.infoBars.bars; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.infoBars.bars; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliances.infoBars.Bar; | ||||
| import eu.mhsl.craftattack.spawn.util.statistics.ServerMonitor; | ||||
| import eu.mhsl.craftattack.spawn.util.text.ColorUtil; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.infoBars.Bar; | ||||
| import eu.mhsl.craftattack.spawn.core.util.statistics.ServerMonitor; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.ColorUtil; | ||||
| import net.kyori.adventure.bossbar.BossBar; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| @@ -10,6 +10,8 @@ import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import java.time.Duration; | ||||
| 
 | ||||
| public class MsptBar extends Bar { | ||||
|     public static String name = "msptd"; | ||||
| 
 | ||||
|     @Override | ||||
|     protected Duration refresh() { | ||||
|         return Duration.ofSeconds(3); | ||||
| @@ -17,7 +19,7 @@ public class MsptBar extends Bar { | ||||
| 
 | ||||
|     @Override | ||||
|     protected String name() { | ||||
|         return "mspt"; | ||||
|         return name; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @@ -25,10 +27,10 @@ public class MsptBar extends Bar { | ||||
|         return Component.text() | ||||
|             .append(Component.text("M")) | ||||
|             .append(Component.text("illi", NamedTextColor.GRAY)) | ||||
|             .append(Component.text("S")) | ||||
|             .append(Component.text("econds ", NamedTextColor.GRAY)) | ||||
|             .append(Component.text("P")) | ||||
|             .append(Component.text("er ", NamedTextColor.GRAY)) | ||||
|             .append(Component.text("s")) | ||||
|             .append(Component.text("ekunden ", NamedTextColor.GRAY)) | ||||
|             .append(Component.text("p")) | ||||
|             .append(Component.text("ro ", NamedTextColor.GRAY)) | ||||
|             .append(Component.text("T")) | ||||
|             .append(Component.text("ick", NamedTextColor.GRAY)) | ||||
|             .append(Component.text(": ")) | ||||
| @@ -43,7 +45,7 @@ public class MsptBar extends Bar { | ||||
| 
 | ||||
|     @Override | ||||
|     protected BossBar.Color color() { | ||||
|         return BossBar.Color.BLUE; | ||||
|         return this.currentMSPT() <= 50 ? BossBar.Color.GREEN : BossBar.Color.RED; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @@ -1,9 +1,9 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.infoBars.bars; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.infoBars.bars; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.Main; | ||||
| import eu.mhsl.craftattack.spawn.appliances.infoBars.Bar; | ||||
| import eu.mhsl.craftattack.spawn.appliances.playerlimit.PlayerLimit; | ||||
| import eu.mhsl.craftattack.spawn.util.text.ColorUtil; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.infoBars.Bar; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.tooling.playerlimit.PlayerLimit; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.ColorUtil; | ||||
| import net.kyori.adventure.bossbar.BossBar; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.TextColor; | ||||
| @@ -12,6 +12,8 @@ import org.bukkit.Bukkit; | ||||
| import java.time.Duration; | ||||
| 
 | ||||
| public class PlayerCounterBar extends Bar { | ||||
|     public static String name = "playerCounter"; | ||||
| 
 | ||||
|     @Override | ||||
|     protected Duration refresh() { | ||||
|         return Duration.ofSeconds(3); | ||||
| @@ -19,7 +21,7 @@ public class PlayerCounterBar extends Bar { | ||||
| 
 | ||||
|     @Override | ||||
|     protected String name() { | ||||
|         return "playerCounter"; | ||||
|         return name; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @@ -38,7 +40,10 @@ public class PlayerCounterBar extends Bar { | ||||
| 
 | ||||
|     @Override | ||||
|     protected BossBar.Color color() { | ||||
|         return BossBar.Color.BLUE; | ||||
|         int freeSlots = this.getMaxPlayerCount() - this.getCurrentPlayerCount(); | ||||
|         return freeSlots <= 0 | ||||
|             ? BossBar.Color.RED | ||||
|             : freeSlots < 5 ? BossBar.Color.YELLOW : BossBar.Color.GREEN; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @@ -1,7 +1,7 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.infoBars.bars; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.infoBars.bars; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliances.infoBars.Bar; | ||||
| import eu.mhsl.craftattack.spawn.util.text.ColorUtil; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.infoBars.Bar; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.ColorUtil; | ||||
| import net.kyori.adventure.bossbar.BossBar; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| @@ -10,6 +10,8 @@ import org.bukkit.Bukkit; | ||||
| import java.time.Duration; | ||||
| 
 | ||||
| public class TpsBar extends Bar { | ||||
|     public static String name = "tps"; | ||||
| 
 | ||||
|     @Override | ||||
|     protected Duration refresh() { | ||||
|         return Duration.ofSeconds(3); | ||||
| @@ -17,7 +19,7 @@ public class TpsBar extends Bar { | ||||
| 
 | ||||
|     @Override | ||||
|     protected String name() { | ||||
|         return "tps"; | ||||
|         return name; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @@ -25,10 +27,10 @@ public class TpsBar extends Bar { | ||||
|         return Component.text() | ||||
|             .append(Component.text("T")) | ||||
|             .append(Component.text("icks ", NamedTextColor.GRAY)) | ||||
|             .append(Component.text("P")) | ||||
|             .append(Component.text("er ", NamedTextColor.GRAY)) | ||||
|             .append(Component.text("p")) | ||||
|             .append(Component.text("ro ", NamedTextColor.GRAY)) | ||||
|             .append(Component.text("S")) | ||||
|             .append(Component.text("econds", NamedTextColor.GRAY)) | ||||
|             .append(Component.text("ekunde", NamedTextColor.GRAY)) | ||||
|             .append(Component.text(": ")) | ||||
|             .append(Component.text(String.format("%.2f", this.currentTps()), ColorUtil.tpsColor(this.currentTps()))) | ||||
|             .build(); | ||||
| @@ -41,7 +43,9 @@ public class TpsBar extends Bar { | ||||
| 
 | ||||
|     @Override | ||||
|     protected BossBar.Color color() { | ||||
|         return BossBar.Color.BLUE; | ||||
|         return this.currentTps() >= 18 | ||||
|             ? BossBar.Color.GREEN | ||||
|             : this.currentTps() >= 15 ? BossBar.Color.YELLOW : BossBar.Color.RED; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @@ -1,11 +1,11 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.privateMessage; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.privateMessage; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.Main; | ||||
| import eu.mhsl.craftattack.spawn.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.appliances.chatMessages.ChatMessages; | ||||
| import eu.mhsl.craftattack.spawn.appliances.privateMessage.commands.PrivateMessageCommand; | ||||
| import eu.mhsl.craftattack.spawn.appliances.privateMessage.commands.PrivateReplyCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.chatMessages.ChatMessages; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.privateMessage.commands.PrivateMessageCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.privateMessage.commands.PrivateReplyCommand; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.ComponentBuilder; | ||||
| import net.kyori.adventure.text.TextComponent; | ||||
| @@ -23,7 +23,9 @@ public class PrivateMessage extends Appliance { | ||||
|     public final int targetChangeTimeoutSeconds = 30; | ||||
|     public final int conversationTimeoutMinutes = 30; | ||||
| 
 | ||||
|     private record Conversation(UUID target, Long lastSet) {} | ||||
|     private record Conversation(UUID target, Long lastSet) { | ||||
|     } | ||||
| 
 | ||||
|     private final Map<Player, List<Conversation>> replyMapping = new WeakHashMap<>(); | ||||
| 
 | ||||
|     public void reply(Player sender, String message) { | ||||
| @@ -32,7 +34,7 @@ public class PrivateMessage extends Appliance { | ||||
|         List<Conversation> replyList = this.replyMapping.get(sender); | ||||
| 
 | ||||
|         List<Conversation> tooOldConversations = replyList.stream() | ||||
|             .filter(conversation -> conversation.lastSet < System.currentTimeMillis() - (conversationTimeoutMinutes*60*1000)) | ||||
|             .filter(conversation -> conversation.lastSet < System.currentTimeMillis() - (this.conversationTimeoutMinutes * 60 * 1000)) | ||||
|             .toList(); | ||||
|         replyList.removeAll(tooOldConversations); | ||||
| 
 | ||||
| @@ -60,12 +62,13 @@ public class PrivateMessage extends Appliance { | ||||
|         } | ||||
| 
 | ||||
|         List<Conversation> oldConversations = replyList.stream() | ||||
|             .filter(conversation -> conversation.lastSet < System.currentTimeMillis() - (targetChangeTimeoutSeconds*1000)) | ||||
|             .filter(conversation -> conversation.lastSet < System.currentTimeMillis() - (this.targetChangeTimeoutSeconds * 1000)) | ||||
|             .toList(); | ||||
| 
 | ||||
|         if(oldConversations.contains(youngestEntry) || replyList.size() == 1) { | ||||
|             Player target = Bukkit.getPlayer(youngestEntry.target()); | ||||
|             if(target == null) throw new ApplianceCommand.Error("Der Spieler " + Bukkit.getOfflinePlayer(youngestEntry.target()).getName() + " ist nicht mehr verfügbar."); | ||||
|             if(target == null) | ||||
|                 throw new ApplianceCommand.Error("Der Spieler " + Bukkit.getOfflinePlayer(youngestEntry.target()).getName() + " ist nicht mehr verfügbar."); | ||||
| 
 | ||||
|             replyList.clear(); | ||||
|             this.sendWhisper(sender, new ResolvedPmUserArguments(target, message)); | ||||
| @@ -167,12 +170,16 @@ public class PrivateMessage extends Appliance { | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     public record ResolvedPmUserArguments(Player receiver, String message) {} | ||||
|     public record ResolvedPmUserArguments(Player receiver, String message) { | ||||
|     } | ||||
| 
 | ||||
|     public ResolvedPmUserArguments resolveImplicit(String[] args) { | ||||
|         if(args.length < 2) throw new ApplianceCommand.Error("Es muss ein Spieler sowie eine Nachricht angegeben werden."); | ||||
|         if(args.length < 2) | ||||
|             throw new ApplianceCommand.Error("Es muss ein Spieler sowie eine Nachricht angegeben werden."); | ||||
|         List<String> arguments = List.of(args); | ||||
|         Player targetPlayer = Bukkit.getPlayer(arguments.getFirst()); | ||||
|         if(targetPlayer == null) throw new ApplianceCommand.Error(String.format("Der Spieler %s konnte nicht gefunden werden.", arguments.getFirst())); | ||||
|         if(targetPlayer == null) | ||||
|             throw new ApplianceCommand.Error(String.format("Der Spieler %s konnte nicht gefunden werden.", arguments.getFirst())); | ||||
|         String message = arguments.stream().skip(1).collect(Collectors.joining(" ")); | ||||
|         return new ResolvedPmUserArguments(targetPlayer, message); | ||||
|     } | ||||
| @@ -1,7 +1,7 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.privateMessage.commands; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.privateMessage.commands; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.appliances.privateMessage.PrivateMessage; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.privateMessage.PrivateMessage; | ||||
| import org.bukkit.command.Command; | ||||
| import org.bukkit.command.CommandSender; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| @@ -13,6 +13,6 @@ public class PrivateMessageCommand extends ApplianceCommand.PlayerChecked<Privat | ||||
| 
 | ||||
|     @Override | ||||
|     protected void execute(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) throws Exception { | ||||
|         getAppliance().sendWhisper(getPlayer(), getAppliance().resolveImplicit(args)); | ||||
|         this.getAppliance().sendWhisper(this.getPlayer(), this.getAppliance().resolveImplicit(args)); | ||||
|     } | ||||
| } | ||||
| @@ -1,7 +1,7 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.privateMessage.commands; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.privateMessage.commands; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.appliances.privateMessage.PrivateMessage; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.privateMessage.PrivateMessage; | ||||
| import org.bukkit.command.Command; | ||||
| import org.bukkit.command.CommandSender; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| @@ -13,6 +13,6 @@ public class PrivateReplyCommand extends ApplianceCommand.PlayerChecked<PrivateM | ||||
| 
 | ||||
|     @Override | ||||
|     protected void execute(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) throws Exception { | ||||
|         getAppliance().reply(getPlayer(), String.join(" ", args)); | ||||
|         this.getAppliance().reply(this.getPlayer(), String.join(" ", args)); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,179 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.report; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.common.api.repositories.ReportRepository; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.api.client.ReqResp; | ||||
| import eu.mhsl.craftattack.spawn.common.api.repositories.CraftAttackReportRepository; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.ComponentBuilder; | ||||
| import net.kyori.adventure.text.TextComponent; | ||||
| import net.kyori.adventure.text.event.ClickEvent; | ||||
| import net.kyori.adventure.text.event.HoverEvent; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.OfflinePlayer; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.Optional; | ||||
|  | ||||
| public class Report extends Appliance { | ||||
|     public static Component helpText() { | ||||
|         return Component.text() | ||||
|             .appendNewline() | ||||
|             .append(Component.text(" Um einen Spieler zu melden, verwende ", NamedTextColor.GRAY)).appendNewline() | ||||
|             .append(Component.text("/report", NamedTextColor.GOLD)).appendNewline() | ||||
|             .append(Component.text("oder", NamedTextColor.GRAY)).appendNewline() | ||||
|             .append(Component.text("/report <spieler> [grund]", NamedTextColor.GOLD)).appendNewline() | ||||
|             .build(); | ||||
|     } | ||||
|  | ||||
|     public Report() { | ||||
|         super("report"); | ||||
|     } | ||||
|  | ||||
|     public void reportToUnknown(@NotNull Player issuer) { | ||||
|         CraftAttackReportRepository.ReportCreationInfo request = new CraftAttackReportRepository.ReportCreationInfo(issuer.getUniqueId(), null, ""); | ||||
|         Bukkit.getScheduler().runTaskAsynchronously( | ||||
|             Main.instance(), | ||||
|             () -> this.createReport(issuer, request) | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     public void reportToKnown(@NotNull Player issuer, @NotNull String targetUsername, @Nullable String reason) { | ||||
|         OfflinePlayer offlinePlayer = Bukkit.getOfflinePlayer(targetUsername); | ||||
|         if(issuer.getUniqueId().equals(offlinePlayer.getUniqueId())) { | ||||
|             issuer.sendMessage(Component.text("Du kannst dich nicht selbst reporten.", NamedTextColor.RED)); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         ReportRepository.ReportCreationInfo request = new ReportRepository.ReportCreationInfo( | ||||
|             issuer.getUniqueId(), | ||||
|             offlinePlayer.getUniqueId(), | ||||
|             Optional.ofNullable(reason).orElse("") | ||||
|         ); | ||||
|         Bukkit.getScheduler().runTaskAsynchronously( | ||||
|             Main.instance(), | ||||
|             () -> this.createReport(issuer, request) | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     private void createReport(Player issuer, ReportRepository.ReportCreationInfo reportRequest) { | ||||
|         ReqResp<ReportRepository.ReportUrl> createdReport = this.queryRepository(CraftAttackReportRepository.class) // TODO: Besser machen!! | ||||
|             .createReport(reportRequest); | ||||
|  | ||||
|         switch(createdReport.status()) { | ||||
|             case 200: // varo-endpoint specific | ||||
|             case 201: | ||||
|                 issuer.sendMessage( | ||||
|                     Component.text() | ||||
|                         .append(Component.text("\\/".repeat(20), NamedTextColor.DARK_GRAY)) | ||||
|                         .appendNewline() | ||||
|                         .append(Component.text("⚠ Der Report muss über den folgenden Link fertiggestellt werden:", NamedTextColor.GOLD)) | ||||
|                         .appendNewline() | ||||
|                         .appendNewline() | ||||
|                         .append( | ||||
|                             Component | ||||
|                                 .text(createdReport.data().url(), NamedTextColor.GRAY) // URL mit Weltkugel-Emoji | ||||
|                                 .clickEvent(ClickEvent.clickEvent(ClickEvent.Action.OPEN_URL, createdReport.data().url())) | ||||
|                         ) | ||||
|                         .appendNewline() | ||||
|                         .appendNewline() | ||||
|                         .append(Component.text("Ohne das Fertigstellen des Reports wird dieser nicht bearbeitet!", NamedTextColor.DARK_RED)) | ||||
|                         .appendNewline() | ||||
|                         .append(Component.text("/\\".repeat(20), NamedTextColor.DARK_GRAY)) | ||||
|                 ); | ||||
|                 break; | ||||
|  | ||||
|             case 400: | ||||
|                 issuer.sendMessage( | ||||
|                     Component.text() | ||||
|                         .append(Component.text("Der angegebene Nutzer ist in unserem System nicht bekannt.", NamedTextColor.RED)) | ||||
|                         .appendNewline() | ||||
|                         .append(Component.text("Bist du sicher, dass du den Namen richtig geschrieben hast?", NamedTextColor.RED)) | ||||
|                         .appendNewline() | ||||
|                         .append(Component.text("Du kannst dich alternativ jederzeit bei einem Admin melden.", NamedTextColor.GRAY)) | ||||
|                 ); | ||||
|                 break; | ||||
|  | ||||
|             case 401: | ||||
|             default: | ||||
|                 Main.logger().warning("Failed to request Report: " + createdReport.status()); | ||||
|                 issuer.sendMessage( | ||||
|                     Component.text() | ||||
|                         .append(Component.text("Interner Serverfehler beim anlegen des Reports.", NamedTextColor.RED)) | ||||
|                         .appendNewline() | ||||
|                         .append(Component.text("Bitte melde dich bei einem Admin!", NamedTextColor.RED)) | ||||
|                 ); | ||||
|                 break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     public void queryReports(Player issuer) { | ||||
|         ReqResp<ReportRepository.PlayerReports> userReports = this.queryRepository(CraftAttackReportRepository.class) // TODO: Besser machen!! | ||||
|             .queryReports(issuer.getUniqueId()); | ||||
|  | ||||
|         if(userReports.status() != 200) { | ||||
|             Main.logger().warning("Failed to request Reports: " + userReports.status()); | ||||
|             issuer.sendMessage( | ||||
|                 Component.text() | ||||
|                     .append(Component.text("Interner Serverfehler beim abfragen der Reports.", NamedTextColor.RED)) | ||||
|                     .appendNewline() | ||||
|                     .append(Component.text("Bitte melde dich bei einem Admin!", NamedTextColor.RED)) | ||||
|             ); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         List<ReportRepository.PlayerReports.Report> reports = userReports | ||||
|             .data() | ||||
|             .from_self() | ||||
|             .stream() | ||||
|             .filter(report -> !report.draft()) | ||||
|             .toList() | ||||
|             .reversed(); | ||||
|  | ||||
|         if(reports.isEmpty()) { | ||||
|             issuer.sendMessage( | ||||
|                 Component.text() | ||||
|                     .append(Component.text("Du hast noch niemanden reportet.", NamedTextColor.RED)) | ||||
|                     .appendNewline() | ||||
|                     .append(Component.text("Um jemanden zu melden, nutze /report", NamedTextColor.GRAY)) | ||||
|             ); | ||||
|             return; | ||||
|         } | ||||
|  | ||||
|         ComponentBuilder<TextComponent, TextComponent.Builder> component = Component.text() | ||||
|             .append(Component.newline()) | ||||
|             .append(Component.text("Von dir erstellte Reports: ", NamedTextColor.GOLD)) | ||||
|             .appendNewline(); | ||||
|  | ||||
|         reports.forEach(report -> { | ||||
|             component | ||||
|                 .append(Component.text(" - ", NamedTextColor.WHITE)) | ||||
|                 .append( | ||||
|                     report.reported() != null | ||||
|                         ? Component.text(report.reported().username(), NamedTextColor.WHITE) | ||||
|                         : Component.text("Unbekannt", NamedTextColor.YELLOW) | ||||
|                 ) | ||||
|                 .append(Component.text(String.format(": %s", report.subject()), NamedTextColor.GRAY)) | ||||
|                 .clickEvent(ClickEvent.openUrl(report.url())) | ||||
|                 .hoverEvent(HoverEvent.showText(Component.text("Klicke, um den Report einzusehen.", NamedTextColor.GOLD))); | ||||
|             component.appendNewline(); | ||||
|         }); | ||||
|  | ||||
|         issuer.sendMessage(component.build()); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     @NotNull | ||||
|     protected List<ApplianceCommand<?>> commands() { | ||||
|         return List.of( | ||||
|             new ReportCommand(), | ||||
|             new ReportsCommand() | ||||
|         ); | ||||
|     } | ||||
| } | ||||
| @@ -1,8 +1,7 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.report; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.report; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.ComponentUtil; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.OfflinePlayer; | ||||
| import org.bukkit.command.Command; | ||||
| @@ -17,31 +16,25 @@ import java.util.List; | ||||
| import java.util.stream.Collectors; | ||||
| import java.util.stream.Stream; | ||||
| 
 | ||||
| public class ReportCommand extends ApplianceCommand.PlayerChecked<Report> { | ||||
| class ReportCommand extends ApplianceCommand.PlayerChecked<Report> { | ||||
|     public ReportCommand() { | ||||
|         super("report"); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     protected void execute(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) { | ||||
|         sender.sendMessage( | ||||
|             Component.newline() | ||||
|                 .append(Component.text("Reportanfrage wird erstellt...", NamedTextColor.GREEN)) | ||||
|                 .appendNewline() | ||||
|                 .append(Component.text("Bitte warte einen Augenblick", NamedTextColor.GRAY)) | ||||
|                 .appendNewline() | ||||
|         ); | ||||
|         sender.sendMessage(ComponentUtil.pleaseWait()); | ||||
| 
 | ||||
|         if(args.length == 0) { | ||||
|             getAppliance().reportToUnknown(getPlayer()); | ||||
|             this.getAppliance().reportToUnknown(this.getPlayer()); | ||||
|         } | ||||
| 
 | ||||
|         if(args.length == 1) { | ||||
|             getAppliance().reportToKnown(getPlayer(), args[0], null); | ||||
|             this.getAppliance().reportToKnown(this.getPlayer(), args[0], null); | ||||
|         } | ||||
| 
 | ||||
|         if(args.length > 1) { | ||||
|             getAppliance().reportToKnown(getPlayer(), args[0], Arrays.stream(args).skip(1).collect(Collectors.joining(" "))); | ||||
|             this.getAppliance().reportToKnown(this.getPlayer(), args[0], Arrays.stream(args).skip(1).collect(Collectors.joining(" "))); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
| @@ -0,0 +1,24 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.report; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.ComponentUtil; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.command.Command; | ||||
| import org.bukkit.command.CommandSender; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
|  | ||||
| class ReportsCommand extends ApplianceCommand.PlayerChecked<Report> { | ||||
|     public ReportsCommand() { | ||||
|         super("reports"); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void execute(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) { | ||||
|         sender.sendMessage(ComponentUtil.pleaseWait()); | ||||
|         Bukkit.getScheduler().runTaskAsynchronously( | ||||
|             Main.instance(), | ||||
|             () -> this.getAppliance().queryReports(this.getPlayer()) | ||||
|         ); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,5 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings; | ||||
|  | ||||
| public interface CategorizedSetting { | ||||
|     SettingCategory category(); | ||||
| } | ||||
| @@ -0,0 +1,7 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings; | ||||
|  | ||||
| public enum SettingCategory { | ||||
|     Gameplay, | ||||
|     Visuals, | ||||
|     Misc, | ||||
| } | ||||
| @@ -1,10 +1,12 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.settings; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.datatypes.Setting; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.listeners.OpenSettingsShortcutListener; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.listeners.SettingsInventoryListener; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.datatypes.Setting; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.listeners.OpenSettingsShortcutListener; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.listeners.SettingsInventoryListener; | ||||
| import eu.mhsl.craftattack.spawn.core.util.world.InteractSounds; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.entity.Player; | ||||
| @@ -15,6 +17,7 @@ import org.jetbrains.annotations.NotNull; | ||||
| import java.lang.reflect.InvocationTargetException; | ||||
| import java.util.*; | ||||
| import java.util.concurrent.atomic.AtomicInteger; | ||||
| import java.util.function.Consumer; | ||||
| import java.util.stream.Collectors; | ||||
| 
 | ||||
| public class Settings extends Appliance { | ||||
| @@ -32,11 +35,16 @@ public class Settings extends Appliance { | ||||
|         ChatMentions, | ||||
|         DoubleDoors, | ||||
|         KnockDoors, | ||||
|         BorderWarning, | ||||
|         LocatorBar, | ||||
|         InfoBars, | ||||
|         CoordinateDisplay, | ||||
|         Bloodmoon | ||||
|     } | ||||
| 
 | ||||
|     public static Settings instance() { | ||||
|         if(settingsInstance != null) return settingsInstance; | ||||
|         Settings.settingsInstance = queryAppliance(Settings.class); | ||||
|         Settings.settingsInstance = Main.instance().getAppliance(Settings.class); | ||||
|         return settingsInstance; | ||||
|     } | ||||
| 
 | ||||
| @@ -55,22 +63,32 @@ public class Settings extends Appliance { | ||||
| 
 | ||||
|     private final WeakHashMap<Player, OpenSettingsInventory> openSettingsInventories = new WeakHashMap<>(); | ||||
|     private final WeakHashMap<Player, List<Setting<?>>> settingsCache = new WeakHashMap<>(); | ||||
|     protected final Map<Class<? extends Setting<?>>, Consumer<Player>> changeListeners = new WeakHashMap<>(); | ||||
| 
 | ||||
|     public <TDataType extends Setting<?>> void addChangeListener(Class<TDataType> setting, Consumer<Player> listener) { | ||||
|         this.changeListeners.merge(setting, listener, Consumer::andThen); | ||||
|     } | ||||
| 
 | ||||
|     public <TDataType extends Setting<?>> void invokeChangeListener(Player player, Class<TDataType> setting) { | ||||
|         Optional.ofNullable(this.changeListeners.get(setting)) | ||||
|             .ifPresent(listener -> listener.accept(player)); | ||||
|     } | ||||
| 
 | ||||
|     private List<Setting<?>> getSettings(Player player) { | ||||
|         if(settingsCache.containsKey(player)) return settingsCache.get(player); | ||||
|         if(this.settingsCache.containsKey(player)) return this.settingsCache.get(player); | ||||
| 
 | ||||
|         List<Setting<?>> settings = this.declaredSettings.stream() | ||||
|             .map(clazz -> { | ||||
|                 try { | ||||
|                     return clazz.getDeclaredConstructor(); | ||||
|                 } catch (NoSuchMethodException e) { | ||||
|                 } catch(NoSuchMethodException e) { | ||||
|                     throw new RuntimeException(String.format("Setting '%s' does not have an accessible constructor", clazz.getName()), e); | ||||
|                 } | ||||
|             }) | ||||
|             .map(constructor -> { | ||||
|                 try { | ||||
|                     return constructor.newInstance(); | ||||
|                 } catch (InstantiationException | IllegalAccessException | InvocationTargetException e) { | ||||
|                 } catch(InstantiationException | IllegalAccessException | InvocationTargetException e) { | ||||
|                     throw new RuntimeException(String.format("Failed to create instance of setting '%s'", constructor.getDeclaringClass().getName()), e); | ||||
|                 } | ||||
|             }) | ||||
| @@ -82,7 +100,7 @@ public class Settings extends Appliance { | ||||
|     } | ||||
| 
 | ||||
|     public <T> T getSetting(Player player, Key key, Class<T> clazz) { | ||||
|         Setting<?> setting = getSettings(player).stream() | ||||
|         Setting<?> setting = this.getSettings(player).stream() | ||||
|             .filter(s -> Objects.equals(s.getKey(), key)) | ||||
|             .findFirst() | ||||
|             .orElseThrow(); | ||||
| @@ -95,8 +113,8 @@ public class Settings extends Appliance { | ||||
|     } | ||||
| 
 | ||||
|     public void openSettings(Player player) { | ||||
|         List<Setting<?>> settings = getSettings(player); | ||||
|         Inventory inventory = Bukkit.createInventory(null, calculateInvSize(settings), Component.text("Einstellungen")); | ||||
|         List<Setting<?>> settings = this.getSettings(player); | ||||
|         Inventory inventory = Bukkit.createInventory(null, this.calculateInvSize(settings), Component.text("Einstellungen")); | ||||
| 
 | ||||
|         AtomicInteger row = new AtomicInteger(0); | ||||
|         Arrays.stream(SettingCategory.values()) | ||||
| @@ -106,11 +124,14 @@ public class Settings extends Appliance { | ||||
|                     .filter(setting -> ((CategorizedSetting) setting).category().equals(category)) | ||||
|                     .toList(); | ||||
| 
 | ||||
|                 for (int i = 0; i < categorizedSettings.size(); i++) { | ||||
|                 //skip empty category rows | ||||
|                 if(categorizedSettings.isEmpty()) return; | ||||
| 
 | ||||
|                 for(int i = 0; i < categorizedSettings.size(); i++) { | ||||
|                     int slot = row.get() * 9 + i % 9; | ||||
|                     inventory.setItem(slot, categorizedSettings.get(i).buildItem()); | ||||
| 
 | ||||
|                     if (i % 9 == 8) { | ||||
|                     if(i % 9 == 8) { | ||||
|                         row.incrementAndGet(); | ||||
|                     } | ||||
|                 } | ||||
| @@ -121,16 +142,17 @@ public class Settings extends Appliance { | ||||
|             .filter(setting -> !(setting instanceof CategorizedSetting)) | ||||
|             .toList(); | ||||
| 
 | ||||
|         for (int i = 0; i < uncategorizedSettings.size(); i++) { | ||||
|         for(int i = 0; i < uncategorizedSettings.size(); i++) { | ||||
|             int slot = row.get() * 9 + i % 9; | ||||
|             inventory.setItem(slot, uncategorizedSettings.get(i).buildItem()); | ||||
| 
 | ||||
|             if (i % 9 == 8) { | ||||
|             if(i % 9 == 8) { | ||||
|                 row.incrementAndGet(); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         player.openInventory(inventory); | ||||
|         InteractSounds.of(player).open(); | ||||
|         this.openSettingsInventories.put(player, new OpenSettingsInventory(inventory, settings)); | ||||
|     } | ||||
| 
 | ||||
| @@ -138,23 +160,30 @@ public class Settings extends Appliance { | ||||
|         int countOfUncategorized = (int) settings.stream() | ||||
|             .filter(setting -> !(setting instanceof CategorizedSetting)) | ||||
|             .count(); | ||||
|         int invSizeForUncategorized = (int) Math.ceil((double) countOfUncategorized / 9) * 9; | ||||
| 
 | ||||
|         return Arrays.stream(SettingCategory.values()) | ||||
|         int invSizeForCategorized = Arrays.stream(SettingCategory.values()) | ||||
|             .map(settingCategory -> settings.stream() | ||||
|                 .filter(setting -> setting instanceof CategorizedSetting) | ||||
|                 .map(setting -> (CategorizedSetting) setting) | ||||
|                 .filter(categorizedSetting -> categorizedSetting.category().equals(settingCategory)) | ||||
|                 .count()) | ||||
|             .map(itemCount -> itemCount + countOfUncategorized) | ||||
|             .map(itemCount -> (int) Math.ceil((double) itemCount / 9)) | ||||
|             .reduce(Integer::sum) | ||||
|             .orElse(1) * 9; | ||||
| 
 | ||||
|         int invSize = invSizeForUncategorized + invSizeForCategorized; | ||||
|         if(invSize % 9 != 0) throw new IllegalStateException( | ||||
|             String.format("Failed to calculate settings inventory size. %d is not an multiple of 9", invSize) | ||||
|         ); | ||||
|         return invSize; | ||||
|     } | ||||
| 
 | ||||
|     public void onSettingsClose(Player player) { | ||||
|         if(!openSettingsInventories.containsKey(player)) return; | ||||
|         openSettingsInventories.remove(player); | ||||
|         if(!this.openSettingsInventories.containsKey(player)) return; | ||||
|         this.openSettingsInventories.remove(player); | ||||
|         player.updateInventory(); | ||||
|         InteractSounds.of(player).close(); | ||||
|     } | ||||
| 
 | ||||
|     public boolean hasSettingsNotOpen(Player player) { | ||||
| @@ -162,7 +191,8 @@ public class Settings extends Appliance { | ||||
|     } | ||||
| 
 | ||||
|     public OpenSettingsInventory getOpenInventory(Player player) { | ||||
|         if(hasSettingsNotOpen(player)) throw new RuntimeException("Cannot retrieve data from closed Settings inventory!"); | ||||
|         if(this.hasSettingsNotOpen(player)) | ||||
|             throw new RuntimeException("Cannot retrieve data from closed Settings inventory!"); | ||||
|         return this.openSettingsInventories.get(player); | ||||
|     } | ||||
| 
 | ||||
| @@ -1,17 +1,17 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.settings; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import org.bukkit.command.Command; | ||||
| import org.bukkit.command.CommandSender; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| 
 | ||||
| public class SettingsCommand extends ApplianceCommand.PlayerChecked<Settings> { | ||||
| class SettingsCommand extends ApplianceCommand.PlayerChecked<Settings> { | ||||
|     public SettingsCommand() { | ||||
|         super("settings"); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     protected void execute(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) throws Exception { | ||||
|         getAppliance().openSettings(getPlayer()); | ||||
|         this.getAppliance().openSettings(this.getPlayer()); | ||||
|     } | ||||
| } | ||||
| @@ -1,6 +1,6 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.settings; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.datatypes.BoolSetting; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.datatypes.BoolSetting; | ||||
| import org.bukkit.Material; | ||||
| 
 | ||||
| public class SettingsShortcutSetting extends BoolSetting implements CategorizedSetting { | ||||
| @@ -1,4 +1,4 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.settings.datatypes; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.datatypes; | ||||
| 
 | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| @@ -17,14 +17,14 @@ public abstract class ActionSetting extends Setting<Void> { | ||||
| 
 | ||||
|     @Override | ||||
|     public ItemMeta buildMeta(ItemMeta meta) { | ||||
|         meta.displayName(Component.text(title(), NamedTextColor.WHITE)); | ||||
|         meta.lore(buildDescription(description())); | ||||
|         meta.displayName(Component.text(this.title(), NamedTextColor.WHITE)); | ||||
|         meta.lore(this.buildDescription(this.description())); | ||||
|         return meta; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     protected void change(Player player, ClickType clickType) { | ||||
|         onAction(player, clickType); | ||||
|         this.onAction(player, clickType); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @@ -33,10 +33,12 @@ public abstract class ActionSetting extends Setting<Void> { | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     protected void fromStorage(PersistentDataContainer container) {} | ||||
|     protected void fromStorage(PersistentDataContainer container) { | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     protected void toStorage(PersistentDataContainer container, Void value) {} | ||||
|     protected void toStorage(PersistentDataContainer container, Void value) { | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public Class<?> dataType() { | ||||
| @@ -1,6 +1,6 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.settings.datatypes; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.datatypes; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.entity.Player; | ||||
| @@ -20,19 +20,19 @@ public abstract class BoolSetting extends Setting<Boolean> { | ||||
| 
 | ||||
|     @Override | ||||
|     public void fromStorage(PersistentDataContainer container) { | ||||
|         this.state = container.has(getNamespacedKey()) | ||||
|             ? Objects.requireNonNull(container.get(getNamespacedKey(), PersistentDataType.BOOLEAN)) | ||||
|             : defaultValue(); | ||||
|         this.state = container.has(this.getNamespacedKey()) | ||||
|             ? Objects.requireNonNull(container.get(this.getNamespacedKey(), PersistentDataType.BOOLEAN)) | ||||
|             : this.defaultValue(); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     protected void toStorage(PersistentDataContainer container, Boolean value) { | ||||
|         container.set(getNamespacedKey(), PersistentDataType.BOOLEAN, value); | ||||
|         container.set(this.getNamespacedKey(), PersistentDataType.BOOLEAN, value); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public ItemMeta buildMeta(ItemMeta meta) { | ||||
|         meta.displayName(Component.text(title(), NamedTextColor.WHITE)); | ||||
|         meta.displayName(Component.text(this.title(), NamedTextColor.WHITE)); | ||||
|         List<Component> lore = new ArrayList<>(List.of( | ||||
|             Component.empty() | ||||
|                 .append(Component.text("Status: ", NamedTextColor.DARK_GRAY)) | ||||
| @@ -42,7 +42,7 @@ public abstract class BoolSetting extends Setting<Boolean> { | ||||
|                 ), | ||||
|             Component.empty() | ||||
|         )); | ||||
|         lore.addAll(buildDescription(description())); | ||||
|         lore.addAll(this.buildDescription(this.description())); | ||||
|         meta.lore(lore); | ||||
|         return meta; | ||||
|     } | ||||
| @@ -59,6 +59,6 @@ public abstract class BoolSetting extends Setting<Boolean> { | ||||
| 
 | ||||
|     @Override | ||||
|     public Boolean state() { | ||||
|         return state; | ||||
|         return this.state; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,103 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.datatypes; | ||||
|  | ||||
| import com.google.gson.Gson; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.event.inventory.ClickType; | ||||
| import org.bukkit.inventory.meta.ItemMeta; | ||||
| import org.bukkit.persistence.PersistentDataContainer; | ||||
| import org.bukkit.persistence.PersistentDataType; | ||||
|  | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| import java.util.Objects; | ||||
| import java.util.stream.IntStream; | ||||
| import java.util.stream.Stream; | ||||
|  | ||||
| public abstract class IntegerSetting extends Setting<Integer> { | ||||
|     private final List<Integer> options; | ||||
|  | ||||
|     public IntegerSetting(Settings.Key key, int minimum, int maximum) { | ||||
|         this(key, IntStream.range(minimum, maximum+1).boxed().toList()); | ||||
|     } | ||||
|  | ||||
|     public IntegerSetting(Settings.Key key, List<Integer> options) { | ||||
|         super(key); | ||||
|         this.options = options; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public ItemMeta buildMeta(ItemMeta meta) { | ||||
|         Component componentBefore = Component.text("  " + this.fillWithSpaces(this.options.getLast())); | ||||
|         Component componentAfter = Component.text("  " + this.fillWithSpaces(this.options.getFirst())); | ||||
|         int listIndex = this.options.indexOf(this.state); | ||||
|         if(listIndex > 0) componentBefore = Component.text("  " + this.fillWithSpaces(this.options.get(listIndex-1))); | ||||
|         if(listIndex < this.options.size()-1) componentAfter = Component.text("  " + this.fillWithSpaces(this.options.get(listIndex+1))); | ||||
|  | ||||
|         meta.displayName(Component.text(this.title(), NamedTextColor.WHITE)); | ||||
|         List<Component> lore = new ArrayList<>(Stream.of( | ||||
|             Component.empty() | ||||
|                 .append(Component.text("Wert: ", NamedTextColor.DARK_GRAY)), | ||||
|             Component.empty() | ||||
|                 .append(componentBefore.color(NamedTextColor.DARK_GRAY)) | ||||
|                 .append(Component.text("  " + this.fillWithSpaces(this.state), NamedTextColor.GREEN)) | ||||
|                 .append(componentAfter.color(NamedTextColor.DARK_GRAY)), | ||||
|             Component.empty() | ||||
|         ).toList()); | ||||
|         lore.addAll(this.buildDescription(this.description())); | ||||
|         meta.lore(lore); | ||||
|         return meta; | ||||
|     } | ||||
|  | ||||
|     private String fillWithSpaces(Integer option) { | ||||
|         String optionString = option.toString(); | ||||
|         int optionLength = optionString.length(); | ||||
|         int maxInteger = this.options.stream().mapToInt(value -> value).max().orElse(0); | ||||
|         int maxLength = String.valueOf(maxInteger).length(); | ||||
|         int padding = maxLength - optionLength; | ||||
|  | ||||
|         int padEnd = padding / 2; | ||||
|         int padStart = padding - padEnd; | ||||
|  | ||||
|         optionString = " ".repeat(padStart) + optionString + " ".repeat(padEnd); | ||||
|         return optionString; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void change(Player player, ClickType clickType) { | ||||
|         int elementBefore = this.options.getLast(); | ||||
|         int elementAfter = this.options.getFirst(); | ||||
|         int listIndex = this.options.indexOf(this.state); | ||||
|         if(listIndex > 0) elementBefore = this.options.get(listIndex-1); | ||||
|         if(listIndex < this.options.size()-1) elementAfter = this.options.get(listIndex+1); | ||||
|  | ||||
|         if(clickType.equals(ClickType.LEFT)) this.state = elementBefore; | ||||
|         if(clickType.equals(ClickType.RIGHT)) this.state = elementAfter; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void fromStorage(PersistentDataContainer container) { | ||||
|         this.state = container.has(this.getNamespacedKey()) | ||||
|             ? Integer.valueOf(Objects.requireNonNull(container.get(this.getNamespacedKey(), PersistentDataType.STRING))) | ||||
|             : this.defaultValue(); | ||||
|  | ||||
|         if(!this.options.contains(this.state)) this.state = this.defaultValue(); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void toStorage(PersistentDataContainer container, Integer value) { | ||||
|         container.set(this.getNamespacedKey(), PersistentDataType.STRING, new Gson().toJson(value)); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public Class<?> dataType() { | ||||
|         return Integer.class; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public Integer state() { | ||||
|         return this.state; | ||||
|     } | ||||
| } | ||||
| @@ -1,7 +1,7 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.settings.datatypes; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.datatypes; | ||||
| 
 | ||||
| import com.google.gson.Gson; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.entity.Player; | ||||
| @@ -35,9 +35,10 @@ public abstract class MultiBoolSetting<T> extends Setting<T> { | ||||
| 
 | ||||
|     @Override | ||||
|     public ItemMeta buildMeta(ItemMeta meta) { | ||||
|         record SettingField(String name, String displayName, Boolean value) {} | ||||
|         record SettingField(String name, String displayName, Boolean value) { | ||||
|         } | ||||
| 
 | ||||
|         meta.displayName(Component.text(title(), NamedTextColor.WHITE)); | ||||
|         meta.displayName(Component.text(this.title(), NamedTextColor.WHITE)); | ||||
|         List<Component> lore = new ArrayList<>(); | ||||
|         lore.add(Component.text("Status: ", NamedTextColor.DARK_GRAY)); | ||||
| 
 | ||||
| @@ -54,12 +55,12 @@ public abstract class MultiBoolSetting<T> extends Setting<T> { | ||||
|                             : component.getName(); | ||||
| 
 | ||||
|                         return new SettingField(component.getName(), displayName, value); | ||||
|                     } catch (NoSuchMethodException | InvocationTargetException | IllegalAccessException e) { | ||||
|                     } catch(NoSuchMethodException | InvocationTargetException | IllegalAccessException e) { | ||||
|                         throw new RuntimeException(e); | ||||
|                     } | ||||
|                 }) | ||||
|                 .map(field -> { | ||||
|                     if (cursorPosition == null) cursorPosition = field.name; | ||||
|                     if(this.cursorPosition == null) this.cursorPosition = field.name; | ||||
|                     boolean isSelected = field.name.equals(this.cursorPosition); | ||||
|                     return Component.text() | ||||
|                         .append(Component.text( | ||||
| @@ -79,7 +80,7 @@ public abstract class MultiBoolSetting<T> extends Setting<T> { | ||||
|                 .toList() | ||||
|         ); | ||||
|         lore.add(Component.empty()); | ||||
|         lore.addAll(buildDescription(description())); | ||||
|         lore.addAll(this.buildDescription(this.description())); | ||||
|         lore.add(Component.empty()); | ||||
|         lore.add(Component.text("Linksklick", NamedTextColor.AQUA).append(Component.text(" zum Wählen der Option", NamedTextColor.GRAY))); | ||||
|         lore.add(Component.text("Rechtsklick", NamedTextColor.AQUA).append(Component.text(" zum Ändern des Wertes", NamedTextColor.GRAY))); | ||||
| @@ -96,16 +97,16 @@ public abstract class MultiBoolSetting<T> extends Setting<T> { | ||||
|             .findFirst() | ||||
|             .orElse(-1); | ||||
| 
 | ||||
|         if (clickType.equals(ClickType.LEFT)) { | ||||
|         if(clickType.equals(ClickType.LEFT)) { | ||||
|             currentIndex = (currentIndex + 1) % recordComponents.length; | ||||
|             this.cursorPosition = recordComponents[currentIndex].getName(); | ||||
|         } else if (clickType.equals(ClickType.RIGHT)) { | ||||
|         } else if(clickType.equals(ClickType.RIGHT)) { | ||||
|             try { | ||||
|                 Object[] values = Arrays.stream(recordComponents) | ||||
|                     .map(rc -> { | ||||
|                         try { | ||||
|                             return this.state.getClass().getDeclaredMethod(rc.getName()).invoke(this.state); | ||||
|                         } catch (NoSuchMethodException | IllegalAccessException | InvocationTargetException e) { | ||||
|                         } catch(NoSuchMethodException | IllegalAccessException | InvocationTargetException e) { | ||||
|                             throw new RuntimeException(e); | ||||
|                         } | ||||
|                     }) | ||||
| @@ -119,7 +120,8 @@ public abstract class MultiBoolSetting<T> extends Setting<T> { | ||||
|                 this.state = (T) this.state.getClass().getConstructor( | ||||
|                     Arrays.stream(recordComponents).map(RecordComponent::getType).toArray(Class[]::new) | ||||
|                 ).newInstance(values); | ||||
|             } catch (NoSuchMethodException | InvocationTargetException | IllegalAccessException | InstantiationException e) { | ||||
|             } catch(NoSuchMethodException | InvocationTargetException | IllegalAccessException | | ||||
|                     InstantiationException e) { | ||||
|                 throw new RuntimeException(e); | ||||
|             } | ||||
|         } | ||||
| @@ -127,21 +129,21 @@ public abstract class MultiBoolSetting<T> extends Setting<T> { | ||||
| 
 | ||||
|     @Override | ||||
|     protected void fromStorage(PersistentDataContainer container) { | ||||
|         String data = container.has(getNamespacedKey()) | ||||
|             ? Objects.requireNonNull(container.get(getNamespacedKey(), PersistentDataType.STRING)) | ||||
|             : new Gson().toJson(defaultValue()); | ||||
|         String data = container.has(this.getNamespacedKey()) | ||||
|             ? Objects.requireNonNull(container.get(this.getNamespacedKey(), PersistentDataType.STRING)) | ||||
|             : new Gson().toJson(this.defaultValue()); | ||||
| 
 | ||||
|         try { | ||||
|             //noinspection unchecked | ||||
|             this.state = (T) new Gson().fromJson(data, dataType()); | ||||
|             this.state = (T) new Gson().fromJson(data, this.dataType()); | ||||
|         } catch(Exception e) { | ||||
|             this.state = defaultValue(); | ||||
|             this.state = this.defaultValue(); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     protected void toStorage(PersistentDataContainer container, T value) { | ||||
|         container.set(getNamespacedKey(), PersistentDataType.STRING, new Gson().toJson(value)); | ||||
|         container.set(this.getNamespacedKey(), PersistentDataType.STRING, new Gson().toJson(value)); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @@ -1,6 +1,6 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.settings.datatypes; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.datatypes; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.Material; | ||||
| @@ -37,7 +37,7 @@ public abstract class SelectSetting extends Setting<SelectSetting.Options.Option | ||||
| 
 | ||||
|     @Override | ||||
|     public ItemMeta buildMeta(ItemMeta meta) { | ||||
|         meta.displayName(Component.text(title(), NamedTextColor.WHITE)); | ||||
|         meta.displayName(Component.text(this.title(), NamedTextColor.WHITE)); | ||||
|         List<Component> lore = new ArrayList<>(); | ||||
|         lore.add(Component.text("Status: ", NamedTextColor.DARK_GRAY)); | ||||
|         lore.addAll( | ||||
| @@ -52,7 +52,7 @@ public abstract class SelectSetting extends Setting<SelectSetting.Options.Option | ||||
|                 .toList() | ||||
|         ); | ||||
|         lore.add(Component.empty()); | ||||
|         lore.addAll(buildDescription(description())); | ||||
|         lore.addAll(this.buildDescription(this.description())); | ||||
|         meta.lore(lore); | ||||
|         return meta; | ||||
|     } | ||||
| @@ -65,9 +65,9 @@ public abstract class SelectSetting extends Setting<SelectSetting.Options.Option | ||||
|             .filter(i -> options.get(i).equals(this.state)) | ||||
|             .mapToObj(i -> { | ||||
|                 int nextIndex = i + optionModifier; | ||||
|                 if (nextIndex >= options.size()) { | ||||
|                 if(nextIndex >= options.size()) { | ||||
|                     return options.getFirst(); | ||||
|                 } else if (nextIndex < 0) { | ||||
|                 } else if(nextIndex < 0) { | ||||
|                     return options.getLast(); | ||||
|                 } else { | ||||
|                     return options.get(nextIndex); | ||||
| @@ -79,19 +79,19 @@ public abstract class SelectSetting extends Setting<SelectSetting.Options.Option | ||||
| 
 | ||||
|     @Override | ||||
|     protected void fromStorage(PersistentDataContainer container) { | ||||
|         String data = container.has(getNamespacedKey()) | ||||
|             ? Objects.requireNonNull(container.get(getNamespacedKey(), PersistentDataType.STRING)) | ||||
|             : defaultValue().key.asString(); | ||||
|         String data = container.has(this.getNamespacedKey()) | ||||
|             ? Objects.requireNonNull(container.get(this.getNamespacedKey(), PersistentDataType.STRING)) | ||||
|             : this.defaultValue().key.asString(); | ||||
| 
 | ||||
|         this.state = this.options.options.stream() | ||||
|             .filter(option -> option.key.asString().equals(data)) | ||||
|             .findFirst() | ||||
|             .orElse(defaultValue()); | ||||
|             .orElse(this.defaultValue()); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     protected void toStorage(PersistentDataContainer container, Options.Option value) { | ||||
|         container.set(getNamespacedKey(), PersistentDataType.STRING, value.key.asString()); | ||||
|         container.set(this.getNamespacedKey(), PersistentDataType.STRING, value.key.asString()); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @@ -1,8 +1,9 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.settings.datatypes; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.datatypes; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.Main; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.util.text.ComponentUtil; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.ComponentUtil; | ||||
| import eu.mhsl.craftattack.spawn.core.util.world.InteractSounds; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.TextComponent; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| @@ -17,7 +18,7 @@ import org.bukkit.persistence.PersistentDataContainer; | ||||
| import java.util.List; | ||||
| 
 | ||||
| public abstract class Setting<TDataType> { | ||||
|     TDataType state; | ||||
|     protected TDataType state; | ||||
|     private final Settings.Key key; | ||||
| 
 | ||||
|     public Setting(Settings.Key key) { | ||||
| @@ -25,25 +26,45 @@ public abstract class Setting<TDataType> { | ||||
|     } | ||||
| 
 | ||||
|     public NamespacedKey getNamespacedKey() { | ||||
|         return new NamespacedKey(Main.instance(), key.name()); | ||||
|         return new NamespacedKey(Settings.class.getSimpleName().toLowerCase(), this.key.name().toLowerCase()); | ||||
|     } | ||||
| 
 | ||||
|     public Settings.Key getKey() { | ||||
|         return key; | ||||
|         return this.key; | ||||
|     } | ||||
| 
 | ||||
|     public void initializeFromPlayer(Player p) { | ||||
|         fromStorage(p.getPersistentDataContainer()); | ||||
|         PersistentDataContainer dataContainer = p.getPersistentDataContainer(); | ||||
|         try { | ||||
|             this.fromStorage(dataContainer); | ||||
|         } catch(IllegalArgumentException e) { | ||||
|             Main.logger().warning(String.format( | ||||
|                 "Could not load state of setting %s from player %s: '%s'\n Did the datatype of the setting change?", | ||||
|                 this.getNamespacedKey(), | ||||
|                 e.getMessage(), | ||||
|                 p.getName() | ||||
|             )); | ||||
|             dataContainer.remove(this.getNamespacedKey()); | ||||
|             this.fromStorage(dataContainer); | ||||
|             Main.logger().info(String.format( | ||||
|                 "Restoring defaults of setting %s of player %s", | ||||
|                 this.getNamespacedKey(), | ||||
|                 p.getName() | ||||
|             )); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     public void triggerChange(Player p, ClickType clickType) { | ||||
|         if(clickType.equals(ClickType.DOUBLE_CLICK)) return; | ||||
|         this.change(p, clickType); | ||||
|         toStorage(p.getPersistentDataContainer(), this.state()); | ||||
|         InteractSounds.of(p).click(); | ||||
|         this.toStorage(p.getPersistentDataContainer(), this.state()); | ||||
|         Settings.instance().invokeChangeListener(p, this.getClass()); | ||||
|     } | ||||
| 
 | ||||
|     public ItemStack buildItem() { | ||||
|         ItemStack stack = new ItemStack(icon(), 1); | ||||
|         stack.setItemMeta(buildMeta(stack.getItemMeta())); | ||||
|         ItemStack stack = new ItemStack(this.icon(), 1); | ||||
|         stack.setItemMeta(this.buildMeta(stack.getItemMeta())); | ||||
|         return stack; | ||||
|     } | ||||
| 
 | ||||
| @@ -1,7 +1,7 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.settings.listeners; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.listeners; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceListener; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceListener; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.player.PlayerSwapHandItemsEvent; | ||||
| 
 | ||||
| @@ -9,8 +9,9 @@ public class OpenSettingsShortcutListener extends ApplianceListener<Settings> { | ||||
|     @EventHandler | ||||
|     public void onItemSwitch(PlayerSwapHandItemsEvent event) { | ||||
|         if(!event.getPlayer().isSneaking()) return; | ||||
|         if(!Settings.instance().getSetting(event.getPlayer(), Settings.Key.EnableSettingsShortcut, Boolean.class)) return; | ||||
|         if(!Settings.instance().getSetting(event.getPlayer(), Settings.Key.EnableSettingsShortcut, Boolean.class)) | ||||
|             return; | ||||
|         event.setCancelled(true); | ||||
|         getAppliance().openSettings(event.getPlayer()); | ||||
|         this.getAppliance().openSettings(event.getPlayer()); | ||||
|     } | ||||
| } | ||||
| @@ -1,7 +1,7 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.settings.listeners; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.listeners; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceListener; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceListener; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.inventory.InventoryClickEvent; | ||||
| @@ -11,10 +11,10 @@ public class SettingsInventoryListener extends ApplianceListener<Settings> { | ||||
|     @EventHandler | ||||
|     public void onInventoryClick(InventoryClickEvent event) { | ||||
|         Player player = (Player) event.getWhoClicked(); | ||||
|         if(getAppliance().hasSettingsNotOpen(player)) return; | ||||
|         if(this.getAppliance().hasSettingsNotOpen(player)) return; | ||||
|         event.setCancelled(true); | ||||
| 
 | ||||
|         Settings.OpenSettingsInventory openInventory = getAppliance().getOpenInventory(player); | ||||
|         Settings.OpenSettingsInventory openInventory = this.getAppliance().getOpenInventory(player); | ||||
|         openInventory.settings().stream() | ||||
|             .filter(setting -> setting.buildItem().equals(event.getCurrentItem())) | ||||
|             .findFirst() | ||||
| @@ -26,6 +26,6 @@ public class SettingsInventoryListener extends ApplianceListener<Settings> { | ||||
| 
 | ||||
|     @EventHandler | ||||
|     public void onInventoryClose(InventoryCloseEvent event) { | ||||
|         getAppliance().onSettingsClose((Player) event.getPlayer()); | ||||
|         this.getAppliance().onSettingsClose((Player) event.getPlayer()); | ||||
|     } | ||||
| } | ||||
| @@ -1,13 +1,13 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.tablist; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.tablist; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.Main; | ||||
| import eu.mhsl.craftattack.spawn.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.appliances.report.Report; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.util.IteratorUtil; | ||||
| import eu.mhsl.craftattack.spawn.util.statistics.NetworkMonitor; | ||||
| import eu.mhsl.craftattack.spawn.util.text.ComponentUtil; | ||||
| import eu.mhsl.craftattack.spawn.util.text.RainbowComponent; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.report.Report; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.core.util.IteratorUtil; | ||||
| import eu.mhsl.craftattack.spawn.core.util.statistics.NetworkMonitor; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.ComponentUtil; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.RainbowComponent; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import net.kyori.adventure.util.Ticks; | ||||
| @@ -23,7 +23,8 @@ import java.util.List; | ||||
| 
 | ||||
| 
 | ||||
| public class Tablist extends Appliance { | ||||
|     private final RainbowComponent serverName = new RainbowComponent(" CraftAttack 7 ", 7, 3); | ||||
|     private final String projectTitle = this.localConfig().getString("projectTitle", "Title not configured"); | ||||
|     private final RainbowComponent serverName = new RainbowComponent(String.format(" %s ", this.projectTitle), 7, 3); | ||||
|     private NetworkMonitor networkMonitor; | ||||
|     private OperatingSystemMXBean systemMonitor; | ||||
| 
 | ||||
| @@ -36,7 +37,7 @@ public class Tablist extends Appliance { | ||||
|         Settings.instance().declareSetting(TechnicalTablistSetting.class); | ||||
| 
 | ||||
|         int tabRefreshRate = 3; | ||||
|         this.networkMonitor = new NetworkMonitor(localConfig().getString("interface"), Duration.ofSeconds(1)); | ||||
|         this.networkMonitor = new NetworkMonitor(this.localConfig().getString("interface"), Duration.ofSeconds(1)); | ||||
|         this.systemMonitor = ManagementFactory.getOperatingSystemMXBean(); | ||||
| 
 | ||||
|         Bukkit.getScheduler().runTaskTimerAsynchronously( | ||||
| @@ -53,14 +54,14 @@ public class Tablist extends Appliance { | ||||
|     } | ||||
| 
 | ||||
|     public void fullUpdate(Player player) { | ||||
|         updateHeader(player); | ||||
|         updateFooter(player); | ||||
|         this.updateHeader(player); | ||||
|         this.updateFooter(player); | ||||
|     } | ||||
| 
 | ||||
|     private void updateHeader(Player player) { | ||||
|         boolean detailedInfo = queryAppliance(Settings.class).getSetting(player, Settings.Key.TechnicalTab, Boolean.class); | ||||
|         boolean detailedInfo = this.queryAppliance(Settings.class).getSetting(player, Settings.Key.TechnicalTab, Boolean.class); | ||||
|         Component header = Component.newline() | ||||
|             .append(serverName.getRainbowState()).appendNewline() | ||||
|             .append(this.serverName.getRainbowState()).appendNewline() | ||||
|             .append(Component.text("mhsl.eu", NamedTextColor.GOLD)).appendNewline().appendNewline() | ||||
|             .append(ComponentUtil.getFormattedTickTimes(detailedInfo)).appendNewline(); | ||||
| 
 | ||||
| @@ -0,0 +1,12 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.tablist; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceListener; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.player.PlayerJoinEvent; | ||||
|  | ||||
| class TablistListener extends ApplianceListener<Tablist> { | ||||
|     @EventHandler | ||||
|     public void onPlayerJoin(PlayerJoinEvent event) { | ||||
|         this.getAppliance().fullUpdate(event.getPlayer()); | ||||
|     } | ||||
| } | ||||
| @@ -1,9 +1,9 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.tablist; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.tablist; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.CategorizedSetting; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.SettingCategory; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.appliances.settings.datatypes.BoolSetting; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.CategorizedSetting; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.SettingCategory; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.Settings; | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.metaGameplay.settings.datatypes.BoolSetting; | ||||
| import org.bukkit.Material; | ||||
| 
 | ||||
| public class TechnicalTablistSetting extends BoolSetting implements CategorizedSetting { | ||||
| @@ -0,0 +1,62 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.security.antiAutoTotem; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.common.appliances.tooling.acInform.AcInform; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.Material; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.bukkit.inventory.PlayerInventory; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.concurrent.atomic.AtomicInteger; | ||||
| import java.util.function.Function; | ||||
| import java.util.function.Supplier; | ||||
|  | ||||
| public class AntiAutoTotem extends Appliance { | ||||
|  | ||||
|     public void checkTotemUse(Player player) { | ||||
|         PlayerInventory playerInv = player.getInventory(); | ||||
|  | ||||
|         Supplier<List<Material>> getHeldItems = () -> List.of( | ||||
|             playerInv.getItemInMainHand().getType(), | ||||
|             playerInv.getItemInOffHand().getType() | ||||
|         ); | ||||
|         Function<Player, Boolean> isCurrentlyHoldingTotem = (p) -> getHeldItems.get().contains(Material.TOTEM_OF_UNDYING); | ||||
|  | ||||
|         if(!isCurrentlyHoldingTotem.apply(player)) return; | ||||
|         if(getHeldItems.get().stream().allMatch(material -> material.equals(Material.TOTEM_OF_UNDYING))) return; | ||||
|  | ||||
|         AtomicInteger tickCounter = new AtomicInteger(); | ||||
|         Bukkit.getScheduler().runTaskTimer( | ||||
|             Main.instance(), | ||||
|             (task) -> { | ||||
|                 if(tickCounter.incrementAndGet() > 10) { | ||||
|                     task.cancel(); | ||||
|                     return; | ||||
|                 } | ||||
|  | ||||
|                 if(isCurrentlyHoldingTotem.apply(player)) { | ||||
|                     task.cancel(); | ||||
|                     Main.instance().getAppliance(AcInform.class).notifyAdmins( | ||||
|                         "internal", | ||||
|                         player.getName(), | ||||
|                         "antiAutoTotem", | ||||
|                         (float) tickCounter.get() | ||||
|                     ); | ||||
|                 } | ||||
|             }, | ||||
|             1, | ||||
|             1 | ||||
|         ); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected @NotNull List<Listener> listeners() { | ||||
|         return List.of( | ||||
|             new OnTotemUseListener() | ||||
|         ); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,15 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.security.antiAutoTotem; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceListener; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.entity.EntityResurrectEvent; | ||||
|  | ||||
| class OnTotemUseListener extends ApplianceListener<AntiAutoTotem> { | ||||
|     @EventHandler | ||||
|     public void onTotem(EntityResurrectEvent event) { | ||||
|         if(event.isCancelled()) return; | ||||
|         if(!(event.getEntity() instanceof Player player)) return; | ||||
|         this.getAppliance().checkTotemUse(player); | ||||
|     } | ||||
| } | ||||
| @@ -1,7 +1,7 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.acInform; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.acInform; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.ComponentBuilder; | ||||
| import net.kyori.adventure.text.TextComponent; | ||||
| @@ -11,62 +11,60 @@ import org.bukkit.Bukkit; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
| 
 | ||||
| import java.util.*; | ||||
| import java.util.List; | ||||
| 
 | ||||
| public class AcInform extends Appliance { | ||||
|     public void processCommand(@NotNull String[] args) { | ||||
|         String anticheatName = null; | ||||
|         String playerName = null; | ||||
|         String checkName = null; | ||||
|         Integer violationCount = null; | ||||
|         Float violationCount = null; | ||||
| 
 | ||||
|         for(int i = 0; i < args.length; i++) { | ||||
|             if(!args[i].startsWith("--")) continue; | ||||
|             if(i == args.length-1) continue; | ||||
|             String nextArgument = args[i+1]; | ||||
|             if(nextArgument.startsWith("--")) continue; | ||||
| 
 | ||||
|             StringBuilder valueBuilder = new StringBuilder(); | ||||
|             for(int j = i + 1; j < args.length; j++) { | ||||
|                 if(args[j].startsWith("--")) break; | ||||
|                 if(!valueBuilder.isEmpty()) valueBuilder.append(" "); | ||||
|                 valueBuilder.append(args[j]); | ||||
|             } | ||||
| 
 | ||||
|             String value = valueBuilder.toString(); | ||||
|             switch(args[i]) { | ||||
|                 case "--anticheatName" -> anticheatName = nextArgument; | ||||
|                 case "--playerName" -> playerName = nextArgument; | ||||
|                 case "--check" -> checkName = nextArgument; | ||||
|                 case "--violationCount" -> violationCount = Integer.valueOf(nextArgument); | ||||
|                 case "--anticheatName" -> anticheatName = value; | ||||
|                 case "--playerName" -> playerName = value; | ||||
|                 case "--check" -> checkName = value; | ||||
|                 case "--violationCount" -> violationCount = value.isEmpty() ? null : Float.valueOf(value); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         this.notifyAdmins(anticheatName, playerName, checkName, violationCount); | ||||
|     } | ||||
| 
 | ||||
|     public void notifyAdmins(@Nullable String anticheatName, @Nullable String playerName, @Nullable String checkName, @Nullable Integer violationCount) { | ||||
|     public void notifyAdmins(@Nullable String anticheatName, @Nullable String playerName, @Nullable String checkName, @Nullable Float violationCount) { | ||||
|         ComponentBuilder<TextComponent, TextComponent.Builder> component = Component.text(); | ||||
|         Component prefix = Component.text("# ", NamedTextColor.DARK_RED); | ||||
|         NamedTextColor textColor = NamedTextColor.GRAY; | ||||
| 
 | ||||
|         if(playerName == null || playerName.isBlank()) throw new ApplianceCommand.Error("acinform command needs a player (--playerName)"); | ||||
|         if(playerName == null || playerName.isBlank()) | ||||
|             throw new ApplianceCommand.Error("acinform command needs a player (--playerName)"); | ||||
| 
 | ||||
|         if(anticheatName != null && !anticheatName.isBlank()) { | ||||
|             component.append( | ||||
|                 Component.newline() | ||||
|                     .append(prefix) | ||||
|             component | ||||
|                 .append(Component.text("⊤ ", NamedTextColor.GRAY)) | ||||
|                 .append(Component.text("[", textColor)) | ||||
|                     .append(Component.text("Anticheat", NamedTextColor.RED)) | ||||
|                     .append(Component.text("] ", textColor)) | ||||
|                     .append(Component.text(anticheatName, NamedTextColor.WHITE)) | ||||
|                     .append(Component.text(":", textColor)) | ||||
|             ); | ||||
|                 .append(Component.text(anticheatName, NamedTextColor.RED)) | ||||
|                 .append(Component.text("]: ", textColor)); | ||||
|         } | ||||
| 
 | ||||
|         component.append( | ||||
|             Component.newline() | ||||
|                 .append(prefix) | ||||
|         component | ||||
|             .append(Component.text("Player ", textColor)) | ||||
|             .append(Component.text(playerName, NamedTextColor.WHITE)) | ||||
|                 .append(Component.text(" ")) | ||||
|         ); | ||||
|             .append(Component.text(" ")); | ||||
| 
 | ||||
|         if(checkName == null || checkName.isBlank()) { | ||||
|             component.append(Component.text("got detected by Anticheat", textColor)); | ||||
|         } else if(violationCount != null){ | ||||
|         } else if(violationCount != null) { | ||||
|             component.append( | ||||
|                 Component.text("failed ", textColor) | ||||
|                     .append(Component.text(String.format("%sx ", violationCount), NamedTextColor.WHITE)) | ||||
| @@ -85,8 +83,7 @@ public class AcInform extends Appliance { | ||||
| 
 | ||||
|         component.append( | ||||
|             Component.newline() | ||||
|                 .append(prefix) | ||||
| 
 | ||||
|                 .append(Component.text("⊥ ", NamedTextColor.GRAY)) | ||||
|                 .append(Component.text("[", NamedTextColor.GRAY)) | ||||
|                 .append(Component.text("Report", NamedTextColor.GOLD)) | ||||
|                 .append(Component.text("]", NamedTextColor.GRAY)) | ||||
| @@ -109,12 +106,11 @@ public class AcInform extends Appliance { | ||||
| 
 | ||||
|         component.append( | ||||
|             Component.text(" [", NamedTextColor.GRAY) | ||||
|                 .append(Component.text("Teleport", NamedTextColor.GOLD)) | ||||
|                 .append(Component.text("Spectate/Teleport", NamedTextColor.GOLD)) | ||||
|                 .append(Component.text("]", NamedTextColor.GRAY)) | ||||
|                 .clickEvent(ClickEvent.suggestCommand(String.format("/tp %s", playerName))) | ||||
|                 .clickEvent(ClickEvent.suggestCommand(String.format("/grim spectate %s", playerName))) | ||||
|         ); | ||||
| 
 | ||||
|         component.appendNewline(); | ||||
|         TextComponent finalMessage = component.build(); | ||||
| 
 | ||||
|         Bukkit.getOnlinePlayers().stream() | ||||
| @@ -1,12 +1,12 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.acInform; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.acInform; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import org.bukkit.command.Command; | ||||
| import org.bukkit.command.CommandSender; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| 
 | ||||
| public class AcInformCommand extends ApplianceCommand<AcInform> { | ||||
| class AcInformCommand extends ApplianceCommand<AcInform> { | ||||
|     public AcInformCommand() { | ||||
|         super("acInform"); | ||||
|     } | ||||
| @@ -14,6 +14,6 @@ public class AcInformCommand extends ApplianceCommand<AcInform> { | ||||
|     @Override | ||||
|     protected void execute(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) throws Exception { | ||||
|         if(sender instanceof Player) throw new ApplianceCommand.Error("Dieser Command ist nicht für Spieler!"); | ||||
|         getAppliance().processCommand(args); | ||||
|         this.getAppliance().processCommand(args); | ||||
|     } | ||||
| } | ||||
| @@ -1,7 +1,7 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.adminChat; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.adminChat; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.event.ClickEvent; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| @@ -1,11 +1,11 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.adminChat; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.adminChat; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import org.bukkit.command.Command; | ||||
| import org.bukkit.command.CommandSender; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| 
 | ||||
| public class AdminChatCommand extends ApplianceCommand.PlayerChecked<AdminChat> { | ||||
| class AdminChatCommand extends ApplianceCommand.PlayerChecked<AdminChat> { | ||||
|     public static final String commandName = "adminchat"; | ||||
| 
 | ||||
|     public AdminChatCommand() { | ||||
| @@ -17,6 +17,6 @@ public class AdminChatCommand extends ApplianceCommand.PlayerChecked<AdminChat> | ||||
|         if(!sender.hasPermission("admin")) return; | ||||
| 
 | ||||
|         String message = String.join(" ", args); | ||||
|         getAppliance().sendMessage(getPlayer(), message); | ||||
|         this.getAppliance().sendMessage(this.getPlayer(), message); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,44 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.chatMute; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import org.bukkit.NamespacedKey; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.bukkit.persistence.PersistentDataContainer; | ||||
| import org.bukkit.persistence.PersistentDataType; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.Locale; | ||||
| import java.util.Objects; | ||||
|  | ||||
| public class ChatMute extends Appliance { | ||||
|     private static final String namespace = ChatMute.class.getSimpleName().toLowerCase(Locale.ROOT); | ||||
|     public static NamespacedKey mutedUntilKey = new NamespacedKey(namespace, "mutedUntilMillis".toLowerCase()); | ||||
|  | ||||
|     public void mutePlayer(Player player, int durationHours) { | ||||
|         PersistentDataContainer container = player.getPersistentDataContainer(); | ||||
|         long mutedUntil = System.currentTimeMillis() + (long) durationHours * 60 * 60 * 1000; | ||||
|         container.set(ChatMute.mutedUntilKey, PersistentDataType.LONG, mutedUntil); | ||||
|     } | ||||
|  | ||||
|     public @Nullable Long muteStatus(Player player) { | ||||
|         PersistentDataContainer container = player.getPersistentDataContainer(); | ||||
|         if(!container.has(mutedUntilKey)) return null; | ||||
|         long mutedUntil = Objects.requireNonNull(container.get(mutedUntilKey, PersistentDataType.LONG)); | ||||
|         if(mutedUntil < System.currentTimeMillis()) return null; | ||||
|         return mutedUntil; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected @NotNull List<ApplianceCommand<?>> commands() { | ||||
|         return List.of(new MuteCommand()); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected @NotNull List<Listener> listeners() { | ||||
|         return List.of(new ChatMuteListener()); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,25 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.chatMute; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceListener; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.DataSizeConverter; | ||||
| import io.papermc.paper.event.player.AsyncChatEvent; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
|  | ||||
| class ChatMuteListener extends ApplianceListener<ChatMute> { | ||||
|     @EventHandler | ||||
|     public void onChat(AsyncChatEvent event) { | ||||
|         @Nullable Long muteDuration = this.getAppliance().muteStatus(event.getPlayer()); | ||||
|         if(muteDuration == null) return; | ||||
|         event.setCancelled(true); | ||||
|         event.getPlayer().sendMessage(Component.text( | ||||
|             String.format( | ||||
|                 "Du bist für %s gestummt!", | ||||
|                 DataSizeConverter.formatSecondsToHumanReadable((int) ((muteDuration - System.currentTimeMillis()) / 1000)) | ||||
|             ), | ||||
|             NamedTextColor.RED | ||||
|         )); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,35 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.chatMute; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.command.Command; | ||||
| import org.bukkit.command.CommandSender; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
|  | ||||
| import java.util.List; | ||||
| import java.util.Objects; | ||||
|  | ||||
| class MuteCommand extends ApplianceCommand<ChatMute> { | ||||
|     public MuteCommand() { | ||||
|         super("mute"); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected void execute(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) throws Exception { | ||||
|         if(args.length < 2) throw new Error("Syntax: mute <player> <duration in hours>"); | ||||
|         Player player = Objects.requireNonNull(Bukkit.getPlayer(args[0])); | ||||
|         int durationInHours = Integer.parseInt(args[1]); | ||||
|         this.getAppliance().mutePlayer(player, durationInHours); | ||||
|         sender.sendMessage(String.format("%s wurde für %d Stunden gestummt!", player.getName(), durationInHours)); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     public @Nullable List<String> onTabComplete(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) { | ||||
|         if(args.length == 2) { | ||||
|             return List.of("1", "2", "4", "8", "24", "48"); | ||||
|         } | ||||
|         return null; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,63 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.endPrevent; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.config.Configuration; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
|  | ||||
| import java.util.List; | ||||
|  | ||||
| public class EndPrevent extends Appliance { | ||||
|     private final String endPreventKey = "endPrevent"; | ||||
|     private State endPreventState; | ||||
|     private final World endWorld = Bukkit.getWorlds().stream().filter(world -> world.getEnvironment().equals(World.Environment.THE_END)).findFirst().orElseThrow(); | ||||
|  | ||||
|     public enum State { | ||||
|         OPEN, | ||||
|         CLOSED, | ||||
|         NO_OUTER | ||||
|     } | ||||
|  | ||||
|     public EndPrevent() { | ||||
|         super("endPrevent"); | ||||
|         this.endPreventState = State.valueOf(this.localConfig().getString(this.endPreventKey, State.OPEN.name())); | ||||
|         this.updateEndBorder(); | ||||
|     } | ||||
|  | ||||
|     private void updateEndBorder() { | ||||
|         if(this.endPreventState == State.NO_OUTER) this.endWorld.getWorldBorder().setSize(500); | ||||
|         if(this.endPreventState == State.OPEN) this.endWorld.getWorldBorder().setSize(this.endWorld.getWorldBorder().getMaxSize()); | ||||
|     } | ||||
|  | ||||
|     public void setEndState(State state) { | ||||
|         this.localConfig().set(this.endPreventKey, state.name()); | ||||
|         Configuration.saveChanges(); | ||||
|         this.endPreventState = state; | ||||
|         this.updateEndBorder(); | ||||
|     } | ||||
|  | ||||
|     public boolean isEndClosed() { | ||||
|         return this.endPreventState.equals(State.CLOSED); | ||||
|     } | ||||
|  | ||||
|     public boolean isOnlyInner() { | ||||
|         return this.endPreventState.equals(State.NO_OUTER); | ||||
|     } | ||||
|  | ||||
|     public State getEndPreventState() { | ||||
|         return this.endPreventState; | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected @NotNull List<Listener> listeners() { | ||||
|         return List.of(new EndPreventListener()); | ||||
|     } | ||||
|  | ||||
|     @Override | ||||
|     protected @NotNull List<ApplianceCommand<?>> commands() { | ||||
|         return List.of(new EndPreventCommand()); | ||||
|     } | ||||
| } | ||||
| @@ -1,6 +1,6 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.endPrevent; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.endPrevent; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.command.Command; | ||||
| @@ -11,8 +11,12 @@ import org.jetbrains.annotations.Nullable; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| 
 | ||||
| public class EndPreventCommand extends ApplianceCommand<EndPrevent> { | ||||
|     private final Map<String, Boolean> arguments = Map.of("preventEnd", true, "allowEnd", false); | ||||
| class EndPreventCommand extends ApplianceCommand<EndPrevent> { | ||||
|     private final Map<String, EndPrevent.State> arguments = Map.of( | ||||
|         "preventEnd", EndPrevent.State.CLOSED, | ||||
|         "allowEnd", EndPrevent.State.OPEN, | ||||
|         "onlyInnerEnd", EndPrevent.State.NO_OUTER | ||||
|     ); | ||||
| 
 | ||||
|     public EndPreventCommand() { | ||||
|         super("endPrevent"); | ||||
| @@ -20,18 +24,18 @@ public class EndPreventCommand extends ApplianceCommand<EndPrevent> { | ||||
| 
 | ||||
|     @Override | ||||
|     protected void execute(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) throws Exception { | ||||
|         if(args.length == 1 && arguments.containsKey(args[0])) { | ||||
|             getAppliance().setEndDisabled(arguments.get(args[0])); | ||||
|         if(args.length == 1 && this.arguments.containsKey(args[0])) { | ||||
|             this.getAppliance().setEndState(this.arguments.get(args[0])); | ||||
|             sender.sendMessage(Component.text("Setting updated!", NamedTextColor.GREEN)); | ||||
|         } | ||||
|         sender.sendMessage(Component.text( | ||||
|             String.format("The End is %s!", getAppliance().isEndDisabled() ? "open" : "closed"), | ||||
|             String.format("The End is now on '%s'!", this.getAppliance().getEndPreventState().name()), | ||||
|             NamedTextColor.GOLD | ||||
|         )); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public @Nullable List<String> onTabComplete(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) { | ||||
|         return arguments.keySet().stream().toList(); | ||||
|         return this.arguments.keySet().stream().toList(); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,49 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.endPrevent; | ||||
|  | ||||
| import com.destroystokyo.paper.event.player.PlayerTeleportEndGatewayEvent; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceListener; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.Material; | ||||
| import org.bukkit.World; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.event.Cancellable; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.player.PlayerInteractEvent; | ||||
| import org.bukkit.event.player.PlayerPortalEvent; | ||||
|  | ||||
| class EndPreventListener extends ApplianceListener<EndPrevent> { | ||||
|     @EventHandler | ||||
|     public void onEnderEyeInteraction(PlayerInteractEvent event) { | ||||
|         if(event.getClickedBlock() == null) return; | ||||
|         if(!event.getClickedBlock().getType().equals(Material.END_PORTAL_FRAME)) return; | ||||
|         if(event.getItem() == null) return; | ||||
|         if(!event.getItem().getType().equals(Material.ENDER_EYE)) return; | ||||
|  | ||||
|         this.cancelIfClosed(event, event.getPlayer()); | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
|     public void onPlayerPortal(PlayerPortalEvent event) { | ||||
|         if(!event.getTo().getWorld().getEnvironment().equals(World.Environment.THE_END)) return; | ||||
|  | ||||
|         this.cancelIfClosed(event, event.getPlayer()); | ||||
|     } | ||||
|  | ||||
|     @EventHandler | ||||
|     public void onPlayerEndGateway(PlayerTeleportEndGatewayEvent event) { | ||||
|         this.cancelIfOnlyInner(event, event.getPlayer()); | ||||
|     } | ||||
|  | ||||
|     private void cancelIfClosed(Cancellable event, Player p) { | ||||
|         if(!this.getAppliance().isEndClosed()) return; | ||||
|         event.setCancelled(true); | ||||
|         p.sendActionBar(Component.text("Das End ist nicht freigeschaltet!", NamedTextColor.RED)); | ||||
|     } | ||||
|  | ||||
|     private void cancelIfOnlyInner(Cancellable event, Player p) { | ||||
|         if(!this.getAppliance().isOnlyInner()) return; | ||||
|         event.setCancelled(true); | ||||
|         p.sendActionBar(Component.text("Das Outer End ist nicht freigeschaltet!", NamedTextColor.RED)); | ||||
|     } | ||||
| } | ||||
| @@ -1,8 +1,8 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.kick; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.kick; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.util.text.DisconnectInfo; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.DisconnectInfo; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| @@ -1,6 +1,6 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.kick; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.kick; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.command.Command; | ||||
| import org.bukkit.command.CommandSender; | ||||
| @@ -12,14 +12,15 @@ import java.util.Arrays; | ||||
| import java.util.List; | ||||
| import java.util.stream.Collectors; | ||||
| 
 | ||||
| public class KickCommand extends ApplianceCommand<Kick> { | ||||
| class KickCommand extends ApplianceCommand<Kick> { | ||||
|     public KickCommand() { | ||||
|         super("kick"); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     protected void execute(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) throws Exception { | ||||
|         getAppliance().kick( | ||||
|         if(args.length < 1) throw new Error("Es muss ein Spielername angegeben werden!"); | ||||
|         this.getAppliance().kick( | ||||
|             args[0], | ||||
|             Arrays.stream(args).skip(1).collect(Collectors.joining(" ")) | ||||
|         ); | ||||
| @@ -1,8 +1,8 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.maintenance; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.maintenance; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.config.Configuration; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.config.Configuration; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| 
 | ||||
| @@ -18,17 +18,17 @@ public class Maintenance extends Appliance { | ||||
| 
 | ||||
|     @Override | ||||
|     public void onEnable() { | ||||
|         this.isInMaintenance = localConfig().getBoolean(configKey, false); | ||||
|         this.isInMaintenance = this.localConfig().getBoolean(this.configKey, false); | ||||
|     } | ||||
| 
 | ||||
|     public void setState(boolean enabled) { | ||||
|         this.isInMaintenance = enabled; | ||||
|         localConfig().set(configKey, enabled); | ||||
|         this.localConfig().set(this.configKey, enabled); | ||||
|         Configuration.saveChanges(); | ||||
|     } | ||||
| 
 | ||||
|     public boolean isInMaintenance() { | ||||
|         return isInMaintenance; | ||||
|         return this.isInMaintenance; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @@ -1,6 +1,6 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.maintenance; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.maintenance; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.command.Command; | ||||
| @@ -11,7 +11,7 @@ import org.jetbrains.annotations.Nullable; | ||||
| import java.util.List; | ||||
| import java.util.Map; | ||||
| 
 | ||||
| public class MaintenanceCommand extends ApplianceCommand<Maintenance> { | ||||
| class MaintenanceCommand extends ApplianceCommand<Maintenance> { | ||||
|     private final Map<String, Boolean> arguments = Map.of("enable", true, "disable", false); | ||||
| 
 | ||||
|     public MaintenanceCommand() { | ||||
| @@ -20,18 +20,18 @@ public class MaintenanceCommand extends ApplianceCommand<Maintenance> { | ||||
| 
 | ||||
|     @Override | ||||
|     protected void execute(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) throws Exception { | ||||
|         if(args.length == 1 && arguments.containsKey(args[0])) { | ||||
|             getAppliance().setState(arguments.get(args[0])); | ||||
|         if(args.length == 1 && this.arguments.containsKey(args[0])) { | ||||
|             this.getAppliance().setState(this.arguments.get(args[0])); | ||||
|             sender.sendMessage(Component.text("Maintanance mode updated!", NamedTextColor.GREEN)); | ||||
|         } | ||||
|         sender.sendMessage(Component.text( | ||||
|             String.format("Maintanance mode is %b", getAppliance().isInMaintenance()), | ||||
|             String.format("Maintanance mode is %b", this.getAppliance().isInMaintenance()), | ||||
|             NamedTextColor.GOLD | ||||
|         )); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public @Nullable List<String> onTabComplete(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) { | ||||
|         return arguments.keySet().stream().toList(); | ||||
|         return this.arguments.keySet().stream().toList(); | ||||
|     } | ||||
| } | ||||
| @@ -1,14 +1,14 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.maintenance; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.maintenance; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceListener; | ||||
| import eu.mhsl.craftattack.spawn.util.text.DisconnectInfo; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceListener; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.DisconnectInfo; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.player.PlayerLoginEvent; | ||||
| 
 | ||||
| public class PreventMaintenanceJoinListener extends ApplianceListener<Maintenance> { | ||||
| class PreventMaintenanceJoinListener extends ApplianceListener<Maintenance> { | ||||
|     @EventHandler | ||||
|     public void onJoin(PlayerLoginEvent event) { | ||||
|         if(!getAppliance().isInMaintenance()) return; | ||||
|         if(!this.getAppliance().isInMaintenance()) return; | ||||
|         if(event.getPlayer().hasPermission("bypassMaintainance")) return; | ||||
| 
 | ||||
|         DisconnectInfo disconnectInfo = new DisconnectInfo( | ||||
| @@ -1,8 +1,8 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.panicBan; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.panicBan; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.util.text.DisconnectInfo; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.DisconnectInfo; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.entity.Player; | ||||
| import org.bukkit.event.Listener; | ||||
| @@ -22,15 +22,15 @@ public class PanicBan extends Appliance { | ||||
|         if(player == null) | ||||
|             throw new ApplianceCommand.Error("Player not found"); | ||||
| 
 | ||||
|         panicBans.put(player.getUniqueId(), System.currentTimeMillis()); | ||||
|         this.panicBans.put(player.getUniqueId(), System.currentTimeMillis()); | ||||
|         this.getDisconnectInfo(player.getUniqueId()).applyKick(player); | ||||
|     } | ||||
| 
 | ||||
|     public boolean isBanned(UUID player) { | ||||
|         if(panicBans.containsKey(player) && panicBans.get(player) < System.currentTimeMillis() - 15 * 60 * 1000) | ||||
|             panicBans.remove(player); | ||||
|         if(this.panicBans.containsKey(player) && this.panicBans.get(player) < System.currentTimeMillis() - 15 * 60 * 1000) | ||||
|             this.panicBans.remove(player); | ||||
| 
 | ||||
|         return panicBans.containsKey(player); | ||||
|         return this.panicBans.containsKey(player); | ||||
|     } | ||||
| 
 | ||||
|     public DisconnectInfo getDisconnectInfo(UUID playerUuid) { | ||||
| @@ -1,6 +1,6 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.panicBan; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.panicBan; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.OfflinePlayer; | ||||
| import org.bukkit.command.Command; | ||||
| @@ -13,14 +13,14 @@ import java.util.Arrays; | ||||
| import java.util.List; | ||||
| import java.util.stream.Stream; | ||||
| 
 | ||||
| public class PanicBanCommand extends ApplianceCommand<PanicBan> { | ||||
| class PanicBanCommand extends ApplianceCommand<PanicBan> { | ||||
|     public PanicBanCommand() { | ||||
|         super("panicBan"); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     protected void execute(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) throws Exception { | ||||
|         getAppliance().panicBan(args[0]); | ||||
|         this.getAppliance().panicBan(args[0]); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @@ -0,0 +1,15 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.panicBan; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceListener; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.player.AsyncPlayerPreLoginEvent; | ||||
|  | ||||
| class PanicBanJoinListener extends ApplianceListener<PanicBan> { | ||||
|     @EventHandler | ||||
|     public void onLogin(AsyncPlayerPreLoginEvent event) { | ||||
|         if(this.getAppliance().isBanned(event.getUniqueId())) { | ||||
|             event.kickMessage(this.getAppliance().getDisconnectInfo(event.getUniqueId()).getComponent()); | ||||
|             event.setLoginResult(AsyncPlayerPreLoginEvent.Result.KICK_BANNED); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,8 +1,8 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.playerlimit; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.playerlimit; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.config.Configuration; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.config.Configuration; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| @@ -15,18 +15,18 @@ public class PlayerLimit extends Appliance { | ||||
| 
 | ||||
|     public PlayerLimit() { | ||||
|         super("playerLimit"); | ||||
|         this.limit = localConfig().getInt(playerLimitKey); | ||||
|         this.limit = this.localConfig().getInt(playerLimitKey); | ||||
|         Bukkit.setMaxPlayers(Integer.MAX_VALUE); | ||||
|     } | ||||
| 
 | ||||
|     public void setPlayerLimit(int limit) { | ||||
|         this.limit = limit; | ||||
|         localConfig().set(playerLimitKey, limit); | ||||
|         this.localConfig().set(playerLimitKey, limit); | ||||
|         Configuration.saveChanges(); | ||||
|     } | ||||
| 
 | ||||
|     public int getLimit() { | ||||
|         return limit; | ||||
|         return this.limit; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @@ -0,0 +1,24 @@ | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.playerlimit; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceListener; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.DisconnectInfo; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.event.EventHandler; | ||||
| import org.bukkit.event.player.AsyncPlayerPreLoginEvent; | ||||
|  | ||||
| class PlayerLimiterListener extends ApplianceListener<PlayerLimit> { | ||||
|     @EventHandler | ||||
|     public void onLogin(AsyncPlayerPreLoginEvent playerPreLoginEvent) { | ||||
|         if(Bukkit.getOnlinePlayers().size() >= this.getAppliance().getLimit()) { | ||||
|             playerPreLoginEvent.kickMessage( | ||||
|                 new DisconnectInfo( | ||||
|                     "Hohe Serverauslastung", | ||||
|                     "Der Server ist momentan an seiner Kapazitätsgrenze angelangt!", | ||||
|                     "Bitte versuche es zu einem späteren Zeitpunkt erneut.", | ||||
|                     playerPreLoginEvent.getUniqueId() | ||||
|                 ).getComponent() | ||||
|             ); | ||||
|             playerPreLoginEvent.setLoginResult(AsyncPlayerPreLoginEvent.Result.KICK_FULL); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,13 +1,13 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.playerlimit; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.playerlimit; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.command.Command; | ||||
| import org.bukkit.command.CommandSender; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| 
 | ||||
| public class SetPlayerLimitCommand extends ApplianceCommand<PlayerLimit> { | ||||
| class SetPlayerLimitCommand extends ApplianceCommand<PlayerLimit> { | ||||
|     public SetPlayerLimitCommand() { | ||||
|         super("setPlayerLimit"); | ||||
|     } | ||||
| @@ -18,11 +18,11 @@ public class SetPlayerLimitCommand extends ApplianceCommand<PlayerLimit> { | ||||
|             sender.sendMessage( | ||||
|                 Component.text() | ||||
|                     .append(Component.text("Das aktuelle Spielerlimit beträgt: ", NamedTextColor.GRAY)) | ||||
|                     .append(Component.text(getAppliance().getLimit(), NamedTextColor.GOLD)) | ||||
|                     .append(Component.text(this.getAppliance().getLimit(), NamedTextColor.GOLD)) | ||||
|             ); | ||||
|             return; | ||||
|         } | ||||
| 
 | ||||
|         getAppliance().setPlayerLimit(Integer.parseInt(args[0])); | ||||
|         this.getAppliance().setPlayerLimit(Integer.parseInt(args[0])); | ||||
|     } | ||||
| } | ||||
| @@ -1,18 +1,17 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.restart.command; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.restart; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.appliances.restart.Restart; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import org.bukkit.command.Command; | ||||
| import org.bukkit.command.CommandSender; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| 
 | ||||
| public class CancelRestartCommand extends ApplianceCommand<Restart> { | ||||
| class CancelRestartCommand extends ApplianceCommand<Restart> { | ||||
|     public CancelRestartCommand() { | ||||
|         super("cancelRestart"); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     protected void execute(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) throws Exception { | ||||
|         getAppliance().cancelRestart(); | ||||
|         this.getAppliance().cancelRestart(); | ||||
|     } | ||||
| } | ||||
| @@ -1,12 +1,9 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.restart; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.restart; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.appliances.restart.command.CancelRestartCommand; | ||||
| import eu.mhsl.craftattack.spawn.appliances.restart.command.ScheduleRestartCommand; | ||||
| import eu.mhsl.craftattack.spawn.util.IteratorUtil; | ||||
| import eu.mhsl.craftattack.spawn.util.text.Countdown; | ||||
| import eu.mhsl.craftattack.spawn.util.text.DisconnectInfo; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.core.util.IteratorUtil; | ||||
| import eu.mhsl.craftattack.spawn.core.util.text.Countdown; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.Bukkit; | ||||
| @@ -51,13 +48,6 @@ public class Restart extends Appliance { | ||||
|     } | ||||
| 
 | ||||
|     private void onDone() { | ||||
|         IteratorUtil.onlinePlayers( | ||||
|             player -> new DisconnectInfo( | ||||
|                 "Serverneustart", | ||||
|                 "Wir sind gleich wieder online!", | ||||
|                 "Verbinde Dich dann erneut.", | ||||
|                 player.getUniqueId() | ||||
|             ).applyKick(player)); | ||||
|         Bukkit.shutdown(); | ||||
|     } | ||||
| 
 | ||||
| @@ -1,18 +1,17 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliances.restart.command; | ||||
| package eu.mhsl.craftattack.spawn.common.appliances.tooling.restart; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.appliance.ApplianceCommand; | ||||
| import eu.mhsl.craftattack.spawn.appliances.restart.Restart; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.ApplianceCommand; | ||||
| import org.bukkit.command.Command; | ||||
| import org.bukkit.command.CommandSender; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| 
 | ||||
| public class ScheduleRestartCommand extends ApplianceCommand<Restart> { | ||||
| class ScheduleRestartCommand extends ApplianceCommand<Restart> { | ||||
|     public ScheduleRestartCommand() { | ||||
|         super("scheduleRestart"); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     protected void execute(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) throws Exception { | ||||
|         getAppliance().scheduleRestart(); | ||||
|         this.getAppliance().scheduleRestart(); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										7
									
								
								core/build.gradle
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								core/build.gradle
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| dependencies { | ||||
|     compileOnly 'io.papermc.paper:paper-api:1.21.8-R0.1-SNAPSHOT' | ||||
|     compileOnly 'org.geysermc.floodgate:api:2.2.4-SNAPSHOT' | ||||
|     implementation 'org.apache.httpcomponents:httpclient:4.5.14' | ||||
|     implementation 'com.sparkjava:spark-core:2.9.4' | ||||
|     implementation 'org.reflections:reflections:0.10.2' | ||||
| } | ||||
| @@ -1,8 +1,9 @@ | ||||
| package eu.mhsl.craftattack.spawn; | ||||
| package eu.mhsl.craftattack.spawn.core; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.api.HttpServer; | ||||
| import eu.mhsl.craftattack.spawn.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.config.Configuration; | ||||
| import eu.mhsl.craftattack.spawn.core.api.client.RepositoryLoader; | ||||
| import eu.mhsl.craftattack.spawn.core.api.server.HttpServer; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.config.Configuration; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.event.HandlerList; | ||||
| import org.bukkit.plugin.java.JavaPlugin; | ||||
| @@ -15,19 +16,22 @@ import java.util.logging.Level; | ||||
| import java.util.logging.Logger; | ||||
| 
 | ||||
| public final class Main extends JavaPlugin { | ||||
|     public static final String projectPackage = "eu.mhsl.craftattack.spawn"; | ||||
|     private static Main instance; | ||||
|     private static Logger logger; | ||||
| 
 | ||||
|     private List<Appliance> appliances; | ||||
|     private RepositoryLoader repositoryLoader; | ||||
|     private Reflections reflections; | ||||
| 
 | ||||
|     @Override | ||||
|     public void onEnable() { | ||||
|         instance = this; | ||||
|         logger = instance().getLogger(); | ||||
|         saveDefaultConfig(); | ||||
|         this.saveDefaultConfig(); | ||||
|         try { | ||||
|             this.wrappedEnable(); | ||||
|         } catch (Exception e) { | ||||
|         } catch(Exception e) { | ||||
|             Main.logger().log(Level.SEVERE, "Error while initializing Spawn plugin, shutting down!", e); | ||||
|             Bukkit.shutdown(); | ||||
|         } | ||||
| @@ -37,40 +41,46 @@ public final class Main extends JavaPlugin { | ||||
|         Configuration.readConfig(); | ||||
|         List<String> disabledAppliances = Configuration.pluginConfig.getStringList("disabledAppliances"); | ||||
| 
 | ||||
|         Main.logger.info("Loading appliances..."); | ||||
|         Reflections reflections = new Reflections(this.getClass().getPackageName()); | ||||
|         Set<Class<? extends Appliance>> applianceClasses = reflections.getSubTypesOf(Appliance.class); | ||||
|         Main.logger().info("Initializing reflections..."); | ||||
|         this.reflections = new Reflections(projectPackage); | ||||
| 
 | ||||
|         this.appliances = applianceClasses.stream() | ||||
|         Main.logger().info("Loading Repositories..."); | ||||
|         this.repositoryLoader = new RepositoryLoader(); | ||||
|         Main.logger().info(String.format("Loaded %d repositories!", this.repositoryLoader.getRepositories().size())); | ||||
| 
 | ||||
|         Main.logger().info("Loading appliances..."); | ||||
|         this.appliances = this.findSubtypesOf(Appliance.class).stream() | ||||
|             .filter(applianceClass -> !disabledAppliances.contains(applianceClass.getSimpleName())) | ||||
|             .map(applianceClass -> { | ||||
|                 try { | ||||
|                     return (Appliance) applianceClass.getDeclaredConstructor().newInstance(); | ||||
|                 } catch (Exception e) { | ||||
|                 } catch(Exception e) { | ||||
|                     throw new RuntimeException(String.format("Failed to create instance of '%s'", applianceClass.getName()), e); | ||||
|                 } | ||||
|             }) | ||||
|             .toList(); | ||||
|         Main.logger().info(String.format("Loaded %d appliances!", appliances.size())); | ||||
|         Main.logger().info(String.format("Loaded %d appliances!", this.appliances.size())); | ||||
| 
 | ||||
|         Main.logger().info("Initializing appliances..."); | ||||
|         this.appliances.forEach(appliance -> { | ||||
|             appliance.onEnable(); | ||||
|             appliance.initialize(this); | ||||
|         }); | ||||
|         Main.logger().info(String.format("Initialized %d appliances!", appliances.size())); | ||||
|         Main.logger().info(String.format("Initialized %d appliances!", this.appliances.size())); | ||||
| 
 | ||||
|         if(Configuration.pluginConfig.getBoolean("httpServerEnabled", true)) { | ||||
|             Main.logger().info("Starting HTTP API..."); | ||||
|             new HttpServer(); | ||||
|         } | ||||
| 
 | ||||
|         getServer().getMessenger().registerOutgoingPluginChannel(this, "BungeeCord"); | ||||
|         this.getServer().getMessenger().registerOutgoingPluginChannel(this, "BungeeCord"); | ||||
|         Main.logger().info("Startup complete!"); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void onDisable() { | ||||
|         Main.logger().info("Disabling appliances..."); | ||||
|         appliances.forEach(appliance -> { | ||||
|         this.appliances.forEach(appliance -> { | ||||
|             Main.logger().info("Disabling " + appliance.getClass().getSimpleName()); | ||||
|             appliance.onDisable(); | ||||
|             appliance.destruct(this); | ||||
| @@ -78,7 +88,7 @@ public final class Main extends JavaPlugin { | ||||
| 
 | ||||
|         HandlerList.unregisterAll(this); | ||||
|         Bukkit.getScheduler().cancelTasks(this); | ||||
|         Main.logger().info("Disabled " + appliances.size() + " appliances!"); | ||||
|         Main.logger().info("Disabled " + this.appliances.size() + " appliances!"); | ||||
|     } | ||||
| 
 | ||||
|     public <T extends Appliance> T getAppliance(Class<T> clazz) { | ||||
| @@ -89,13 +99,21 @@ public final class Main extends JavaPlugin { | ||||
|             .orElseThrow(() -> new RuntimeException(String.format("Appliance %s not loaded or instantiated!", clazz))); | ||||
|     } | ||||
| 
 | ||||
|     public <T> Set<Class<? extends T>> findSubtypesOf(Class<T> type) { | ||||
|         return this.reflections.getSubTypesOf(type); | ||||
|     } | ||||
| 
 | ||||
|     @SuppressWarnings("unchecked") | ||||
|     public static <T> Class<T> getApplianceType(Class<?> clazz) { | ||||
|         return (Class<T>) ((ParameterizedType) clazz.getGenericSuperclass()).getActualTypeArguments()[0]; | ||||
|     } | ||||
| 
 | ||||
|     public List<Appliance> getAppliances() { | ||||
|         return appliances; | ||||
|         return this.appliances; | ||||
|     } | ||||
| 
 | ||||
|     public RepositoryLoader getRepositoryLoader() { | ||||
|         return this.repositoryLoader; | ||||
|     } | ||||
| 
 | ||||
|     public static Main instance() { | ||||
| @@ -0,0 +1,14 @@ | ||||
| package eu.mhsl.craftattack.spawn.core.api; | ||||
|  | ||||
| public class HttpStatus { | ||||
|     public static final int OK = 200; | ||||
|     public static final int CREATED = 201; | ||||
|     public static final int ACCEPTED = 202; | ||||
|     public static final int NO_CONTENT = 204; | ||||
|     public static final int BAD_REQUEST = 400; | ||||
|     public static final int UNAUTHORIZED = 401; | ||||
|     public static final int FORBIDDEN = 403; | ||||
|     public static final int NOT_FOUND = 404; | ||||
|     public static final int INTERNAL_SERVER_ERROR = 500; | ||||
|     public static final int SERVICE_UNAVAILABLE = 503; | ||||
| } | ||||
| @@ -0,0 +1,127 @@ | ||||
| package eu.mhsl.craftattack.spawn.core.api.client; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import org.apache.http.client.utils.URIBuilder; | ||||
| import org.jetbrains.annotations.Nullable; | ||||
|  | ||||
| import java.io.IOException; | ||||
| import java.net.URI; | ||||
| import java.net.URISyntaxException; | ||||
| import java.net.http.HttpClient; | ||||
| import java.net.http.HttpRequest; | ||||
| import java.net.http.HttpResponse; | ||||
| import java.util.List; | ||||
| import java.util.Objects; | ||||
| import java.util.function.Consumer; | ||||
|  | ||||
| @RepositoryLoader.Abstraction | ||||
| public abstract class HttpRepository extends Repository { | ||||
|     public record RequestModifier( | ||||
|         @Nullable Consumer<URIBuilder> uri, | ||||
|         @Nullable Consumer<HttpRequest.Builder> header | ||||
|     ) {} | ||||
|  | ||||
|     private final List<RequestModifier> baseRequestModifier; | ||||
|  | ||||
|     public HttpRepository(URI basePath) { | ||||
|         this(basePath, new RequestModifier(null, null)); | ||||
|     } | ||||
|  | ||||
|     public HttpRepository(URI basePath, RequestModifier... baseRequestModifier) { | ||||
|         super(basePath); | ||||
|  | ||||
|         this.baseRequestModifier = baseRequestModifier == null | ||||
|             ? List.of() | ||||
|             : List.of(baseRequestModifier); | ||||
|     } | ||||
|  | ||||
|     protected <TInput, TOutput> ReqResp<TOutput> post(String command, TInput data, Class<TOutput> outputType) { | ||||
|         return this.post(command, parameters -> { | ||||
|         }, data, outputType); | ||||
|     } | ||||
|  | ||||
|     protected <TInput, TOutput> ReqResp<TOutput> post(String command, Consumer<URIBuilder> parameters, TInput data, Class<TOutput> outputType) { | ||||
|         HttpRequest request = this.getRequestBuilder(this.getUri(command, parameters)) | ||||
|             .POST(HttpRequest.BodyPublishers.ofString(this.gson.toJson(data))) | ||||
|             .build(); | ||||
|  | ||||
|         return this.execute(request, outputType, data); | ||||
|     } | ||||
|  | ||||
|     protected <TInput, TOutput> ReqResp<TOutput> put(String command, TInput data, Class<TOutput> outputType) { | ||||
|         return this.put(command, parameters -> { | ||||
|         }, data, outputType); | ||||
|     } | ||||
|  | ||||
|     protected <TInput, TOutput> ReqResp<TOutput> put(String command, Consumer<URIBuilder> parameters, TInput data, Class<TOutput> outputType) { | ||||
|         HttpRequest request = this.getRequestBuilder(this.getUri(command, parameters)) | ||||
|             .PUT(HttpRequest.BodyPublishers.ofString(this.gson.toJson(data))) | ||||
|             .build(); | ||||
|  | ||||
|         return this.execute(request, outputType, data); | ||||
|     } | ||||
|  | ||||
|     protected <TOutput> ReqResp<TOutput> get(String command, Class<TOutput> outputType) { | ||||
|         return this.get(command, parameters -> { | ||||
|         }, outputType); | ||||
|     } | ||||
|  | ||||
|     protected <TOutput> ReqResp<TOutput> get(String command, Consumer<URIBuilder> parameters, Class<TOutput> outputType) { | ||||
|         HttpRequest request = this.getRequestBuilder(this.getUri(command, parameters)) | ||||
|             .GET() | ||||
|             .build(); | ||||
|  | ||||
|         return this.execute(request, outputType, null); | ||||
|     } | ||||
|  | ||||
|     private URI getUri(String command, Consumer<URIBuilder> parameters) { | ||||
|         try { | ||||
|             URIBuilder builder = new URIBuilder(this.basePath + "/" + command); | ||||
|             this.baseRequestModifier.stream() | ||||
|                 .map(requestModifier -> requestModifier.uri) | ||||
|                 .filter(Objects::nonNull) | ||||
|                 .forEach(modifier -> modifier.accept(builder)); | ||||
|  | ||||
|             parameters.accept(builder); | ||||
|             return builder.build(); | ||||
|         } catch(URISyntaxException e) { | ||||
|             throw new RuntimeException(e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private HttpRequest.Builder getRequestBuilder(URI endpoint) { | ||||
|         HttpRequest.Builder builder = HttpRequest.newBuilder() | ||||
|             .uri(endpoint) | ||||
|             .header("User-Agent", Main.instance().getServer().getBukkitVersion()) | ||||
|             .header("Content-Type", "application/json"); | ||||
|  | ||||
|         this.baseRequestModifier.stream() | ||||
|             .map(requestModifier -> requestModifier.header) | ||||
|             .filter(Objects::nonNull) | ||||
|             .forEach(modifier -> modifier.accept(builder)); | ||||
|  | ||||
|         return builder; | ||||
|     } | ||||
|  | ||||
|     private <TResponse> ReqResp<TResponse> execute(HttpRequest request, Class<TResponse> clazz, Object original) { | ||||
|         ReqResp<String> rawResponse = this.sendHttp(request); | ||||
|         Main.logger().info(String.format( | ||||
|             "Request: %s\nRequest-Data: %s\nResponse: %s", | ||||
|             request, | ||||
|             this.gson.toJson(original), | ||||
|             rawResponse | ||||
|         )); | ||||
|         return new ReqResp<>(rawResponse.status(), this.gson.fromJson(rawResponse.data(), clazz)); | ||||
|     } | ||||
|  | ||||
|     private ReqResp<String> sendHttp(HttpRequest request) { | ||||
|         try(HttpClient client = HttpClient.newHttpClient()) { | ||||
|             this.validateThread(request.uri().getPath()); | ||||
|             HttpResponse<String> httpResponse = client.send(request, HttpResponse.BodyHandlers.ofString()); | ||||
|             return new ReqResp<>(httpResponse.statusCode(), httpResponse.body()); | ||||
|         } catch(IOException | InterruptedException e) { | ||||
|             throw new RuntimeException(e); | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
| @@ -0,0 +1,30 @@ | ||||
| package eu.mhsl.craftattack.spawn.core.api.client; | ||||
|  | ||||
| import com.google.gson.Gson; | ||||
| import com.google.gson.GsonBuilder; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import org.bukkit.Bukkit; | ||||
|  | ||||
| import java.net.URI; | ||||
|  | ||||
| public abstract class Repository { | ||||
|     protected URI basePath; | ||||
|     protected Gson gson; | ||||
|  | ||||
|     public Repository(URI basePath) { | ||||
|         this.basePath = basePath; | ||||
|         this.gson = new GsonBuilder() | ||||
|             .serializeNulls() | ||||
|             .create(); | ||||
|     } | ||||
|  | ||||
|     protected void validateThread(String commandName) { | ||||
|         if(!Bukkit.isPrimaryThread()) return; | ||||
|  | ||||
|         Main.logger().warning(String.format( | ||||
|             "Repository '%s' was called synchronously with command '%s'!", | ||||
|             this.getClass().getSimpleName(), | ||||
|             commandName | ||||
|         )); | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,52 @@ | ||||
| package eu.mhsl.craftattack.spawn.core.api.client; | ||||
|  | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import org.apache.commons.lang3.NotImplementedException; | ||||
| import org.reflections.Reflections; | ||||
|  | ||||
| import java.lang.annotation.Retention; | ||||
| import java.lang.annotation.RetentionPolicy; | ||||
| import java.lang.reflect.InvocationTargetException; | ||||
| import java.util.List; | ||||
| import java.util.Set; | ||||
|  | ||||
| public class RepositoryLoader { | ||||
|     private final List<Repository> repositories; | ||||
|  | ||||
|     /** | ||||
|      * Defines a repository as an abstraction and will not be loaded by this RepositoryLoader | ||||
|      */ | ||||
|     @Retention(RetentionPolicy.RUNTIME) | ||||
|     public @interface Abstraction { | ||||
|     } | ||||
|  | ||||
|     public RepositoryLoader() { | ||||
|         Reflections reflections = new Reflections(Main.projectPackage); | ||||
|         Set<Class<? extends Repository>> repositories = reflections.getSubTypesOf(Repository.class); | ||||
|  | ||||
|         this.repositories = repositories.stream() | ||||
|             .filter(repository -> !repository.isAnnotationPresent(Abstraction.class)) | ||||
|             .map(repository -> { | ||||
|                 try { | ||||
|                     return (Repository) repository.getDeclaredConstructor().newInstance(); | ||||
|                 } catch(InstantiationException | IllegalAccessException | InvocationTargetException | | ||||
|                         NoSuchMethodException e) { | ||||
|                     throw new RuntimeException(e); | ||||
|                 } | ||||
|             }) | ||||
|             .toList(); | ||||
|     } | ||||
|  | ||||
|     public <T> T getRepository(Class<T> clazz) { | ||||
|         //noinspection unchecked | ||||
|         return this.repositories.stream() | ||||
|             .filter(clazz::isInstance) | ||||
|             .map(repository -> (T) repository) | ||||
|             .findFirst() | ||||
|             .orElseThrow(() -> new NotImplementedException(String.format("Repository '%s' not found!", clazz.getSimpleName()))); | ||||
|     } | ||||
|  | ||||
|     public List<Repository> getRepositories() { | ||||
|         return this.repositories; | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,16 @@ | ||||
| package eu.mhsl.craftattack.spawn.core.api.client; | ||||
|  | ||||
| import com.google.gson.Gson; | ||||
| import java.lang.reflect.Type; | ||||
|  | ||||
| public record ReqResp<TData>(int status, TData data) { | ||||
|     public ReqResp<?> convertToTypeToken(Type type) { | ||||
|         var gson = new Gson(); | ||||
|         return new ReqResp<>(this.status, gson.fromJson(gson.toJson(this.data), type)); | ||||
|     } | ||||
|  | ||||
|     @SuppressWarnings("unchecked") | ||||
|     public <T> T cast() { | ||||
|         return (T) this; | ||||
|     } | ||||
| } | ||||
| @@ -1,8 +1,8 @@ | ||||
| package eu.mhsl.craftattack.spawn.api; | ||||
| package eu.mhsl.craftattack.spawn.core.api.server; | ||||
| 
 | ||||
| import com.google.gson.Gson; | ||||
| import eu.mhsl.craftattack.spawn.Main; | ||||
| import eu.mhsl.craftattack.spawn.appliance.Appliance; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.appliance.Appliance; | ||||
| import org.bukkit.configuration.ConfigurationSection; | ||||
| import spark.Request; | ||||
| import spark.Spark; | ||||
| @@ -1,16 +1,19 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliance; | ||||
| package eu.mhsl.craftattack.spawn.core.appliance; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.Main; | ||||
| import eu.mhsl.craftattack.spawn.api.HttpServer; | ||||
| import eu.mhsl.craftattack.spawn.config.Configuration; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.api.client.Repository; | ||||
| import eu.mhsl.craftattack.spawn.core.api.server.HttpServer; | ||||
| import eu.mhsl.craftattack.spawn.core.config.Configuration; | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.command.PluginCommand; | ||||
| import org.bukkit.configuration.ConfigurationSection; | ||||
| import org.bukkit.event.HandlerList; | ||||
| import org.bukkit.event.Listener; | ||||
| import org.bukkit.plugin.Plugin; | ||||
| import org.bukkit.plugin.java.JavaPlugin; | ||||
| import org.jetbrains.annotations.NotNull; | ||||
| 
 | ||||
| import java.lang.reflect.Constructor; | ||||
| import java.util.ArrayList; | ||||
| import java.util.List; | ||||
| import java.util.Optional; | ||||
| @@ -72,8 +75,8 @@ public abstract class Appliance { | ||||
|      */ | ||||
|     @NotNull | ||||
|     public ConfigurationSection localConfig() { | ||||
|         return Optional.ofNullable(Configuration.cfg.getConfigurationSection(localConfigPath)) | ||||
|             .orElseGet(() -> Configuration.cfg.createSection(localConfigPath)); | ||||
|         return Optional.ofNullable(Configuration.cfg.getConfigurationSection(this.localConfigPath)) | ||||
|             .orElseGet(() -> Configuration.cfg.createSection(this.localConfigPath)); | ||||
|     } | ||||
| 
 | ||||
|     public void onEnable() { | ||||
| @@ -83,37 +86,47 @@ public abstract class Appliance { | ||||
|     } | ||||
| 
 | ||||
|     public void initialize(@NotNull JavaPlugin plugin) { | ||||
|         this.listeners = listeners(); | ||||
|         this.commands = commands(); | ||||
|         this.listeners = this.listeners(); | ||||
|         this.commands = this.commands(); | ||||
| 
 | ||||
|         listeners.forEach(listener -> Bukkit.getPluginManager().registerEvents(listener, plugin)); | ||||
|         commands.forEach(command -> setCommandExecutor(plugin, command.commandName, command)); | ||||
|         this.listeners.forEach(listener -> Bukkit.getPluginManager().registerEvents(listener, plugin)); | ||||
|         this.commands.forEach(command -> this.setCommandExecutor(plugin, command.commandName, command)); | ||||
|     } | ||||
| 
 | ||||
|     public void destruct(@NotNull JavaPlugin plugin) { | ||||
|         listeners.forEach(HandlerList::unregisterAll); | ||||
|         this.listeners.forEach(HandlerList::unregisterAll); | ||||
|     } | ||||
| 
 | ||||
|     protected static <T extends Appliance> T queryAppliance(Class<T> clazz) { | ||||
|     protected <T extends Appliance> T queryAppliance(Class<T> clazz) { | ||||
|         return Main.instance().getAppliance(clazz); | ||||
|     } | ||||
| 
 | ||||
|     protected <T extends Repository> T queryRepository(Class<T> clazz) { | ||||
|         return Main.instance().getRepositoryLoader().getRepository(clazz); | ||||
|     } | ||||
| 
 | ||||
|     private void setCommandExecutor(JavaPlugin plugin, String name, ApplianceCommand<?> executor) { | ||||
|         PluginCommand command = plugin.getCommand(name); | ||||
|         if(command != null && executor != null) { | ||||
|         try { | ||||
|             PluginCommand command = this.createPluginCommand(name, plugin); | ||||
|             command.setExecutor(executor); | ||||
|             command.setTabCompleter(executor); | ||||
|         } else { | ||||
|             Main.logger().warning("Command " + name + " is not specified in plugin.yml!"); | ||||
|             throw new RuntimeException("All commands must be registered in plugin.yml. Missing command: " + name); | ||||
|             plugin.getServer().getCommandMap().register(plugin.getName(), command); | ||||
|         } catch(Exception e) { | ||||
|             throw new RuntimeException(String.format("Failed to register command '%s'", name), e); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     private PluginCommand createPluginCommand(String name, JavaPlugin plugin) throws Exception { | ||||
|         Constructor<PluginCommand> constructor = PluginCommand.class.getDeclaredConstructor(String.class, Plugin.class); | ||||
|         constructor.setAccessible(true); | ||||
|         return constructor.newInstance(name, plugin); | ||||
|     } | ||||
| 
 | ||||
|     public List<Listener> getListeners() { | ||||
|         return listeners; | ||||
|         return this.listeners; | ||||
|     } | ||||
| 
 | ||||
|     public List<ApplianceCommand<?>> getCommands() { | ||||
|         return commands; | ||||
|         return this.commands; | ||||
|     } | ||||
| } | ||||
| @@ -1,6 +1,6 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliance; | ||||
| package eu.mhsl.craftattack.spawn.core.appliance; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import net.kyori.adventure.text.Component; | ||||
| import net.kyori.adventure.text.format.NamedTextColor; | ||||
| import org.bukkit.command.Command; | ||||
| @@ -33,12 +33,12 @@ public abstract class ApplianceCommand<T extends Appliance> extends CachedApplia | ||||
|     @Override | ||||
|     public boolean onCommand(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) { | ||||
|         try { | ||||
|             execute(sender, command, label, args); | ||||
|             this.execute(sender, command, label, args); | ||||
|         } catch(Error e) { | ||||
|             sender.sendMessage(errorMessage.append(Component.text(e.getMessage()))); | ||||
|             sender.sendMessage(this.errorMessage.append(Component.text(e.getMessage()))); | ||||
|         } catch(Exception e) { | ||||
|             sender.sendMessage(errorMessage.append(Component.text("Interner Fehler"))); | ||||
|             Main.logger().warning("Error executing appliance command " + commandName + ": " + e.getMessage()); | ||||
|             sender.sendMessage(this.errorMessage.append(Component.text("Interner Fehler"))); | ||||
|             Main.logger().warning("Error executing appliance command " + this.commandName + ": " + e.getMessage()); | ||||
|             e.printStackTrace(System.err); | ||||
|             return false; | ||||
|         } | ||||
| @@ -80,7 +80,7 @@ public abstract class ApplianceCommand<T extends Appliance> extends CachedApplia | ||||
|         @Override | ||||
|         public boolean onCommand(@NotNull CommandSender sender, @NotNull Command command, @NotNull String label, @NotNull String[] args) { | ||||
|             if(!(sender instanceof Player)) { | ||||
|                 sender.sendMessage(notPlayerMessage); | ||||
|                 sender.sendMessage(this.notPlayerMessage); | ||||
|                 return false; | ||||
|             } | ||||
|             this.player = (Player) sender; | ||||
| @@ -1,4 +1,4 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliance; | ||||
| package eu.mhsl.craftattack.spawn.core.appliance; | ||||
| 
 | ||||
| import org.bukkit.event.Listener; | ||||
| 
 | ||||
| @@ -1,12 +1,12 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliance; | ||||
| package eu.mhsl.craftattack.spawn.core.appliance; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| 
 | ||||
| public class CachedApplianceSupplier<T extends Appliance> implements IApplianceSupplier<T> { | ||||
|     private final T appliance; | ||||
| 
 | ||||
|     public CachedApplianceSupplier() { | ||||
|         this.appliance = Main.instance().getAppliance(Main.getApplianceType(getClass())); | ||||
|         this.appliance = Main.instance().getAppliance(Main.getApplianceType(this.getClass())); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
| @@ -1,4 +1,4 @@ | ||||
| package eu.mhsl.craftattack.spawn.appliance; | ||||
| package eu.mhsl.craftattack.spawn.core.appliance; | ||||
| 
 | ||||
| public interface IApplianceSupplier<T extends Appliance> { | ||||
|     T getAppliance(); | ||||
| @@ -1,4 +1,4 @@ | ||||
| package eu.mhsl.craftattack.spawn.config; | ||||
| package eu.mhsl.craftattack.spawn.core.config; | ||||
| 
 | ||||
| import org.bukkit.Bukkit; | ||||
| import org.bukkit.Location; | ||||
| @@ -1,6 +1,6 @@ | ||||
| package eu.mhsl.craftattack.spawn.config; | ||||
| package eu.mhsl.craftattack.spawn.core.config; | ||||
| 
 | ||||
| import eu.mhsl.craftattack.spawn.Main; | ||||
| import eu.mhsl.craftattack.spawn.core.Main; | ||||
| import org.bukkit.configuration.ConfigurationSection; | ||||
| import org.bukkit.configuration.file.FileConfiguration; | ||||
| import org.bukkit.configuration.file.YamlConfiguration; | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user