Compare commits
83 Commits
10b392-cus
...
148b5fc634
Author | SHA1 | Date | |
---|---|---|---|
148b5fc634 | |||
fa69d4976d | |||
e4fff421f5 | |||
e6bded1c9e | |||
37a63e10b0 | |||
13cc6c30b5 | |||
3dd41979f7 | |||
a4c46bc298 | |||
aaad777f9b | |||
c62c7cfd1a | |||
4575164e80 | |||
edf26785a3 | |||
4d90f5fc28 | |||
343decb05a | |||
36c6c93edb | |||
24af27f2e3 | |||
13132eace6 | |||
8d479b69e3 | |||
5ca8ad8bd7 | |||
3f2ba1e428 | |||
5bab1a1ac7 | |||
8fdd0487bf | |||
e0ee1e66c9 | |||
5acb44b0e1 | |||
7cd849946b | |||
a71aca5d6c | |||
ea08ac7a81 | |||
ee192c1035 | |||
81dbf16d3f | |||
f03011e4f1 | |||
9069ead9e9 | |||
7ced598bfd | |||
f04f1b33e3 | |||
0715771bfc | |||
c18ac59442 | |||
0a1ae69f53 | |||
8451463b73 | |||
7e27a05596 | |||
ba260519e0 | |||
378d872283 | |||
bc27c35f1a | |||
18689ac0df | |||
3fe57d5fe9 | |||
0699206c21 | |||
e663f3f105 | |||
55be88b7da | |||
dc24f28b8f | |||
cc371a9c12 | |||
f85ebbbb5d | |||
720d3c8d65 | |||
e397d69d7a | |||
bef697e5fc | |||
6638a48677 | |||
73a374e529 | |||
f838317af0 | |||
3c25f5f079 | |||
179fa2e4d7 | |||
075db7a91b | |||
c9d00f4f77 | |||
4de66d65a2 | |||
8594f8029c | |||
e192ae4433 | |||
bdb7c85ceb | |||
1b448e749e | |||
2799a40c58 | |||
710838645f | |||
f56023004a | |||
55a98b6464 | |||
236d372746 | |||
1332a42bf6 | |||
5acfe8f6af | |||
9d287f7c2f | |||
7ea39f9aad | |||
21ab8c4bd3 | |||
3f247bfc90 | |||
843fa26c08 | |||
cfbb2688d2 | |||
350cf108dd | |||
c9ef7197cc | |||
40f02449d9 | |||
e71dccb98d | |||
a3f2a06f6a | |||
a321d243ba |
27
build.gradle
27
build.gradle
@@ -10,8 +10,6 @@ group 'eu.mhsl.minenet'
|
|||||||
version '1.0-SNAPSHOT'
|
version '1.0-SNAPSHOT'
|
||||||
|
|
||||||
repositories {
|
repositories {
|
||||||
//maven 'https://repo.unnamed.team/repository/unnamed-public/'
|
|
||||||
|
|
||||||
mavenCentral()
|
mavenCentral()
|
||||||
google()
|
google()
|
||||||
|
|
||||||
@@ -28,9 +26,16 @@ repositories {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
allprojects {
|
||||||
|
repositories {
|
||||||
|
maven { url "https://jitpack.io" }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
java {
|
java {
|
||||||
toolchain {
|
toolchain {
|
||||||
languageVersion = JavaLanguageVersion.of(17)
|
languageVersion = JavaLanguageVersion.of(21)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -39,28 +44,20 @@ dependencies {
|
|||||||
testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine:5.9.0'
|
testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine:5.9.0'
|
||||||
|
|
||||||
//https://jitpack.io/#Minestom/Minestom
|
//https://jitpack.io/#Minestom/Minestom
|
||||||
// implementation 'com.github.Minestom:Minestom:c496ee357'
|
implementation 'net.minestom:minestom-snapshots:fd51c8d17a'
|
||||||
// implementation 'com.github.waxeria:Minestom:e0427a36f3'
|
|
||||||
|
|
||||||
// implementation 'dev.hollowcube:minestom-ce:5bcc72b911'
|
|
||||||
implementation 'dev.hollowcube:minestom-ce:8715f4305d'
|
|
||||||
implementation 'dev.hollowcube:minestom-ce-extensions:1.2.0'
|
|
||||||
|
|
||||||
|
|
||||||
//Tools
|
//Tools
|
||||||
implementation 'de.articdive:jnoise:3.0.2'
|
implementation 'de.articdive:jnoise:3.0.2'
|
||||||
implementation 'net.md-5:bungeecord-config:1.19-R0.1-SNAPSHOT'
|
implementation 'net.md-5:bungeecord-config:1.21-R0.3'
|
||||||
implementation 'org.apache.commons:commons-text:1.10.0'
|
implementation 'org.apache.commons:commons-text:1.10.0'
|
||||||
implementation 'org.spongepowered:configurate-yaml:4.1.2'
|
implementation 'org.spongepowered:configurate-yaml:4.1.2'
|
||||||
implementation 'com.sparkjava:spark-core:2.9.4'
|
implementation 'com.sparkjava:spark-core:2.9.4'
|
||||||
implementation 'com.google.code.gson:gson:2.10.1'
|
implementation 'com.google.code.gson:gson:2.10.1'
|
||||||
implementation 'com.google.guava:guava:31.0.1-jre'
|
implementation 'com.google.guava:guava:32.0.1-android'
|
||||||
|
|
||||||
|
|
||||||
//PvP
|
//PvP
|
||||||
implementation 'com.github.TogAr2:MinestomPvP:35e5661'
|
implementation 'io.github.TogAr2:MinestomPvP:PR62-SNAPSHOT'
|
||||||
//implementation 'com.github.TogAr2:MinestomPvP:135ec9e2b7'
|
|
||||||
|
|
||||||
|
|
||||||
// Hephaestus engine
|
// Hephaestus engine
|
||||||
implementation("team.unnamed:hephaestus-api:0.2.1-SNAPSHOT")
|
implementation("team.unnamed:hephaestus-api:0.2.1-SNAPSHOT")
|
||||||
|
4
gradle/wrapper/gradle-wrapper.properties
vendored
4
gradle/wrapper/gradle-wrapper.properties
vendored
@@ -1,5 +1,7 @@
|
|||||||
distributionBase=GRADLE_USER_HOME
|
distributionBase=GRADLE_USER_HOME
|
||||||
distributionPath=wrapper/dists
|
distributionPath=wrapper/dists
|
||||||
distributionUrl=https\://services.gradle.org/distributions/gradle-7.4-bin.zip
|
distributionUrl=https\://services.gradle.org/distributions/gradle-8.10-bin.zip
|
||||||
|
networkTimeout=10000
|
||||||
|
validateDistributionUrl=true
|
||||||
zipStoreBase=GRADLE_USER_HOME
|
zipStoreBase=GRADLE_USER_HOME
|
||||||
zipStorePath=wrapper/dists
|
zipStorePath=wrapper/dists
|
||||||
|
56
gradlew
vendored
Normal file → Executable file
56
gradlew
vendored
Normal file → Executable file
@@ -15,6 +15,8 @@
|
|||||||
# See the License for the specific language governing permissions and
|
# See the License for the specific language governing permissions and
|
||||||
# limitations under the License.
|
# limitations under the License.
|
||||||
#
|
#
|
||||||
|
# SPDX-License-Identifier: Apache-2.0
|
||||||
|
#
|
||||||
|
|
||||||
##############################################################################
|
##############################################################################
|
||||||
#
|
#
|
||||||
@@ -25,7 +27,7 @@
|
|||||||
# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is
|
# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is
|
||||||
# noncompliant, but you have some other compliant shell such as ksh or
|
# noncompliant, but you have some other compliant shell such as ksh or
|
||||||
# bash, then to run this script, type that shell name before the whole
|
# bash, then to run this script, type that shell name before the whole
|
||||||
# eu.mhsl.minenet.minigames.command line, like:
|
# command line, like:
|
||||||
#
|
#
|
||||||
# ksh Gradle
|
# ksh Gradle
|
||||||
#
|
#
|
||||||
@@ -35,7 +37,7 @@
|
|||||||
# * expansions «$var», «${var}», «${var:-default}», «${var+SET}»,
|
# * expansions «$var», «${var}», «${var:-default}», «${var+SET}»,
|
||||||
# «${var#prefix}», «${var%suffix}», and «$( cmd )»;
|
# «${var#prefix}», «${var%suffix}», and «$( cmd )»;
|
||||||
# * compound commands having a testable exit status, especially «case»;
|
# * compound commands having a testable exit status, especially «case»;
|
||||||
# * various built-in commands including «eu.mhsl.minenet.minigames.command», «set», and «ulimit».
|
# * various built-in commands including «command», «set», and «ulimit».
|
||||||
#
|
#
|
||||||
# Important for patching:
|
# Important for patching:
|
||||||
#
|
#
|
||||||
@@ -55,7 +57,7 @@
|
|||||||
# Darwin, MinGW, and NonStop.
|
# Darwin, MinGW, and NonStop.
|
||||||
#
|
#
|
||||||
# (3) This script is generated from the Groovy template
|
# (3) This script is generated from the Groovy template
|
||||||
# https://github.com/gradle/gradle/blob/master/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
|
# https://github.com/gradle/gradle/blob/HEAD/platforms/jvm/plugins-application/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt
|
||||||
# within the Gradle project.
|
# within the Gradle project.
|
||||||
#
|
#
|
||||||
# You can find Gradle at https://github.com/gradle/gradle/.
|
# You can find Gradle at https://github.com/gradle/gradle/.
|
||||||
@@ -80,13 +82,12 @@ do
|
|||||||
esac
|
esac
|
||||||
done
|
done
|
||||||
|
|
||||||
APP_HOME=$( cd "${APP_HOME:-./}" && pwd -P ) || exit
|
# This is normally unused
|
||||||
|
# shellcheck disable=SC2034
|
||||||
APP_NAME="Gradle"
|
|
||||||
APP_BASE_NAME=${0##*/}
|
APP_BASE_NAME=${0##*/}
|
||||||
|
# Discard cd standard output in case $CDPATH is set (https://github.com/gradle/gradle/issues/25036)
|
||||||
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
APP_HOME=$( cd -P "${APP_HOME:-./}" > /dev/null && printf '%s
|
||||||
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
|
' "$PWD" ) || exit
|
||||||
|
|
||||||
# Use the maximum available, or set MAX_FD != -1 to use that value.
|
# Use the maximum available, or set MAX_FD != -1 to use that value.
|
||||||
MAX_FD=maximum
|
MAX_FD=maximum
|
||||||
@@ -117,7 +118,7 @@ esac
|
|||||||
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
|
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
|
||||||
|
|
||||||
|
|
||||||
# Determine the Java eu.mhsl.minenet.minigames.command to use to start the JVM.
|
# Determine the Java command to use to start the JVM.
|
||||||
if [ -n "$JAVA_HOME" ] ; then
|
if [ -n "$JAVA_HOME" ] ; then
|
||||||
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
|
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
|
||||||
# IBM's JDK on AIX uses strange locations for the executables
|
# IBM's JDK on AIX uses strange locations for the executables
|
||||||
@@ -133,29 +134,36 @@ location of your Java installation."
|
|||||||
fi
|
fi
|
||||||
else
|
else
|
||||||
JAVACMD=java
|
JAVACMD=java
|
||||||
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
if ! command -v java >/dev/null 2>&1
|
||||||
|
then
|
||||||
|
die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||||
|
|
||||||
Please set the JAVA_HOME variable in your environment to match the
|
Please set the JAVA_HOME variable in your environment to match the
|
||||||
location of your Java installation."
|
location of your Java installation."
|
||||||
|
fi
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Increase the maximum file descriptors if we can.
|
# Increase the maximum file descriptors if we can.
|
||||||
if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then
|
if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then
|
||||||
case $MAX_FD in #(
|
case $MAX_FD in #(
|
||||||
max*)
|
max*)
|
||||||
|
# In POSIX sh, ulimit -H is undefined. That's why the result is checked to see if it worked.
|
||||||
|
# shellcheck disable=SC2039,SC3045
|
||||||
MAX_FD=$( ulimit -H -n ) ||
|
MAX_FD=$( ulimit -H -n ) ||
|
||||||
warn "Could not query maximum file descriptor limit"
|
warn "Could not query maximum file descriptor limit"
|
||||||
esac
|
esac
|
||||||
case $MAX_FD in #(
|
case $MAX_FD in #(
|
||||||
'' | soft) :;; #(
|
'' | soft) :;; #(
|
||||||
*)
|
*)
|
||||||
|
# In POSIX sh, ulimit -n is undefined. That's why the result is checked to see if it worked.
|
||||||
|
# shellcheck disable=SC2039,SC3045
|
||||||
ulimit -n "$MAX_FD" ||
|
ulimit -n "$MAX_FD" ||
|
||||||
warn "Could not set maximum file descriptor limit to $MAX_FD"
|
warn "Could not set maximum file descriptor limit to $MAX_FD"
|
||||||
esac
|
esac
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Collect all arguments for the java eu.mhsl.minenet.minigames.command, stacking in reverse order:
|
# Collect all arguments for the java command, stacking in reverse order:
|
||||||
# * args from the eu.mhsl.minenet.minigames.command line
|
# * args from the command line
|
||||||
# * the main class name
|
# * the main class name
|
||||||
# * -classpath
|
# * -classpath
|
||||||
# * -D...appname settings
|
# * -D...appname settings
|
||||||
@@ -193,11 +201,15 @@ if "$cygwin" || "$msys" ; then
|
|||||||
done
|
done
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Collect all arguments for the java eu.mhsl.minenet.minigames.command;
|
|
||||||
# * $DEFAULT_JVM_OPTS, $JAVA_OPTS, and $GRADLE_OPTS can contain fragments of
|
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||||
# shell script including quotes and variable substitutions, so put them in
|
DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
|
||||||
# double quotes to make sure that they get re-expanded; and
|
|
||||||
# * put everything else in single quotes, so that it's not re-expanded.
|
# Collect all arguments for the java command:
|
||||||
|
# * DEFAULT_JVM_OPTS, JAVA_OPTS, JAVA_OPTS, and optsEnvironmentVar are not allowed to contain shell fragments,
|
||||||
|
# and any embedded shellness will be escaped.
|
||||||
|
# * For example: A user cannot expect ${Hostname} to be expanded, as it is an environment variable and will be
|
||||||
|
# treated as '${Hostname}' itself on the command line.
|
||||||
|
|
||||||
set -- \
|
set -- \
|
||||||
"-Dorg.gradle.appname=$APP_BASE_NAME" \
|
"-Dorg.gradle.appname=$APP_BASE_NAME" \
|
||||||
@@ -205,6 +217,12 @@ set -- \
|
|||||||
org.gradle.wrapper.GradleWrapperMain \
|
org.gradle.wrapper.GradleWrapperMain \
|
||||||
"$@"
|
"$@"
|
||||||
|
|
||||||
|
# Stop when "xargs" is not available.
|
||||||
|
if ! command -v xargs >/dev/null 2>&1
|
||||||
|
then
|
||||||
|
die "xargs is not available"
|
||||||
|
fi
|
||||||
|
|
||||||
# Use "xargs" to parse quoted args.
|
# Use "xargs" to parse quoted args.
|
||||||
#
|
#
|
||||||
# With -n1 it outputs one arg per line, with the quotes and backslashes removed.
|
# With -n1 it outputs one arg per line, with the quotes and backslashes removed.
|
||||||
@@ -214,7 +232,7 @@ set -- \
|
|||||||
# readarray ARGS < <( xargs -n1 <<<"$var" ) &&
|
# readarray ARGS < <( xargs -n1 <<<"$var" ) &&
|
||||||
# set -- "${ARGS[@]}" "$@"
|
# set -- "${ARGS[@]}" "$@"
|
||||||
#
|
#
|
||||||
# but POSIX shell has neither arrays nor eu.mhsl.minenet.minigames.command substitution, so instead we
|
# but POSIX shell has neither arrays nor command substitution, so instead we
|
||||||
# post-process each arg (as a line of input to sed) to backslash-escape any
|
# post-process each arg (as a line of input to sed) to backslash-escape any
|
||||||
# character that might be a shell metacharacter, then use eval to reverse
|
# character that might be a shell metacharacter, then use eval to reverse
|
||||||
# that process (while maintaining the separation between arguments), and wrap
|
# that process (while maintaining the separation between arguments), and wrap
|
||||||
|
39
gradlew.bat
vendored
39
gradlew.bat
vendored
@@ -13,8 +13,10 @@
|
|||||||
@rem See the License for the specific language governing permissions and
|
@rem See the License for the specific language governing permissions and
|
||||||
@rem limitations under the License.
|
@rem limitations under the License.
|
||||||
@rem
|
@rem
|
||||||
|
@rem SPDX-License-Identifier: Apache-2.0
|
||||||
|
@rem
|
||||||
|
|
||||||
@if "%DEBUG%" == "" @echo off
|
@if "%DEBUG%"=="" @echo off
|
||||||
@rem ##########################################################################
|
@rem ##########################################################################
|
||||||
@rem
|
@rem
|
||||||
@rem Gradle startup script for Windows
|
@rem Gradle startup script for Windows
|
||||||
@@ -25,7 +27,8 @@
|
|||||||
if "%OS%"=="Windows_NT" setlocal
|
if "%OS%"=="Windows_NT" setlocal
|
||||||
|
|
||||||
set DIRNAME=%~dp0
|
set DIRNAME=%~dp0
|
||||||
if "%DIRNAME%" == "" set DIRNAME=.
|
if "%DIRNAME%"=="" set DIRNAME=.
|
||||||
|
@rem This is normally unused
|
||||||
set APP_BASE_NAME=%~n0
|
set APP_BASE_NAME=%~n0
|
||||||
set APP_HOME=%DIRNAME%
|
set APP_HOME=%DIRNAME%
|
||||||
|
|
||||||
@@ -40,13 +43,13 @@ if defined JAVA_HOME goto findJavaFromJavaHome
|
|||||||
|
|
||||||
set JAVA_EXE=java.exe
|
set JAVA_EXE=java.exe
|
||||||
%JAVA_EXE% -version >NUL 2>&1
|
%JAVA_EXE% -version >NUL 2>&1
|
||||||
if "%ERRORLEVEL%" == "0" goto execute
|
if %ERRORLEVEL% equ 0 goto execute
|
||||||
|
|
||||||
echo.
|
echo. 1>&2
|
||||||
echo ERROR: JAVA_HOME is not set and no 'java' eu.mhsl.minenet.gameList.command could be found in your PATH.
|
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 1>&2
|
||||||
echo.
|
echo. 1>&2
|
||||||
echo Please set the JAVA_HOME variable in your environment to match the
|
echo Please set the JAVA_HOME variable in your environment to match the 1>&2
|
||||||
echo location of your Java installation.
|
echo location of your Java installation. 1>&2
|
||||||
|
|
||||||
goto fail
|
goto fail
|
||||||
|
|
||||||
@@ -56,16 +59,16 @@ set JAVA_EXE=%JAVA_HOME%/bin/java.exe
|
|||||||
|
|
||||||
if exist "%JAVA_EXE%" goto execute
|
if exist "%JAVA_EXE%" goto execute
|
||||||
|
|
||||||
echo.
|
echo. 1>&2
|
||||||
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
|
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 1>&2
|
||||||
echo.
|
echo. 1>&2
|
||||||
echo Please set the JAVA_HOME variable in your environment to match the
|
echo Please set the JAVA_HOME variable in your environment to match the 1>&2
|
||||||
echo location of your Java installation.
|
echo location of your Java installation. 1>&2
|
||||||
|
|
||||||
goto fail
|
goto fail
|
||||||
|
|
||||||
:execute
|
:execute
|
||||||
@rem Setup the eu.mhsl.minenet.gameList.command line
|
@rem Setup the command line
|
||||||
|
|
||||||
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
|
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
|
||||||
|
|
||||||
@@ -75,13 +78,15 @@ set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
|
|||||||
|
|
||||||
:end
|
:end
|
||||||
@rem End local scope for the variables with windows NT shell
|
@rem End local scope for the variables with windows NT shell
|
||||||
if "%ERRORLEVEL%"=="0" goto mainEnd
|
if %ERRORLEVEL% equ 0 goto mainEnd
|
||||||
|
|
||||||
:fail
|
:fail
|
||||||
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
|
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
|
||||||
rem the _cmd.exe /c_ return code!
|
rem the _cmd.exe /c_ return code!
|
||||||
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
|
set EXIT_CODE=%ERRORLEVEL%
|
||||||
exit /b 1
|
if %EXIT_CODE% equ 0 set EXIT_CODE=1
|
||||||
|
if not ""=="%GRADLE_EXIT_CONSOLE%" exit %EXIT_CODE%
|
||||||
|
exit /b %EXIT_CODE%
|
||||||
|
|
||||||
:mainEnd
|
:mainEnd
|
||||||
if "%OS%"=="Windows_NT" endlocal
|
if "%OS%"=="Windows_NT" endlocal
|
||||||
|
@@ -5,8 +5,6 @@ import eu.mhsl.minenet.minigames.command.Commands;
|
|||||||
import eu.mhsl.minenet.minigames.handler.Listeners;
|
import eu.mhsl.minenet.minigames.handler.Listeners;
|
||||||
import eu.mhsl.minenet.minigames.lang.Languages;
|
import eu.mhsl.minenet.minigames.lang.Languages;
|
||||||
import eu.mhsl.minenet.minigames.server.tasks.TablistUpdateTask;
|
import eu.mhsl.minenet.minigames.server.tasks.TablistUpdateTask;
|
||||||
import eu.mhsl.minenet.minigames.server.provider.ByPlayerNameUuidProvider;
|
|
||||||
import io.github.bloepiloepi.pvp.PvpExtension;
|
|
||||||
import net.minestom.server.MinecraftServer;
|
import net.minestom.server.MinecraftServer;
|
||||||
import net.minestom.server.extras.bungee.BungeeCordProxy;
|
import net.minestom.server.extras.bungee.BungeeCordProxy;
|
||||||
import net.minestom.server.extras.lan.OpenToLAN;
|
import net.minestom.server.extras.lan.OpenToLAN;
|
||||||
@@ -42,14 +40,12 @@ public class Main {
|
|||||||
logger.info("Initialize Minecraft server...");
|
logger.info("Initialize Minecraft server...");
|
||||||
|
|
||||||
MinecraftServer server = MinecraftServer.init();
|
MinecraftServer server = MinecraftServer.init();
|
||||||
PvpExtension.init();
|
// MinestomPvP.init();
|
||||||
|
|
||||||
MinecraftServer.setBrandName("mhsl.eu - minenet - credits to minestom");
|
MinecraftServer.setBrandName("mhsl.eu - minenet - credits to minestom");
|
||||||
MinecraftServer.setCompressionThreshold(serverConfig.node("compression-threshold").getInt(0));
|
MinecraftServer.setCompressionThreshold(serverConfig.node("compression-threshold").getInt(0));
|
||||||
System.setProperty("minestom.chunk-view-distance", String.valueOf(serverConfig.node("view-distance").getInt()));
|
System.setProperty("minestom.chunk-view-distance", String.valueOf(serverConfig.node("view-distance").getInt()));
|
||||||
|
|
||||||
MinecraftServer.getConnectionManager().setUuidProvider(new ByPlayerNameUuidProvider());
|
|
||||||
|
|
||||||
Commands.values();
|
Commands.values();
|
||||||
Listeners.values();
|
Listeners.values();
|
||||||
new HttpServer();
|
new HttpServer();
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
package eu.mhsl.minenet.minigames.command;
|
package eu.mhsl.minenet.minigames.command;
|
||||||
|
|
||||||
import eu.mhsl.minenet.minigames.command.anonymous.SkinCommand;
|
import eu.mhsl.minenet.minigames.command.privileged.SkinCommand;
|
||||||
import eu.mhsl.minenet.minigames.command.privileged.*;
|
import eu.mhsl.minenet.minigames.command.privileged.*;
|
||||||
import eu.mhsl.minenet.minigames.command.anonymous.HubCommand;
|
import eu.mhsl.minenet.minigames.command.anonymous.HubCommand;
|
||||||
import eu.mhsl.minenet.minigames.command.anonymous.LeaveCommand;
|
import eu.mhsl.minenet.minigames.command.anonymous.LeaveCommand;
|
||||||
@@ -41,7 +41,7 @@ public enum Commands {
|
|||||||
static {
|
static {
|
||||||
MinecraftServer.getCommandManager().setUnknownCommandCallback((sender, command) -> {
|
MinecraftServer.getCommandManager().setUnknownCommandCallback((sender, command) -> {
|
||||||
if(command.isBlank()) return;
|
if(command.isBlank()) return;
|
||||||
new ChatMessage(Icon.ERROR).appendStatic("Unknown command").quote(command).send(sender);
|
new ChatMessage(Icon.ERROR).appendStatic("Unknown command: ").quote(command).send(sender);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -2,6 +2,7 @@ package eu.mhsl.minenet.minigames.command;
|
|||||||
|
|
||||||
import net.minestom.server.command.builder.Command;
|
import net.minestom.server.command.builder.Command;
|
||||||
import net.minestom.server.command.builder.condition.CommandCondition;
|
import net.minestom.server.command.builder.condition.CommandCondition;
|
||||||
|
import net.minestom.server.entity.Player;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
import org.jetbrains.annotations.Nullable;
|
import org.jetbrains.annotations.Nullable;
|
||||||
|
|
||||||
@@ -27,7 +28,7 @@ public class PrivilegedCommand extends Command {
|
|||||||
}
|
}
|
||||||
|
|
||||||
protected CommandCondition isPrivileged() {
|
protected CommandCondition isPrivileged() {
|
||||||
return (sender, commandString) -> sender.hasPermission("admin");
|
return (sender, commandString) -> ((Player) sender).getPermissionLevel() == 4;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected void addCondition(CommandCondition condition) {
|
protected void addCondition(CommandCondition condition) {
|
||||||
|
@@ -10,9 +10,13 @@ public class HubCommand extends Command {
|
|||||||
public HubCommand() {
|
public HubCommand() {
|
||||||
super("hub");
|
super("hub");
|
||||||
|
|
||||||
setCondition((sender, commandString) -> ((Player) sender).getInstance() instanceof Room);
|
setCondition(
|
||||||
|
(sender, commandString) ->
|
||||||
|
((Player) sender).getInstance() instanceof Room room && !room.apiDriven
|
||||||
|
);
|
||||||
|
|
||||||
setDefaultExecutor((sender, context) -> {
|
setDefaultExecutor((sender, context) -> {
|
||||||
|
if(Room.getRoom((Player) sender).orElseThrow().apiDriven) return;
|
||||||
Room.unsetRoom((Player) sender);
|
Room.unsetRoom((Player) sender);
|
||||||
MoveInstance.move((Player) sender, Hub.INSTANCE);
|
MoveInstance.move((Player) sender, Hub.INSTANCE);
|
||||||
});
|
});
|
||||||
|
@@ -5,7 +5,6 @@ import eu.mhsl.minenet.minigames.message.Icon;
|
|||||||
import eu.mhsl.minenet.minigames.message.type.ActionBarMessage;
|
import eu.mhsl.minenet.minigames.message.type.ActionBarMessage;
|
||||||
import eu.mhsl.minenet.minigames.message.type.ChatMessage;
|
import eu.mhsl.minenet.minigames.message.type.ChatMessage;
|
||||||
import eu.mhsl.minenet.minigames.message.type.TitleMessage;
|
import eu.mhsl.minenet.minigames.message.type.TitleMessage;
|
||||||
import eu.mhsl.minenet.minigames.util.PacketUtil;
|
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
@@ -36,8 +35,6 @@ public class DebugCommand extends PrivilegedCommand {
|
|||||||
.appendTranslated("score#thanks")
|
.appendTranslated("score#thanks")
|
||||||
.send(sender);
|
.send(sender);
|
||||||
|
|
||||||
PacketUtil.resendPlayerList(((Player) sender));
|
|
||||||
|
|
||||||
new ChatMessage(Icon.SCIENCE).appendStatic(((Player) sender).getUuid().toString()).send(sender);
|
new ChatMessage(Icon.SCIENCE).appendStatic(((Player) sender).getUuid().toString()).send(sender);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@@ -6,8 +6,6 @@ import eu.mhsl.minenet.minigames.message.Icon;
|
|||||||
import eu.mhsl.minenet.minigames.message.type.ChatMessage;
|
import eu.mhsl.minenet.minigames.message.type.ChatMessage;
|
||||||
import net.minestom.server.command.builder.arguments.ArgumentType;
|
import net.minestom.server.command.builder.arguments.ArgumentType;
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.entity.fakeplayer.FakePlayer;
|
|
||||||
import net.minestom.server.entity.fakeplayer.FakePlayerOption;
|
|
||||||
|
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
||||||
@@ -18,12 +16,12 @@ public class FakeplayerCommand extends PrivilegedCommand {
|
|||||||
addSyntax((sender, context) -> {
|
addSyntax((sender, context) -> {
|
||||||
if(sender instanceof Player p) {
|
if(sender instanceof Player p) {
|
||||||
if(p.getInstance() instanceof Room room) {
|
if(p.getInstance() instanceof Room room) {
|
||||||
FakePlayer.initPlayer(
|
// FakePlayer.initPlayer( // TODO FakePlayer does no longer exists
|
||||||
UUID.randomUUID(),
|
// UUID.randomUUID(),
|
||||||
context.getRaw("name"),
|
// context.getRaw("name"),
|
||||||
new FakePlayerOption().setInTabList(true).setRegistered(true),
|
// new FakePlayerOption().setInTabList(true).setRegistered(true),
|
||||||
fakePlayer -> Room.setRoom(fakePlayer, room)
|
// fakePlayer -> Room.setRoom(fakePlayer, room)
|
||||||
);
|
// );
|
||||||
} else {
|
} else {
|
||||||
new ChatMessage(Icon.ERROR).appendStatic("Du musst dich in einer Raumlobby befinden!").send(sender);
|
new ChatMessage(Icon.ERROR).appendStatic("Du musst dich in einer Raumlobby befinden!").send(sender);
|
||||||
}
|
}
|
||||||
|
@@ -26,7 +26,7 @@ public class KickCommand extends PrivilegedCommand {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void kick(String playername, String reason) {
|
private void kick(String playername, String reason) {
|
||||||
Player playerToKick = MinecraftServer.getConnectionManager().findPlayer(playername);
|
Player playerToKick = MinecraftServer.getConnectionManager().findOnlinePlayer(playername);
|
||||||
Objects.requireNonNull(playerToKick).kick(reason);
|
Objects.requireNonNull(playerToKick).kick(reason);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -6,16 +6,15 @@ import eu.mhsl.minenet.minigames.message.type.ChatMessage;
|
|||||||
import net.minestom.server.MinecraftServer;
|
import net.minestom.server.MinecraftServer;
|
||||||
import net.minestom.server.command.builder.arguments.ArgumentType;
|
import net.minestom.server.command.builder.arguments.ArgumentType;
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.permission.Permission;
|
|
||||||
|
|
||||||
public class OpCommand extends PrivilegedCommand {
|
public class OpCommand extends PrivilegedCommand {
|
||||||
public OpCommand() {
|
public OpCommand() {
|
||||||
super("op");
|
super("op");
|
||||||
|
|
||||||
addSyntax((sender, context) -> {
|
addSyntax((sender, context) -> {
|
||||||
Player target = MinecraftServer.getConnectionManager().getPlayer(context.getRaw("target"));
|
Player target = MinecraftServer.getConnectionManager().getOnlinePlayerByUsername(context.getRaw("target"));
|
||||||
if(target != null) {
|
if(target != null) {
|
||||||
target.addPermission(new Permission("admin"));
|
target.setPermissionLevel(4);
|
||||||
target.refreshCommands();
|
target.refreshCommands();
|
||||||
} else new ChatMessage(Icon.ERROR).appendStatic("Spieler nicht gefunden").send(sender);
|
} else new ChatMessage(Icon.ERROR).appendStatic("Spieler nicht gefunden").send(sender);
|
||||||
}, ArgumentType.Entity("target").onlyPlayers(true));
|
}, ArgumentType.Entity("target").onlyPlayers(true));
|
||||||
|
@@ -24,7 +24,7 @@ public class PublishRewardCommand extends PrivilegedCommand {
|
|||||||
|
|
||||||
String rewardRequestJson = new Gson().toJson(room.getTournament().getRewards());
|
String rewardRequestJson = new Gson().toJson(room.getTournament().getRewards());
|
||||||
HttpRequest giveRewardsRequest = HttpRequest.newBuilder()
|
HttpRequest giveRewardsRequest = HttpRequest.newBuilder()
|
||||||
.uri(new URI("http://10.20.6.1:8080/api/event/reward"))
|
.uri(new URI("http://10.20.7.1:8080/api/event/reward"))
|
||||||
.POST(HttpRequest.BodyPublishers.ofString(rewardRequestJson))
|
.POST(HttpRequest.BodyPublishers.ofString(rewardRequestJson))
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
|
@@ -27,7 +27,7 @@ public class SetRoomOwnerCommand extends PrivilegedCommand {
|
|||||||
addSyntax((sender, context) -> {
|
addSyntax((sender, context) -> {
|
||||||
System.out.println("Test");
|
System.out.println("Test");
|
||||||
if(sender instanceof Player p) {
|
if(sender instanceof Player p) {
|
||||||
Player newOwner = MinecraftServer.getConnectionManager().getPlayer(context.getRaw("player"));
|
Player newOwner = MinecraftServer.getConnectionManager().getOnlinePlayerByUsername(context.getRaw("player"));
|
||||||
Room.getRoom(p).orElseThrow().setOwner(Objects.requireNonNull(newOwner));
|
Room.getRoom(p).orElseThrow().setOwner(Objects.requireNonNull(newOwner));
|
||||||
new ChatMessage(Icon.SUCCESS).appendStatic("The new owner has been set!").send(sender);
|
new ChatMessage(Icon.SUCCESS).appendStatic("The new owner has been set!").send(sender);
|
||||||
}
|
}
|
||||||
|
@@ -1,11 +1,11 @@
|
|||||||
package eu.mhsl.minenet.minigames.command.anonymous;
|
package eu.mhsl.minenet.minigames.command.privileged;
|
||||||
|
|
||||||
import net.minestom.server.command.builder.Command;
|
import eu.mhsl.minenet.minigames.command.PrivilegedCommand;
|
||||||
import net.minestom.server.command.builder.arguments.ArgumentType;
|
import net.minestom.server.command.builder.arguments.ArgumentType;
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.entity.PlayerSkin;
|
import net.minestom.server.entity.PlayerSkin;
|
||||||
|
|
||||||
public class SkinCommand extends Command {
|
public class SkinCommand extends PrivilegedCommand {
|
||||||
public SkinCommand() {
|
public SkinCommand() {
|
||||||
super("skin");
|
super("skin");
|
||||||
|
|
@@ -6,7 +6,6 @@ import net.minestom.server.event.EventListener;
|
|||||||
|
|
||||||
public enum Listeners {
|
public enum Listeners {
|
||||||
SPAWN(new AddEntityToInstanceEventListener()),
|
SPAWN(new AddEntityToInstanceEventListener()),
|
||||||
CHAT(new PlayerChatHandler()),
|
|
||||||
LOGIN(new PlayerLoginHandler()),
|
LOGIN(new PlayerLoginHandler()),
|
||||||
LEAVE(new PlayerLeaveHandler());
|
LEAVE(new PlayerLeaveHandler());
|
||||||
|
|
||||||
|
@@ -1,7 +1,6 @@
|
|||||||
package eu.mhsl.minenet.minigames.handler.global;
|
package eu.mhsl.minenet.minigames.handler.global;
|
||||||
|
|
||||||
import eu.mhsl.minenet.minigames.instance.Spawnable;
|
import eu.mhsl.minenet.minigames.instance.Spawnable;
|
||||||
import eu.mhsl.minenet.minigames.util.PacketUtil;
|
|
||||||
import net.minestom.server.MinecraftServer;
|
import net.minestom.server.MinecraftServer;
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.event.EventListener;
|
import net.minestom.server.event.EventListener;
|
||||||
@@ -20,15 +19,13 @@ public class AddEntityToInstanceEventListener implements EventListener<AddEntity
|
|||||||
@Override
|
@Override
|
||||||
public @NotNull Result run(@NotNull AddEntityToInstanceEvent event) {
|
public @NotNull Result run(@NotNull AddEntityToInstanceEvent event) {
|
||||||
if(event.getEntity() instanceof Player p) {
|
if(event.getEntity() instanceof Player p) {
|
||||||
MinecraftServer.getSchedulerManager().scheduleNextTick(p::refreshCommands, ExecutionType.ASYNC);
|
MinecraftServer.getSchedulerManager().scheduleNextTick(p::refreshCommands, ExecutionType.TICK_END);
|
||||||
|
|
||||||
if(event.getInstance() instanceof Spawnable instance) {
|
if(event.getInstance() instanceof Spawnable instance) {
|
||||||
p.setRespawnPoint(instance.getSpawn());
|
p.setRespawnPoint(instance.getSpawn());
|
||||||
}
|
}
|
||||||
|
|
||||||
PacketUtil.resendPlayerList(p);
|
p.addEffect(new Potion(PotionEffect.BLINDNESS, (byte) 1, 20));
|
||||||
|
|
||||||
p.addEffect(new Potion(PotionEffect.BLINDNESS, (byte) 1, 20)); //TODO Uncomment, currently buggy causes disconnect see https://github.com/Minestom/Minestom/discussions/1302
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return Result.SUCCESS;
|
return Result.SUCCESS;
|
||||||
|
@@ -1,26 +0,0 @@
|
|||||||
package eu.mhsl.minenet.minigames.handler.global;
|
|
||||||
|
|
||||||
import eu.mhsl.minenet.minigames.message.type.ChatMessage;
|
|
||||||
import net.minestom.server.event.EventListener;
|
|
||||||
import net.minestom.server.event.player.PlayerChatEvent;
|
|
||||||
import org.jetbrains.annotations.NotNull;
|
|
||||||
|
|
||||||
public class PlayerChatHandler implements EventListener<PlayerChatEvent> {
|
|
||||||
@Override
|
|
||||||
public @NotNull Class<PlayerChatEvent> eventType() {
|
|
||||||
return PlayerChatEvent.class;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public @NotNull Result run(@NotNull PlayerChatEvent event) {
|
|
||||||
event.setChatFormat(
|
|
||||||
(messages) -> new ChatMessage()
|
|
||||||
.appendStatic(event.getPlayer().getUsername())
|
|
||||||
.pipe()
|
|
||||||
.appendStatic(messages.getMessage())
|
|
||||||
.build(event.getPlayer())
|
|
||||||
);
|
|
||||||
|
|
||||||
return Result.SUCCESS;
|
|
||||||
}
|
|
||||||
}
|
|
@@ -8,10 +8,11 @@ import eu.mhsl.minenet.minigames.skin.SkinCache;
|
|||||||
import eu.mhsl.minenet.minigames.util.MoveInstance;
|
import eu.mhsl.minenet.minigames.util.MoveInstance;
|
||||||
import net.minestom.server.MinecraftServer;
|
import net.minestom.server.MinecraftServer;
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.event.EventListener;
|
|
||||||
import net.minestom.server.event.player.PlayerLoginEvent;
|
|
||||||
import eu.mhsl.minenet.minigames.instance.hub.Hub;
|
import eu.mhsl.minenet.minigames.instance.hub.Hub;
|
||||||
import net.minestom.server.permission.Permission;
|
import net.minestom.server.event.EventListener;
|
||||||
|
import net.minestom.server.event.player.AsyncPlayerConfigurationEvent;
|
||||||
|
import net.minestom.server.network.packet.server.play.TeamsPacket;
|
||||||
|
import net.minestom.server.scoreboard.Team;
|
||||||
import net.minestom.server.timer.TaskSchedule;
|
import net.minestom.server.timer.TaskSchedule;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
import org.spongepowered.configurate.serialize.SerializationException;
|
import org.spongepowered.configurate.serialize.SerializationException;
|
||||||
@@ -20,14 +21,19 @@ import java.util.Objects;
|
|||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
import java.util.logging.Logger;
|
import java.util.logging.Logger;
|
||||||
|
|
||||||
public class PlayerLoginHandler implements EventListener<PlayerLoginEvent> {
|
public class PlayerLoginHandler implements EventListener<AsyncPlayerConfigurationEvent> {
|
||||||
|
public static final Team globalTeam = MinecraftServer.getTeamManager()
|
||||||
|
.createBuilder("global")
|
||||||
|
.collisionRule(TeamsPacket.CollisionRule.NEVER)
|
||||||
|
.build();
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public @NotNull Class<PlayerLoginEvent> eventType() {
|
public @NotNull Class<AsyncPlayerConfigurationEvent> eventType() {
|
||||||
return PlayerLoginEvent.class;
|
return AsyncPlayerConfigurationEvent.class;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public @NotNull Result run(@NotNull PlayerLoginEvent event) {
|
public @NotNull Result run(@NotNull AsyncPlayerConfigurationEvent event) {
|
||||||
Player p = event.getPlayer();
|
Player p = event.getPlayer();
|
||||||
|
|
||||||
Transfer transferInstance = new Transfer();
|
Transfer transferInstance = new Transfer();
|
||||||
@@ -37,22 +43,23 @@ public class PlayerLoginHandler implements EventListener<PlayerLoginEvent> {
|
|||||||
UUID pushQueue = QueuedPlayerRooms.pullQueue(event.getPlayer().getUuid());
|
UUID pushQueue = QueuedPlayerRooms.pullQueue(event.getPlayer().getUuid());
|
||||||
|
|
||||||
MinecraftServer.getSchedulerManager().scheduleTask(
|
MinecraftServer.getSchedulerManager().scheduleTask(
|
||||||
() -> {
|
() -> {
|
||||||
if(pushQueue != null) {
|
p.setTeam(globalTeam);
|
||||||
Room.setRoom(p, Room.getRoom(pushQueue).orElseThrow());
|
if(pushQueue != null) {
|
||||||
} else {
|
Room.setRoom(p, Room.getRoom(pushQueue).orElseThrow());
|
||||||
MoveInstance.move(p, Hub.INSTANCE);
|
} else {
|
||||||
}
|
MoveInstance.move(p, Hub.INSTANCE);
|
||||||
},
|
}
|
||||||
TaskSchedule.seconds(5),
|
},
|
||||||
TaskSchedule.stop()
|
TaskSchedule.seconds(1),
|
||||||
|
TaskSchedule.stop()
|
||||||
);
|
);
|
||||||
|
|
||||||
SkinCache.applySkin(p);
|
SkinCache.applySkin(p);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
if(Objects.requireNonNull(Main.globalConfig.node("admins").getList(String.class)).stream().anyMatch(s -> s.equalsIgnoreCase(p.getUsername()))) {
|
if(Objects.requireNonNull(Main.globalConfig.node("admins").getList(String.class)).stream().anyMatch(s -> s.equalsIgnoreCase(p.getUsername()))) {
|
||||||
p.addPermission(new Permission("admin"));
|
p.setPermissionLevel(4);
|
||||||
}
|
}
|
||||||
} catch (SerializationException | NullPointerException ignored) {}
|
} catch (SerializationException | NullPointerException ignored) {}
|
||||||
|
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
package eu.mhsl.minenet.minigames.instance;
|
package eu.mhsl.minenet.minigames.instance;
|
||||||
|
|
||||||
import net.minestom.server.MinecraftServer;
|
import net.minestom.server.MinecraftServer;
|
||||||
|
import net.minestom.server.registry.DynamicRegistry;
|
||||||
import net.minestom.server.utils.NamespaceID;
|
import net.minestom.server.utils.NamespaceID;
|
||||||
import net.minestom.server.world.DimensionType;
|
import net.minestom.server.world.DimensionType;
|
||||||
|
|
||||||
@@ -9,33 +10,39 @@ import net.minestom.server.world.DimensionType;
|
|||||||
*/
|
*/
|
||||||
public enum Dimension {
|
public enum Dimension {
|
||||||
OVERWORLD(
|
OVERWORLD(
|
||||||
|
NamespaceID.from("minenet:fullbright_overworld"),
|
||||||
DimensionType
|
DimensionType
|
||||||
.builder(NamespaceID.from("minenet:fullbright_overworld"))
|
.builder()
|
||||||
.ambientLight(2.0f)
|
.ambientLight(2.0f)
|
||||||
.build()
|
.build()
|
||||||
),
|
),
|
||||||
|
|
||||||
NETHER(
|
NETHER(
|
||||||
|
NamespaceID.from("minenet:fullbright_nether"),
|
||||||
DimensionType
|
DimensionType
|
||||||
.builder(NamespaceID.from("minenet:fullbright_nether"))
|
.builder()
|
||||||
.ambientLight(2.0f)
|
.ambientLight(2.0f)
|
||||||
.effects("minecraft:the_nether")
|
.effects("minecraft:the_nether")
|
||||||
.build()
|
.build()
|
||||||
),
|
),
|
||||||
|
|
||||||
THE_END(
|
THE_END(
|
||||||
|
NamespaceID.from("minenet:fullbright_end"),
|
||||||
DimensionType
|
DimensionType
|
||||||
.builder(NamespaceID.from("minenet:fullbright_end"))
|
.builder()
|
||||||
.ambientLight(2.0f)
|
.ambientLight(2.0f)
|
||||||
.effects("minecraft:the_end")
|
.effects("minecraft:the_end")
|
||||||
.build()
|
.build()
|
||||||
);
|
);
|
||||||
|
|
||||||
public final DimensionType DIMENSION;
|
public final DimensionType DIMENSION;
|
||||||
Dimension(DimensionType dimType) {
|
public final NamespaceID namespaceID;
|
||||||
|
public final DynamicRegistry.Key<DimensionType> key;
|
||||||
|
Dimension(NamespaceID namespaceID, DimensionType dimType) {
|
||||||
this.DIMENSION = dimType;
|
this.DIMENSION = dimType;
|
||||||
|
this.namespaceID = namespaceID;
|
||||||
|
|
||||||
MinecraftServer.getDimensionTypeManager().addDimension(this.DIMENSION);
|
this.key = MinecraftServer.getDimensionTypeRegistry().register(namespaceID, DIMENSION);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@@ -7,13 +7,15 @@ import net.minestom.server.event.instance.AddEntityToInstanceEvent;
|
|||||||
import net.minestom.server.event.instance.RemoveEntityFromInstanceEvent;
|
import net.minestom.server.event.instance.RemoveEntityFromInstanceEvent;
|
||||||
import net.minestom.server.instance.Instance;
|
import net.minestom.server.instance.Instance;
|
||||||
import net.minestom.server.instance.InstanceContainer;
|
import net.minestom.server.instance.InstanceContainer;
|
||||||
|
import net.minestom.server.instance.InstanceManager;
|
||||||
|
import net.minestom.server.registry.DynamicRegistry;
|
||||||
import net.minestom.server.timer.TaskSchedule;
|
import net.minestom.server.timer.TaskSchedule;
|
||||||
import net.minestom.server.world.DimensionType;
|
import net.minestom.server.world.DimensionType;
|
||||||
|
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
||||||
public class MineNetInstance extends InstanceContainer {
|
public class MineNetInstance extends InstanceContainer {
|
||||||
public MineNetInstance(DimensionType type) {
|
public MineNetInstance(DynamicRegistry.Key<DimensionType> type) {
|
||||||
super(UUID.randomUUID(), type);
|
super(UUID.randomUUID(), type);
|
||||||
MinecraftServer.getInstanceManager().registerInstance(this);
|
MinecraftServer.getInstanceManager().registerInstance(this);
|
||||||
|
|
||||||
|
@@ -16,6 +16,7 @@ import net.minestom.server.event.item.ItemDropEvent;
|
|||||||
import net.minestom.server.event.player.PlayerBlockBreakEvent;
|
import net.minestom.server.event.player.PlayerBlockBreakEvent;
|
||||||
import net.minestom.server.event.player.PlayerBlockPlaceEvent;
|
import net.minestom.server.event.player.PlayerBlockPlaceEvent;
|
||||||
import net.minestom.server.event.player.PlayerMoveEvent;
|
import net.minestom.server.event.player.PlayerMoveEvent;
|
||||||
|
import net.minestom.server.registry.DynamicRegistry;
|
||||||
import net.minestom.server.timer.ExecutionType;
|
import net.minestom.server.timer.ExecutionType;
|
||||||
import net.minestom.server.timer.TaskSchedule;
|
import net.minestom.server.timer.TaskSchedule;
|
||||||
import net.minestom.server.world.DimensionType;
|
import net.minestom.server.world.DimensionType;
|
||||||
@@ -36,18 +37,18 @@ public abstract class Game extends MineNetInstance implements Spawnable {
|
|||||||
|
|
||||||
protected final Logger logger;
|
protected final Logger logger;
|
||||||
|
|
||||||
public Game(DimensionType dimensionType) {
|
public Game(DynamicRegistry.Key<DimensionType> dimensionType) {
|
||||||
super(dimensionType);
|
super(dimensionType);
|
||||||
|
|
||||||
MinecraftServer.getInstanceManager().registerInstance(this);
|
MinecraftServer.getInstanceManager().registerInstance(this);
|
||||||
|
|
||||||
logger = Logger.getLogger("Game:" + getUniqueId());
|
logger = Logger.getLogger("Game:" + getUuid());
|
||||||
|
|
||||||
eventNode()
|
eventNode()
|
||||||
.addListener(PlayerMoveEvent.class, this::onPlayerMove)
|
.addListener(PlayerMoveEvent.class, this::onPlayerMove)
|
||||||
.addListener(PlayerBlockBreakEvent.class, this::onBlockBreak)
|
.addListener(PlayerBlockBreakEvent.class, this::onBlockBreak)
|
||||||
.addListener(PlayerBlockPlaceEvent.class, this::onBlockPlace)
|
.addListener(PlayerBlockPlaceEvent.class, this::onBlockPlace)
|
||||||
.addListener(ItemDropEvent.class, this::onItemDrop);
|
.addListener(ItemDropEvent.class, this::onItemDrop);
|
||||||
}
|
}
|
||||||
|
|
||||||
public Game setParent(Room parentRoom) {
|
public Game setParent(Room parentRoom) {
|
||||||
@@ -62,6 +63,17 @@ public abstract class Game extends MineNetInstance implements Spawnable {
|
|||||||
game.load();
|
game.load();
|
||||||
Room.getRoom(owner).orElseThrow().moveMembersToInstance(game);
|
Room.getRoom(owner).orElseThrow().moveMembersToInstance(game);
|
||||||
|
|
||||||
|
MinecraftServer.getSchedulerManager().scheduleTask(() -> {
|
||||||
|
game.getPlayers().forEach(player -> new ChatMessage(Icon.SCIENCE)
|
||||||
|
.appendStatic(factory.name().getAssembled(player).asComponent())
|
||||||
|
.newLine()
|
||||||
|
.appendStatic(factory.description().getAssembled(player).asComponent())
|
||||||
|
.send(player));
|
||||||
|
|
||||||
|
return TaskSchedule.stop();
|
||||||
|
}, TaskSchedule.seconds(3));
|
||||||
|
|
||||||
|
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
new ChatMessage(Icon.ERROR).appendStatic("Instance crashed: " + e.getMessage()).send(owner);
|
new ChatMessage(Icon.ERROR).appendStatic("Instance crashed: " + e.getMessage()).send(owner);
|
||||||
MinecraftServer.getSchedulerManager().scheduleNextTick(() -> Room.getRoom(owner).orElseThrow().moveMembersToRoomLobby());
|
MinecraftServer.getSchedulerManager().scheduleNextTick(() -> Room.getRoom(owner).orElseThrow().moveMembersToRoomLobby());
|
||||||
@@ -78,7 +90,7 @@ public abstract class Game extends MineNetInstance implements Spawnable {
|
|||||||
this.onLoad(callback);
|
this.onLoad(callback);
|
||||||
// callback.whenComplete((unused, throwable) -> this.start());
|
// callback.whenComplete((unused, throwable) -> this.start());
|
||||||
return TaskSchedule.stop();
|
return TaskSchedule.stop();
|
||||||
}, ExecutionType.ASYNC);
|
}, ExecutionType.TICK_END);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -101,7 +113,7 @@ public abstract class Game extends MineNetInstance implements Spawnable {
|
|||||||
|
|
||||||
scheduler().scheduleTask(() -> {
|
scheduler().scheduleTask(() -> {
|
||||||
|
|
||||||
logger.info("stopping game instance " + this.uniqueId);
|
logger.info("stopping game instance " + this.uuid);
|
||||||
getPlayers().forEach(player -> player.kick("timeout"));
|
getPlayers().forEach(player -> player.kick("timeout"));
|
||||||
|
|
||||||
MinecraftServer.getInstanceManager().unregisterInstance(this);
|
MinecraftServer.getInstanceManager().unregisterInstance(this);
|
||||||
@@ -117,7 +129,6 @@ public abstract class Game extends MineNetInstance implements Spawnable {
|
|||||||
protected void onStop() {}
|
protected void onStop() {}
|
||||||
protected void onUnload() {}
|
protected void onUnload() {}
|
||||||
|
|
||||||
|
|
||||||
protected void onPlayerMove(@NotNull PlayerMoveEvent playerMoveEvent) {
|
protected void onPlayerMove(@NotNull PlayerMoveEvent playerMoveEvent) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@@ -1,32 +1,44 @@
|
|||||||
package eu.mhsl.minenet.minigames.instance.game;
|
package eu.mhsl.minenet.minigames.instance.game;
|
||||||
|
|
||||||
import eu.mhsl.minenet.minigames.instance.game.stateless.config.GameFactory;
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.GameFactory;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.anvilRun.AnvilRunFactory;
|
||||||
import eu.mhsl.minenet.minigames.instance.game.stateless.types.bowSpleef.BowSpleefFactory;
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.bowSpleef.BowSpleefFactory;
|
||||||
import eu.mhsl.minenet.minigames.instance.game.stateless.types.elytraRace.ElytraRaceFactory;
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.elytraRace.ElytraRaceFactory;
|
||||||
import eu.mhsl.minenet.minigames.instance.game.stateless.types.backrooms.BackroomsFactory;
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.backrooms.BackroomsFactory;
|
||||||
import eu.mhsl.minenet.minigames.instance.game.stateless.types.bedwars.BedwarsFactory;
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.bedwars.BedwarsFactory;
|
||||||
import eu.mhsl.minenet.minigames.instance.game.stateless.types.acidRain.AcidRainFactory;
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.acidRain.AcidRainFactory;
|
||||||
import eu.mhsl.minenet.minigames.instance.game.stateless.types.deathcube.DeathcubeFactory;
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.deathcube.DeathcubeFactory;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.fastbridge.FastbridgeFactory;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.highGround.HighGroundFactory;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.jumpDive.JumpDiveFactory;
|
||||||
import eu.mhsl.minenet.minigames.instance.game.stateless.types.minerun.MinerunFactory;
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.minerun.MinerunFactory;
|
||||||
import eu.mhsl.minenet.minigames.instance.game.stateless.types.spleef.SpleefFactory;
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.spleef.SpleefFactory;
|
||||||
import eu.mhsl.minenet.minigames.instance.game.stateless.types.stickfight.StickFightFactory;
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.stickfight.StickFightFactory;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.tetris.TetrisFactory;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.sumo.SumoFactory;
|
||||||
import eu.mhsl.minenet.minigames.instance.game.stateless.types.tntrun.TntRunFactory;
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.tntrun.TntRunFactory;
|
||||||
import eu.mhsl.minenet.minigames.instance.game.stateless.types.towerdefense.TowerdefenseFactory;
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.towerdefense.TowerdefenseFactory;
|
||||||
import eu.mhsl.minenet.minigames.instance.game.stateless.types.trafficlightrace.TrafficLightRaceFactory;
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.trafficlightrace.TrafficLightRaceFactory;
|
||||||
|
|
||||||
public enum GameList {
|
public enum GameList {
|
||||||
DEATHCUBE(new DeathcubeFactory(), GameType.JUMPNRUN),
|
DEATHCUBE(new DeathcubeFactory(), GameType.JUMPNRUN),
|
||||||
STICKFIGHT(new StickFightFactory(), GameType.PVP),
|
|
||||||
MINERUN(new MinerunFactory(), GameType.JUMPNRUN),
|
MINERUN(new MinerunFactory(), GameType.JUMPNRUN),
|
||||||
TRAFFICLIGHTRACE(new TrafficLightRaceFactory(), GameType.OTHER),
|
TRAFFICLIGHTRACE(new TrafficLightRaceFactory(), GameType.OTHER),
|
||||||
|
STICKFIGHT(new StickFightFactory(), GameType.PROTOTYPE),
|
||||||
TOWERDEFENSE(new TowerdefenseFactory(), GameType.PROTOTYPE),
|
TOWERDEFENSE(new TowerdefenseFactory(), GameType.PROTOTYPE),
|
||||||
BEDWARS(new BedwarsFactory(), GameType.PROTOTYPE),
|
BEDWARS(new BedwarsFactory(), GameType.PROTOTYPE),
|
||||||
BACKROOMS(new BackroomsFactory(), GameType.PROTOTYPE),
|
BACKROOMS(new BackroomsFactory(), GameType.PROTOTYPE),
|
||||||
|
BOWSPLEEF(new BowSpleefFactory(), GameType.PROTOTYPE),
|
||||||
|
TETRIS(new TetrisFactory(), GameType.OTHER),
|
||||||
TNTRUN(new TntRunFactory(), GameType.OTHER),
|
TNTRUN(new TntRunFactory(), GameType.OTHER),
|
||||||
|
ANVILRUN(new AnvilRunFactory(), GameType.PVE),
|
||||||
ACIDRAIN(new AcidRainFactory(), GameType.PVE),
|
ACIDRAIN(new AcidRainFactory(), GameType.PVE),
|
||||||
ELYTRARACE(new ElytraRaceFactory(), GameType.PVP),
|
ELYTRARACE(new ElytraRaceFactory(), GameType.PVP),
|
||||||
SPLEEF(new SpleefFactory(), GameType.PVP),
|
SPLEEF(new SpleefFactory(), GameType.PVP),
|
||||||
BOWSPLEEF(new BowSpleefFactory(), GameType.PVP);
|
JUMPDIVE(new JumpDiveFactory(), GameType.JUMPNRUN),
|
||||||
|
SUMO(new SumoFactory(), GameType.PVP),
|
||||||
|
HIGHGROUND(new HighGroundFactory(), GameType.PVP),
|
||||||
|
FASTBRIDGE(new FastbridgeFactory(), GameType.OTHER);
|
||||||
|
|
||||||
private final GameFactory factory;
|
private final GameFactory factory;
|
||||||
private final GameType type;
|
private final GameType type;
|
||||||
@@ -39,6 +51,6 @@ public enum GameList {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public GameType getType() {
|
public GameType getType() {
|
||||||
return type;
|
return this.type;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -9,7 +9,9 @@ import eu.mhsl.minenet.minigames.score.Score;
|
|||||||
import net.kyori.adventure.audience.Audience;
|
import net.kyori.adventure.audience.Audience;
|
||||||
import net.kyori.adventure.text.Component;
|
import net.kyori.adventure.text.Component;
|
||||||
import net.kyori.adventure.text.format.NamedTextColor;
|
import net.kyori.adventure.text.format.NamedTextColor;
|
||||||
|
import net.minestom.server.registry.DynamicRegistry;
|
||||||
import net.minestom.server.timer.ExecutionType;
|
import net.minestom.server.timer.ExecutionType;
|
||||||
|
import net.minestom.server.timer.Task;
|
||||||
import net.minestom.server.timer.TaskSchedule;
|
import net.minestom.server.timer.TaskSchedule;
|
||||||
import net.minestom.server.world.DimensionType;
|
import net.minestom.server.world.DimensionType;
|
||||||
|
|
||||||
@@ -19,10 +21,12 @@ import java.util.concurrent.CompletableFuture;
|
|||||||
public class StatelessGame extends Game {
|
public class StatelessGame extends Game {
|
||||||
private final String name;
|
private final String name;
|
||||||
private final Score score;
|
private final Score score;
|
||||||
|
private Task timeLimitTask;
|
||||||
|
|
||||||
private int timeLimit = 0;
|
private int timeLimit = 0;
|
||||||
private int timePlayed = 0;
|
private int timePlayed = 0;
|
||||||
public StatelessGame(DimensionType dimensionType, String gameName, Score score) {
|
|
||||||
|
public StatelessGame(DynamicRegistry.Key<DimensionType> dimensionType, String gameName, Score score) {
|
||||||
super(dimensionType);
|
super(dimensionType);
|
||||||
this.score = score;
|
this.score = score;
|
||||||
this.name = gameName;
|
this.name = gameName;
|
||||||
@@ -38,8 +42,12 @@ public class StatelessGame extends Game {
|
|||||||
|
|
||||||
public void setTimeLimit(int limit) {
|
public void setTimeLimit(int limit) {
|
||||||
this.timeLimit = limit;
|
this.timeLimit = limit;
|
||||||
if(timeLimit > 0) {
|
if(this.timeLimitTask != null) {
|
||||||
scheduler().submitTask(() -> {
|
this.timeLimitTask.cancel();
|
||||||
|
this.timePlayed = 0;
|
||||||
|
}
|
||||||
|
if(this.timeLimit > 0) {
|
||||||
|
this.timeLimitTask = scheduler().submitTask(() -> {
|
||||||
if(!isRunning || timeLimit == 0) return TaskSchedule.stop();
|
if(!isRunning || timeLimit == 0) return TaskSchedule.stop();
|
||||||
if(timeLimit <= timePlayed) {
|
if(timeLimit <= timePlayed) {
|
||||||
stop();
|
stop();
|
||||||
@@ -48,14 +56,14 @@ public class StatelessGame extends Game {
|
|||||||
|
|
||||||
int timeLeft = timeLimit - timePlayed;
|
int timeLeft = timeLimit - timePlayed;
|
||||||
switch (timeLeft) {
|
switch (timeLeft) {
|
||||||
case 60, 30, 10, 5, 4, 3, 2, 1 ->
|
case 90, 60, 30, 10, 5, 4, 3, 2, 1 ->
|
||||||
new ChatMessage(Icon.SCIENCE).appendStatic("Noch " + timeLeft + " Sekunden!").send(getPlayers());
|
new ChatMessage(Icon.SCIENCE).appendStatic("Noch " + timeLeft + " Sekunden!").send(getPlayers());
|
||||||
}
|
}
|
||||||
|
|
||||||
timePlayed++;
|
timePlayed++;
|
||||||
|
|
||||||
return TaskSchedule.seconds(1);
|
return TaskSchedule.seconds(1);
|
||||||
}, ExecutionType.SYNC);
|
}, ExecutionType.TICK_START);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -47,7 +47,7 @@ public class GameConfigurationInventory extends InteractableInventory {
|
|||||||
|
|
||||||
setClickableItem(
|
setClickableItem(
|
||||||
ItemStack.builder(Material.RED_WOOL)
|
ItemStack.builder(Material.RED_WOOL)
|
||||||
.displayName(
|
.customName(
|
||||||
TranslatedComponent.byId("common#back")
|
TranslatedComponent.byId("common#back")
|
||||||
.setColor(NamedTextColor.RED)
|
.setColor(NamedTextColor.RED)
|
||||||
.getAssembled(p)
|
.getAssembled(p)
|
||||||
@@ -62,7 +62,7 @@ public class GameConfigurationInventory extends InteractableInventory {
|
|||||||
|
|
||||||
setDummyItem(
|
setDummyItem(
|
||||||
ItemStack.builder(Material.NAME_TAG)
|
ItemStack.builder(Material.NAME_TAG)
|
||||||
.displayName(
|
.customName(
|
||||||
factory.name().setColor(NamedTextColor.GOLD).getAssembled(p)
|
factory.name().setColor(NamedTextColor.GOLD).getAssembled(p)
|
||||||
)
|
)
|
||||||
.build(),
|
.build(),
|
||||||
@@ -80,7 +80,7 @@ public class GameConfigurationInventory extends InteractableInventory {
|
|||||||
if(config == null) {
|
if(config == null) {
|
||||||
setDummyItem(
|
setDummyItem(
|
||||||
ItemStack.builder(Material.BARRIER)
|
ItemStack.builder(Material.BARRIER)
|
||||||
.displayName(
|
.customName(
|
||||||
TranslatedComponent.byId("room#noOption").setColor(NamedTextColor.RED).getAssembled(p)
|
TranslatedComponent.byId("room#noOption").setColor(NamedTextColor.RED).getAssembled(p)
|
||||||
)
|
)
|
||||||
.lore(
|
.lore(
|
||||||
@@ -130,7 +130,7 @@ public class GameConfigurationInventory extends InteractableInventory {
|
|||||||
|
|
||||||
setClickableItem(
|
setClickableItem(
|
||||||
ItemStack.builder(restrictionHandler.getWarnings(restrictionData).size() > 0 ? Material.YELLOW_WOOL : Material.GREEN_WOOL)
|
ItemStack.builder(restrictionHandler.getWarnings(restrictionData).size() > 0 ? Material.YELLOW_WOOL : Material.GREEN_WOOL)
|
||||||
.displayName(TranslatedComponent.byId("restriction#success").setColor(NamedTextColor.GREEN).getAssembled(p))
|
.customName(TranslatedComponent.byId("restriction#success").setColor(NamedTextColor.GREEN).getAssembled(p))
|
||||||
.lore(restrictionHandler.getWarnings(restrictionData).stream().map(translatedComponent -> translatedComponent.getAssembled(p)).collect(Collectors.toList()))
|
.lore(restrictionHandler.getWarnings(restrictionData).stream().map(translatedComponent -> translatedComponent.getAssembled(p)).collect(Collectors.toList()))
|
||||||
.build(),
|
.build(),
|
||||||
8,
|
8,
|
||||||
@@ -142,7 +142,7 @@ public class GameConfigurationInventory extends InteractableInventory {
|
|||||||
|
|
||||||
setClickableItem(
|
setClickableItem(
|
||||||
ItemStack.builder(Material.RED_WOOL)
|
ItemStack.builder(Material.RED_WOOL)
|
||||||
.displayName(TranslatedComponent.byId("restriction#fail").setColor(NamedTextColor.RED).getAssembled(p))
|
.customName(TranslatedComponent.byId("restriction#fail").setColor(NamedTextColor.RED).getAssembled(p))
|
||||||
.lore(
|
.lore(
|
||||||
restrictionHandler.getRestrictions()
|
restrictionHandler.getRestrictions()
|
||||||
.stream()
|
.stream()
|
||||||
|
@@ -9,6 +9,7 @@ import net.minestom.server.item.ItemStack;
|
|||||||
import net.minestom.server.item.Material;
|
import net.minestom.server.item.Material;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Objects;
|
||||||
|
|
||||||
public abstract class Option<T> {
|
public abstract class Option<T> {
|
||||||
private RestrictionHandler restrictionHandler;
|
private RestrictionHandler restrictionHandler;
|
||||||
@@ -25,7 +26,7 @@ public abstract class Option<T> {
|
|||||||
this.name = name;
|
this.name = name;
|
||||||
this.options = options;
|
this.options = options;
|
||||||
|
|
||||||
currentValue = options.get(0);
|
currentValue = options.getFirst();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setRestrictionHandler(RestrictionHandler restrictionHandler) {
|
public void setRestrictionHandler(RestrictionHandler restrictionHandler) {
|
||||||
@@ -44,11 +45,11 @@ public abstract class Option<T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public ItemStack getCurrent(Player p) {
|
public ItemStack getCurrent(Player p) {
|
||||||
int amount = Integer.parseInt(options.get(pointer).toString());
|
String value = options.get(pointer).toString();
|
||||||
return ItemStack.builder(item)
|
return ItemStack.builder(item)
|
||||||
.displayName(name.getAssembled(p))
|
.customName(name.getAssembled(p))
|
||||||
.lore(TranslatedComponent.byId("optionCommon#value").setColor(NamedTextColor.GOLD).getAssembled(p)
|
.lore(TranslatedComponent.byId("optionCommon#value").setColor(NamedTextColor.GOLD).getAssembled(p)
|
||||||
.append(Component.text(": ")).append(Component.text(amount)))
|
.append(Component.text(": ")).append(Component.text(value)))
|
||||||
.build();
|
.build();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -56,6 +57,10 @@ public abstract class Option<T> {
|
|||||||
return Integer.parseInt(getAsString());
|
return Integer.parseInt(getAsString());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean getAsBoolean() {
|
||||||
|
return Objects.equals(getAsString(), "true") || Objects.equals(getAsString(), "1");
|
||||||
|
}
|
||||||
|
|
||||||
public String getAsString() {
|
public String getAsString() {
|
||||||
return currentValue.toString();
|
return currentValue.toString();
|
||||||
}
|
}
|
||||||
|
@@ -6,8 +6,8 @@ import net.minestom.server.item.Material;
|
|||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
public class BoolOption extends Option<Boolean> {
|
public class BoolOption extends Option<String> {
|
||||||
public BoolOption(String id, Material item, TranslatedComponent name) {
|
public BoolOption(String id, Material item, TranslatedComponent name) {
|
||||||
super(id, item, name, List.of(true, false));
|
super(id, item, name, List.of("true", "false"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -17,8 +17,8 @@ import net.minestom.server.event.player.PlayerMoveEvent;
|
|||||||
import net.minestom.server.event.player.PlayerTickEvent;
|
import net.minestom.server.event.player.PlayerTickEvent;
|
||||||
import net.minestom.server.instance.batch.AbsoluteBlockBatch;
|
import net.minestom.server.instance.batch.AbsoluteBlockBatch;
|
||||||
import net.minestom.server.instance.block.Block;
|
import net.minestom.server.instance.block.Block;
|
||||||
|
import net.minestom.server.network.packet.server.play.ParticlePacket;
|
||||||
import net.minestom.server.particle.Particle;
|
import net.minestom.server.particle.Particle;
|
||||||
import net.minestom.server.particle.ParticleCreator;
|
|
||||||
import net.minestom.server.timer.ExecutionType;
|
import net.minestom.server.timer.ExecutionType;
|
||||||
import net.minestom.server.timer.TaskSchedule;
|
import net.minestom.server.timer.TaskSchedule;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
@@ -38,7 +38,7 @@ public class AcidRain extends StatelessGame {
|
|||||||
.setFrequency(0.09)
|
.setFrequency(0.09)
|
||||||
.build();
|
.build();
|
||||||
public AcidRain() {
|
public AcidRain() {
|
||||||
super(Dimension.OVERWORLD.DIMENSION, "acidRain", new LastWinsScore());
|
super(Dimension.OVERWORLD.key, "acidRain", new LastWinsScore());
|
||||||
setGenerator(
|
setGenerator(
|
||||||
new CircularPlateTerrainGenerator(radius)
|
new CircularPlateTerrainGenerator(radius)
|
||||||
.setPlateHeight(50)
|
.setPlateHeight(50)
|
||||||
@@ -62,13 +62,13 @@ public class AcidRain extends StatelessGame {
|
|||||||
MinecraftServer.getSchedulerManager().submitTask(() -> {
|
MinecraftServer.getSchedulerManager().submitTask(() -> {
|
||||||
|
|
||||||
getPlayers().forEach(player -> {
|
getPlayers().forEach(player -> {
|
||||||
player.sendPacket(ParticleCreator.createParticlePacket(Particle.SNEEZE, 0, 60, 0, radius, radius, radius, 500));
|
player.sendPacket(new ParticlePacket(Particle.SNEEZE, 0, 60, 0, radius, radius, radius, 1f, 500));
|
||||||
player.sendPacket(ParticleCreator.createParticlePacket(Particle.ITEM_SLIME, 0, 60, 0, radius, radius, radius, 500));
|
player.sendPacket(new ParticlePacket(Particle.ITEM_SLIME, 0, 60, 0, radius, radius, radius, 1f, 500));
|
||||||
});
|
});
|
||||||
|
|
||||||
if(!isRunning) return TaskSchedule.stop();
|
if(!isRunning) return TaskSchedule.stop();
|
||||||
return TaskSchedule.millis(100);
|
return TaskSchedule.millis(100);
|
||||||
}, ExecutionType.ASYNC);
|
}, ExecutionType.TICK_END);
|
||||||
|
|
||||||
MinecraftServer.getSchedulerManager().submitTask(() -> {
|
MinecraftServer.getSchedulerManager().submitTask(() -> {
|
||||||
generationOffset++;
|
generationOffset++;
|
||||||
@@ -76,7 +76,7 @@ public class AcidRain extends StatelessGame {
|
|||||||
|
|
||||||
if(!isRunning) return TaskSchedule.stop();
|
if(!isRunning) return TaskSchedule.stop();
|
||||||
return TaskSchedule.millis((long) NumberUtil.map(50 - difficulty, 0, 50, 100, 1000));
|
return TaskSchedule.millis((long) NumberUtil.map(50 - difficulty, 0, 50, 100, 1000));
|
||||||
}, ExecutionType.ASYNC);
|
}, ExecutionType.TICK_END);
|
||||||
|
|
||||||
MinecraftServer.getSchedulerManager().submitTask(() -> {
|
MinecraftServer.getSchedulerManager().submitTask(() -> {
|
||||||
difficulty++;
|
difficulty++;
|
||||||
|
@@ -15,6 +15,11 @@ public class AcidRainFactory implements GameFactory {
|
|||||||
return TranslatedComponent.byId("game_AcidRain#name");
|
return TranslatedComponent.byId("game_AcidRain#name");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TranslatedComponent description() {
|
||||||
|
return TranslatedComponent.byId("game_AcidRain#description");
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Material symbol() {
|
public Material symbol() {
|
||||||
return Material.SLIME_BALL;
|
return Material.SLIME_BALL;
|
||||||
|
@@ -0,0 +1,111 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.anvilRun;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.StatelessGame;
|
||||||
|
import eu.mhsl.minenet.minigames.score.LastWinsScore;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.Dimension;
|
||||||
|
import eu.mhsl.minenet.minigames.util.BatchUtil;
|
||||||
|
import eu.mhsl.minenet.minigames.world.generator.terrain.CircularPlateTerrainGenerator;
|
||||||
|
import net.minestom.server.MinecraftServer;
|
||||||
|
import net.minestom.server.coordinate.Pos;
|
||||||
|
import net.minestom.server.entity.Entity;
|
||||||
|
import net.minestom.server.entity.EntityType;
|
||||||
|
import net.minestom.server.entity.metadata.other.FallingBlockMeta;
|
||||||
|
import net.minestom.server.event.entity.EntityTickEvent;
|
||||||
|
import net.minestom.server.event.player.PlayerMoveEvent;
|
||||||
|
import net.minestom.server.instance.batch.AbsoluteBlockBatch;
|
||||||
|
import net.minestom.server.instance.block.Block;
|
||||||
|
import net.minestom.server.timer.Scheduler;
|
||||||
|
import net.minestom.server.timer.TaskSchedule;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.concurrent.CompletableFuture;
|
||||||
|
|
||||||
|
class AnvilRun extends StatelessGame {
|
||||||
|
|
||||||
|
final int spawnHeight = 30;
|
||||||
|
final int radius;
|
||||||
|
int anvilsPerSecond;
|
||||||
|
final int seconds;
|
||||||
|
final int anvilHeight = 200;
|
||||||
|
final List<Pos> anvilSpawnPositions = new ArrayList<>();
|
||||||
|
|
||||||
|
public AnvilRun(int radius, int seconds) {
|
||||||
|
super(Dimension.OVERWORLD.key, "Anvil Run", new LastWinsScore());
|
||||||
|
this.radius = radius;
|
||||||
|
this.seconds = seconds;
|
||||||
|
this.setGenerator(new CircularPlateTerrainGenerator(radius));
|
||||||
|
|
||||||
|
eventNode().addListener(EntityTickEvent.class, this::onEntityTick);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onLoad(@NotNull CompletableFuture<Void> callback) {
|
||||||
|
AbsoluteBlockBatch batch = new AbsoluteBlockBatch();
|
||||||
|
|
||||||
|
for(int x = -radius; x <= radius; x++) {
|
||||||
|
for (int z = -radius; z <= radius; z++) {
|
||||||
|
if(new Pos(x, 0, z).distance(new Pos(0, 0, 0)) > radius) continue;
|
||||||
|
|
||||||
|
anvilSpawnPositions.add(new Pos(x+0.5, anvilHeight, z+0.5));
|
||||||
|
|
||||||
|
batch.setBlock(x, spawnHeight-1, z, Block.SNOW_BLOCK);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.anvilsPerSecond = anvilSpawnPositions.size() / this.seconds;
|
||||||
|
Collections.shuffle(anvilSpawnPositions);
|
||||||
|
|
||||||
|
BatchUtil.loadAndApplyBatch(batch, this, () -> callback.complete(null));
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void spawnAnvil(Pos spawnPosition) {
|
||||||
|
Entity anvil = new Entity(EntityType.FALLING_BLOCK);
|
||||||
|
((FallingBlockMeta) anvil.getEntityMeta()).setBlock(Block.ANVIL);
|
||||||
|
anvil.setInstance(this, spawnPosition);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onStart() {
|
||||||
|
super.onStart();
|
||||||
|
|
||||||
|
Scheduler scheduler = MinecraftServer.getSchedulerManager();
|
||||||
|
for(int i=0; i<this.anvilSpawnPositions.size(); i++) {
|
||||||
|
final int j = i;
|
||||||
|
scheduler.scheduleTask(
|
||||||
|
() -> spawnAnvil(this.anvilSpawnPositions.get(j)),
|
||||||
|
TaskSchedule.millis(
|
||||||
|
(long) Math.floor((double) i/this.anvilsPerSecond * 1000)
|
||||||
|
),
|
||||||
|
TaskSchedule.stop()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onPlayerMove(@NotNull PlayerMoveEvent playerMoveEvent) {
|
||||||
|
super.onPlayerMove(playerMoveEvent);
|
||||||
|
if(isBeforeBeginning && playerMoveEvent.getNewPosition().y() < spawnHeight - 2) {
|
||||||
|
playerMoveEvent.setCancelled(true);
|
||||||
|
playerMoveEvent.getPlayer().teleport(getSpawn());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if(playerMoveEvent.getNewPosition().y() < spawnHeight - 2) getScore().insertResult(playerMoveEvent.getPlayer());
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void onEntityTick(@NotNull EntityTickEvent entityTickEvent) {
|
||||||
|
if(!entityTickEvent.getEntity().getEntityType().equals(EntityType.FALLING_BLOCK)) return;
|
||||||
|
Pos anvilPosition = entityTickEvent.getEntity().getPosition();
|
||||||
|
if(anvilPosition.y() > spawnHeight + 0.5) return;
|
||||||
|
if(anvilPosition.y() < spawnHeight - 3) entityTickEvent.getEntity().remove();
|
||||||
|
if(this.getBlock(anvilPosition.withY(spawnHeight-1)).isAir()) return;
|
||||||
|
this.setBlock(anvilPosition.withY(spawnHeight-1), Block.AIR);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Pos getSpawn() {
|
||||||
|
return new Pos(0, spawnHeight, 0);
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,41 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.anvilRun;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.Game;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.GameFactory;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.Option;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.ConfigManager;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.common.NumericOption;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.room.Room;
|
||||||
|
import eu.mhsl.minenet.minigames.message.component.TranslatedComponent;
|
||||||
|
import net.minestom.server.item.Material;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
public class AnvilRunFactory implements GameFactory {
|
||||||
|
@Override
|
||||||
|
public TranslatedComponent name() {
|
||||||
|
return TranslatedComponent.byId("game_AnvilRun#name");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TranslatedComponent description() {
|
||||||
|
return TranslatedComponent.byId("game_AnvilRun#description");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ConfigManager configuration() {
|
||||||
|
return new ConfigManager()
|
||||||
|
.addOption(new NumericOption("radius", Material.HEART_OF_THE_SEA, TranslatedComponent.byId("optionCommon#radius"), 5, 10, 15, 20, 25, 30))
|
||||||
|
.addOption(new NumericOption("seconds", Material.CLOCK, TranslatedComponent.byId("optionCommon#seconds"), 10, 30, 60, 90, 120));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Game manufacture(Room parent, Map<String, Option<?>> configuration) {
|
||||||
|
return new AnvilRun(configuration.get("radius").getAsInt(), configuration.get("seconds").getAsInt()).setParent(parent);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Material symbol() {
|
||||||
|
return Material.ANVIL;
|
||||||
|
}
|
||||||
|
}
|
@@ -8,7 +8,7 @@ import org.jetbrains.annotations.NotNull;
|
|||||||
|
|
||||||
public class Backrooms extends StatelessGame {
|
public class Backrooms extends StatelessGame {
|
||||||
public Backrooms() {
|
public Backrooms() {
|
||||||
super(Dimension.NETHER.DIMENSION, "Backrooms", new NoScore());
|
super(Dimension.NETHER.key, "Backrooms", new NoScore());
|
||||||
BackroomsGenerator generator = new BackroomsGenerator();
|
BackroomsGenerator generator = new BackroomsGenerator();
|
||||||
setGenerator(unit -> generator.generateRoom(unit, 50));
|
setGenerator(unit -> generator.generateRoom(unit, 50));
|
||||||
}
|
}
|
||||||
|
@@ -21,7 +21,7 @@ import net.minestom.server.event.item.PickupItemEvent;
|
|||||||
import net.minestom.server.event.player.PlayerBlockBreakEvent;
|
import net.minestom.server.event.player.PlayerBlockBreakEvent;
|
||||||
import net.minestom.server.event.player.PlayerBlockPlaceEvent;
|
import net.minestom.server.event.player.PlayerBlockPlaceEvent;
|
||||||
import net.minestom.server.event.player.PlayerMoveEvent;
|
import net.minestom.server.event.player.PlayerMoveEvent;
|
||||||
import net.minestom.server.instance.AnvilLoader;
|
import net.minestom.server.instance.anvil.AnvilLoader;
|
||||||
import net.minestom.server.item.ItemStack;
|
import net.minestom.server.item.ItemStack;
|
||||||
import net.minestom.server.item.Material;
|
import net.minestom.server.item.Material;
|
||||||
import net.minestom.server.timer.ExecutionType;
|
import net.minestom.server.timer.ExecutionType;
|
||||||
@@ -38,7 +38,7 @@ public class Bedwars extends StatelessGame {
|
|||||||
|
|
||||||
|
|
||||||
public Bedwars() throws IOException {
|
public Bedwars() throws IOException {
|
||||||
super(Dimension.OVERWORLD.DIMENSION, "Bedwars", new LastWinsScore());
|
super(Dimension.OVERWORLD.key, "Bedwars", new LastWinsScore());
|
||||||
setChunkLoader(new AnvilLoader(Resource.GAME_MAP.getPath().resolve("bedwars/test")));
|
setChunkLoader(new AnvilLoader(Resource.GAME_MAP.getPath().resolve("bedwars/test")));
|
||||||
|
|
||||||
Configuration config = ConfigurationProvider.getProvider(YamlConfiguration.class).load(Resource.GAME_MAP.getPath().resolve("bedwars/test/config.yml").toFile());
|
Configuration config = ConfigurationProvider.getProvider(YamlConfiguration.class).load(Resource.GAME_MAP.getPath().resolve("bedwars/test/config.yml").toFile());
|
||||||
@@ -78,7 +78,7 @@ public class Bedwars extends StatelessGame {
|
|||||||
});
|
});
|
||||||
|
|
||||||
return TaskSchedule.stop();
|
return TaskSchedule.stop();
|
||||||
}, ExecutionType.SYNC);
|
}, ExecutionType.TICK_END);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -6,31 +6,22 @@ import eu.mhsl.minenet.minigames.message.component.TranslatedComponent;
|
|||||||
import eu.mhsl.minenet.minigames.score.LastWinsScore;
|
import eu.mhsl.minenet.minigames.score.LastWinsScore;
|
||||||
import eu.mhsl.minenet.minigames.util.BatchUtil;
|
import eu.mhsl.minenet.minigames.util.BatchUtil;
|
||||||
import eu.mhsl.minenet.minigames.util.GeneratorUtils;
|
import eu.mhsl.minenet.minigames.util.GeneratorUtils;
|
||||||
import io.github.bloepiloepi.pvp.events.ProjectileHitEvent;
|
|
||||||
import io.github.bloepiloepi.pvp.projectile.CustomEntityProjectile;
|
|
||||||
import net.minestom.server.coordinate.Point;
|
import net.minestom.server.coordinate.Point;
|
||||||
import net.minestom.server.coordinate.Pos;
|
import net.minestom.server.coordinate.Pos;
|
||||||
import net.minestom.server.coordinate.Vec;
|
import net.minestom.server.coordinate.Vec;
|
||||||
import net.minestom.server.entity.*;
|
import net.minestom.server.entity.*;
|
||||||
import net.minestom.server.entity.metadata.arrow.ArrowMeta;
|
|
||||||
import net.minestom.server.entity.metadata.other.PrimedTntMeta;
|
import net.minestom.server.entity.metadata.other.PrimedTntMeta;
|
||||||
import net.minestom.server.event.EventListener;
|
import net.minestom.server.event.EventListener;
|
||||||
import net.minestom.server.event.item.ItemUpdateStateEvent;
|
import net.minestom.server.event.entity.projectile.ProjectileCollideWithBlockEvent;
|
||||||
import net.minestom.server.event.player.PlayerItemAnimationEvent;
|
import net.minestom.server.event.entity.projectile.ProjectileCollideWithEntityEvent;
|
||||||
import net.minestom.server.event.player.PlayerMoveEvent;
|
import net.minestom.server.event.player.PlayerMoveEvent;
|
||||||
import net.minestom.server.instance.batch.AbsoluteBlockBatch;
|
import net.minestom.server.instance.batch.AbsoluteBlockBatch;
|
||||||
import net.minestom.server.instance.block.Block;
|
import net.minestom.server.instance.block.Block;
|
||||||
import net.minestom.server.item.Enchantment;
|
|
||||||
import net.minestom.server.item.ItemHideFlag;
|
|
||||||
import net.minestom.server.item.ItemStack;
|
import net.minestom.server.item.ItemStack;
|
||||||
import net.minestom.server.item.Material;
|
import net.minestom.server.item.Material;
|
||||||
import net.minestom.server.tag.Tag;
|
import net.minestom.server.tag.Tag;
|
||||||
import net.minestom.server.utils.MathUtils;
|
|
||||||
import net.minestom.server.utils.time.TimeUnit;
|
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
import java.time.Duration;
|
|
||||||
import java.util.Objects;
|
|
||||||
import java.util.concurrent.CompletableFuture;
|
import java.util.concurrent.CompletableFuture;
|
||||||
|
|
||||||
public class BowSpleef extends StatelessGame {
|
public class BowSpleef extends StatelessGame {
|
||||||
@@ -39,48 +30,48 @@ public class BowSpleef extends StatelessGame {
|
|||||||
private final int radius = 30;
|
private final int radius = 30;
|
||||||
private final int totalElevation = 50;
|
private final int totalElevation = 50;
|
||||||
public BowSpleef() {
|
public BowSpleef() {
|
||||||
super(Dimension.OVERWORLD.DIMENSION, "bowSpleef", new LastWinsScore());
|
super(Dimension.OVERWORLD.key, "bowSpleef", new LastWinsScore());
|
||||||
|
|
||||||
|
// eventNode().addListener( TODO implement bow mechanism
|
||||||
|
// EventListener
|
||||||
|
// .builder(PlayerItemAnimationEvent.class)
|
||||||
|
// .handler(playerItemAnimationEvent -> playerItemAnimationEvent.getPlayer().setTag(CHARGE_BOW_SINCE, System.currentTimeMillis()))
|
||||||
|
// .filter(playerItemAnimationEvent -> playerItemAnimationEvent.getItemAnimationType() == PlayerItemAnimationEvent.ItemAnimationType.BOW)
|
||||||
|
// .build()
|
||||||
|
// );
|
||||||
|
//
|
||||||
|
// eventNode().addListener(
|
||||||
|
// EventListener
|
||||||
|
// .builder(ItemUpdateStateEvent.class)
|
||||||
|
// .handler(event -> {
|
||||||
|
// final Player player = event.getPlayer();
|
||||||
|
// final double chargedFor = (System.currentTimeMillis() - player.getTag(CHARGE_BOW_SINCE)) / 1000D;
|
||||||
|
// final double power = MathUtils.clamp((chargedFor * chargedFor + 2 * chargedFor) / 2D, 0, 1);
|
||||||
|
//
|
||||||
|
// if (power > 0.2) {
|
||||||
|
// final CustomEntityProjectile projectile = new CustomEntityProjectile(player, EntityType.ARROW);
|
||||||
|
// final ArrowMeta meta = (ArrowMeta) projectile.getEntityMeta();
|
||||||
|
// meta.setCritical(power >= 0.9);
|
||||||
|
// projectile.scheduleRemove(Duration.of(100, TimeUnit.SERVER_TICK));
|
||||||
|
// meta.setOnFire(true);
|
||||||
|
//
|
||||||
|
// final Pos position = player.getPosition().add(0, player.getEyeHeight(), 0);
|
||||||
|
// projectile.setInstance(Objects.requireNonNull(player.getInstance()), position);
|
||||||
|
//
|
||||||
|
// final Vec direction = projectile.getPosition().direction();
|
||||||
|
// projectile.shootFrom(position.add(direction).sub(0, 0.2, 0), power * 3, 1.0);
|
||||||
|
// projectile.setTag(ARROW_FIRST_HIT, true);
|
||||||
|
// }
|
||||||
|
// })
|
||||||
|
// .filter(itemUpdateStateEvent -> itemUpdateStateEvent.getItemStack().material() == Material.BOW)
|
||||||
|
// .build()
|
||||||
|
// );
|
||||||
|
|
||||||
eventNode().addListener(
|
eventNode().addListener(
|
||||||
EventListener
|
EventListener
|
||||||
.builder(PlayerItemAnimationEvent.class)
|
.builder(ProjectileCollideWithBlockEvent.class)
|
||||||
.handler(playerItemAnimationEvent -> playerItemAnimationEvent.getPlayer().setTag(CHARGE_BOW_SINCE, System.currentTimeMillis()))
|
|
||||||
.filter(playerItemAnimationEvent -> playerItemAnimationEvent.getItemAnimationType() == PlayerItemAnimationEvent.ItemAnimationType.BOW)
|
|
||||||
.build()
|
|
||||||
);
|
|
||||||
|
|
||||||
eventNode().addListener(
|
|
||||||
EventListener
|
|
||||||
.builder(ItemUpdateStateEvent.class)
|
|
||||||
.handler(event -> {
|
|
||||||
final Player player = event.getPlayer();
|
|
||||||
final double chargedFor = (System.currentTimeMillis() - player.getTag(CHARGE_BOW_SINCE)) / 1000D;
|
|
||||||
final double power = MathUtils.clamp((chargedFor * chargedFor + 2 * chargedFor) / 2D, 0, 1);
|
|
||||||
|
|
||||||
if (power > 0.2) {
|
|
||||||
final CustomEntityProjectile projectile = new CustomEntityProjectile(player, EntityType.ARROW, true);
|
|
||||||
final ArrowMeta meta = (ArrowMeta) projectile.getEntityMeta();
|
|
||||||
meta.setCritical(power >= 0.9);
|
|
||||||
projectile.scheduleRemove(Duration.of(100, TimeUnit.SERVER_TICK));
|
|
||||||
projectile.setOnFire(true);
|
|
||||||
|
|
||||||
final Pos position = player.getPosition().add(0, player.getEyeHeight(), 0);
|
|
||||||
projectile.setInstance(Objects.requireNonNull(player.getInstance()), position);
|
|
||||||
|
|
||||||
final Vec direction = projectile.getPosition().direction();
|
|
||||||
projectile.shoot(position.add(direction).sub(0, 0.2, 0), power * 3, 1.0);
|
|
||||||
projectile.setTag(ARROW_FIRST_HIT, true);
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.filter(itemUpdateStateEvent -> itemUpdateStateEvent.getItemStack().material() == Material.BOW)
|
|
||||||
.build()
|
|
||||||
);
|
|
||||||
|
|
||||||
eventNode().addListener(
|
|
||||||
EventListener
|
|
||||||
.builder(ProjectileHitEvent.ProjectileBlockHitEvent.class)
|
|
||||||
.handler(projectileBlockHitEvent -> {
|
.handler(projectileBlockHitEvent -> {
|
||||||
CustomEntityProjectile projectile = projectileBlockHitEvent.getEntity();
|
Entity projectile = projectileBlockHitEvent.getEntity();
|
||||||
if(!projectile.getTag(ARROW_FIRST_HIT)) {
|
if(!projectile.getTag(ARROW_FIRST_HIT)) {
|
||||||
projectile.remove();
|
projectile.remove();
|
||||||
return;
|
return;
|
||||||
@@ -108,7 +99,7 @@ public class BowSpleef extends StatelessGame {
|
|||||||
|
|
||||||
eventNode().addListener(
|
eventNode().addListener(
|
||||||
EventListener
|
EventListener
|
||||||
.builder(ProjectileHitEvent.ProjectileEntityHitEvent.class)
|
.builder(ProjectileCollideWithEntityEvent.class)
|
||||||
.handler(projectileEntityHitEvent -> projectileEntityHitEvent.setCancelled(true))
|
.handler(projectileEntityHitEvent -> projectileEntityHitEvent.setCancelled(true))
|
||||||
.build()
|
.build()
|
||||||
);
|
);
|
||||||
@@ -135,9 +126,8 @@ public class BowSpleef extends StatelessGame {
|
|||||||
0,
|
0,
|
||||||
ItemStack
|
ItemStack
|
||||||
.builder(Material.BOW)
|
.builder(Material.BOW)
|
||||||
.displayName(TranslatedComponent.byId("bow").getAssembled(player))
|
.customName(TranslatedComponent.byId("bow").getAssembled(player))
|
||||||
.meta(builder -> builder.enchantment(Enchantment.FLAME, (short) 1).build())
|
.glowing(true)
|
||||||
.meta(builder -> builder.hideFlag(ItemHideFlag.HIDE_ENCHANTS))
|
|
||||||
.build()
|
.build()
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
@@ -12,7 +12,12 @@ import java.util.Map;
|
|||||||
public class BowSpleefFactory implements GameFactory {
|
public class BowSpleefFactory implements GameFactory {
|
||||||
@Override
|
@Override
|
||||||
public TranslatedComponent name() {
|
public TranslatedComponent name() {
|
||||||
return TranslatedComponent.byId("");
|
return TranslatedComponent.byId("game_BowSpleef#name");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TranslatedComponent description() {
|
||||||
|
return TranslatedComponent.byId("game_BowSpleef#description");
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@@ -6,10 +6,9 @@ import eu.mhsl.minenet.minigames.util.BatchUtil;
|
|||||||
import eu.mhsl.minenet.minigames.instance.Dimension;
|
import eu.mhsl.minenet.minigames.instance.Dimension;
|
||||||
import eu.mhsl.minenet.minigames.world.BlockPallet;
|
import eu.mhsl.minenet.minigames.world.BlockPallet;
|
||||||
import eu.mhsl.minenet.minigames.world.generator.terrain.CircularPlateTerrainGenerator;
|
import eu.mhsl.minenet.minigames.world.generator.terrain.CircularPlateTerrainGenerator;
|
||||||
import io.github.bloepiloepi.pvp.config.AttackConfig;
|
import io.github.togar2.pvp.feature.CombatFeatures;
|
||||||
import io.github.bloepiloepi.pvp.config.DamageConfig;
|
|
||||||
import io.github.bloepiloepi.pvp.config.PvPConfig;
|
|
||||||
import net.minestom.server.coordinate.Pos;
|
import net.minestom.server.coordinate.Pos;
|
||||||
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.event.player.PlayerMoveEvent;
|
import net.minestom.server.event.player.PlayerMoveEvent;
|
||||||
import net.minestom.server.instance.batch.AbsoluteBlockBatch;
|
import net.minestom.server.instance.batch.AbsoluteBlockBatch;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
@@ -23,17 +22,18 @@ class Deathcube extends StatelessGame {
|
|||||||
final int percentage;
|
final int percentage;
|
||||||
|
|
||||||
public Deathcube(int radius, int height, int percentage, int pvpEnabled) {
|
public Deathcube(int radius, int height, int percentage, int pvpEnabled) {
|
||||||
super(Dimension.THE_END.DIMENSION, "Deathcube", new FirstWinsScore());
|
super(Dimension.THE_END.key, "Deathcube", new FirstWinsScore());
|
||||||
this.radius = radius;
|
this.radius = radius;
|
||||||
this.height = height + 49;
|
this.height = height + 49;
|
||||||
this.percentage = percentage;
|
this.percentage = percentage;
|
||||||
this.setGenerator(new CircularPlateTerrainGenerator(radius+10).setPlateHeight(50));
|
this.setGenerator(new CircularPlateTerrainGenerator(radius+10).setPlateHeight(50));
|
||||||
|
|
||||||
if(pvpEnabled == 1) eventNode().addChild(
|
if(pvpEnabled == 1) eventNode().addChild(
|
||||||
PvPConfig.emptyBuilder()
|
CombatFeatures.empty()
|
||||||
.damage(DamageConfig.legacyBuilder().fallDamage(false))
|
.add(CombatFeatures.VANILLA_ATTACK)
|
||||||
.attack(AttackConfig.legacyBuilder().attackCooldown(true))
|
.add(CombatFeatures.VANILLA_DAMAGE)
|
||||||
.build().createNode()
|
.add(CombatFeatures.VANILLA_KNOCKBACK)
|
||||||
|
.build().createNode()
|
||||||
);
|
);
|
||||||
System.out.println(radius);
|
System.out.println(radius);
|
||||||
}
|
}
|
||||||
@@ -60,12 +60,20 @@ class Deathcube extends StatelessGame {
|
|||||||
@Override
|
@Override
|
||||||
protected void onPlayerMove(@NotNull PlayerMoveEvent playerMoveEvent) {
|
protected void onPlayerMove(@NotNull PlayerMoveEvent playerMoveEvent) {
|
||||||
super.onPlayerMove(playerMoveEvent);
|
super.onPlayerMove(playerMoveEvent);
|
||||||
if(isBeforeBeginning) if(playerMoveEvent.getNewPosition().y() > 51.5) playerMoveEvent.setCancelled(true);
|
if(playerMoveEvent.getNewPosition().y() < 48) {
|
||||||
|
playerMoveEvent.setCancelled(true);
|
||||||
|
playerMoveEvent.getPlayer().teleport(getSpawn());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if(isBeforeBeginning && playerMoveEvent.getNewPosition().y() > 51.5) {
|
||||||
|
playerMoveEvent.setCancelled(true);
|
||||||
|
return;
|
||||||
|
}
|
||||||
if(playerMoveEvent.getNewPosition().y() > height) getScore().insertResult(playerMoveEvent.getPlayer());
|
if(playerMoveEvent.getNewPosition().y() > height) getScore().insertResult(playerMoveEvent.getPlayer());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Pos getSpawn() {
|
public Pos getSpawn() {
|
||||||
return new Pos(0, 50, 30);
|
return new Pos(0, 50, -(radius+5));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -28,7 +28,7 @@ public class DeathcubeFactory implements GameFactory {
|
|||||||
.addOption(new NumericOption("radius", Material.HEART_OF_THE_SEA, TranslatedComponent.byId("optionCommon#radius"), 10, 20, 30))
|
.addOption(new NumericOption("radius", Material.HEART_OF_THE_SEA, TranslatedComponent.byId("optionCommon#radius"), 10, 20, 30))
|
||||||
.addOption(new NumericOption("height", Material.SCAFFOLDING, TranslatedComponent.byId("optionCommon#height"), 10, 30, 50))
|
.addOption(new NumericOption("height", Material.SCAFFOLDING, TranslatedComponent.byId("optionCommon#height"), 10, 30, 50))
|
||||||
.addOption(new NumericOption("percentage", Material.COBWEB, TranslatedComponent.byId("game_Deathcube#optionPercentageBlocks"), 5, 7, 9, 11, 13))
|
.addOption(new NumericOption("percentage", Material.COBWEB, TranslatedComponent.byId("game_Deathcube#optionPercentageBlocks"), 5, 7, 9, 11, 13))
|
||||||
.addOption(new NumericOption("pvpEnabled", Material.STICK, TranslatedComponent.byId("game_Deathcube#optionPvpEnabled"), 1, 0));
|
.addOption(new NumericOption("pvpEnabled", Material.STICK, TranslatedComponent.byId("game_Deathcube#optionPvpEnabled"), 0, 1));
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -17,6 +17,7 @@ import net.kyori.adventure.text.format.NamedTextColor;
|
|||||||
import net.minestom.server.coordinate.Point;
|
import net.minestom.server.coordinate.Point;
|
||||||
import net.minestom.server.coordinate.Pos;
|
import net.minestom.server.coordinate.Pos;
|
||||||
import net.minestom.server.coordinate.Vec;
|
import net.minestom.server.coordinate.Vec;
|
||||||
|
import net.minestom.server.entity.EquipmentSlot;
|
||||||
import net.minestom.server.entity.GameMode;
|
import net.minestom.server.entity.GameMode;
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.event.player.*;
|
import net.minestom.server.event.player.*;
|
||||||
@@ -26,7 +27,6 @@ import net.minestom.server.item.ItemStack;
|
|||||||
import net.minestom.server.item.Material;
|
import net.minestom.server.item.Material;
|
||||||
import net.minestom.server.network.packet.server.play.ParticlePacket;
|
import net.minestom.server.network.packet.server.play.ParticlePacket;
|
||||||
import net.minestom.server.particle.Particle;
|
import net.minestom.server.particle.Particle;
|
||||||
import net.minestom.server.particle.ParticleCreator;
|
|
||||||
import net.minestom.server.sound.SoundEvent;
|
import net.minestom.server.sound.SoundEvent;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
@@ -40,6 +40,7 @@ public class ElytraRace extends StatelessGame {
|
|||||||
private final ValeGenerator vale = new ValeGenerator();
|
private final ValeGenerator vale = new ValeGenerator();
|
||||||
|
|
||||||
private final int gameHeight = 0;
|
private final int gameHeight = 0;
|
||||||
|
private final int seaLevel = -55;
|
||||||
private final int ringSpacing = 100;
|
private final int ringSpacing = 100;
|
||||||
private final int ringCount;
|
private final int ringCount;
|
||||||
|
|
||||||
@@ -57,12 +58,12 @@ public class ElytraRace extends StatelessGame {
|
|||||||
private final Map<Player, CheckPointData> playerCheckpoints = new HashMap<>();
|
private final Map<Player, CheckPointData> playerCheckpoints = new HashMap<>();
|
||||||
|
|
||||||
public ElytraRace(int ringCount) {
|
public ElytraRace(int ringCount) {
|
||||||
super(Dimension.OVERWORLD.DIMENSION, "ElytraRace", new FirstWinsScore());
|
super(Dimension.OVERWORLD.key, "ElytraRace", new FirstWinsScore());
|
||||||
|
|
||||||
this.ringCount = ringCount;
|
this.ringCount = ringCount;
|
||||||
|
|
||||||
setGenerator(vale);
|
setGenerator(vale);
|
||||||
vale.setCalculateSeaLevel(point -> -55);
|
vale.setCalculateSeaLevel(point -> seaLevel);
|
||||||
vale.setXShiftMultiplier(integer -> NumberUtil.map(integer, 50, 500, 0, 1));
|
vale.setXShiftMultiplier(integer -> NumberUtil.map(integer, 50, 500, 0, 1));
|
||||||
vale.addMixIn(new PlaneTerrainGenerator(gameHeight, Block.BARRIER));
|
vale.addMixIn(new PlaneTerrainGenerator(gameHeight, Block.BARRIER));
|
||||||
|
|
||||||
@@ -106,9 +107,7 @@ public class ElytraRace extends StatelessGame {
|
|||||||
@Override
|
@Override
|
||||||
protected void onLoad(@NotNull CompletableFuture<Void> callback) {
|
protected void onLoad(@NotNull CompletableFuture<Void> callback) {
|
||||||
Point spawnpoint = new Pos(vale.getXShiftAtZ(0), -46, 0);
|
Point spawnpoint = new Pos(vale.getXShiftAtZ(0), -46, 0);
|
||||||
GeneratorUtils.iterateArea(spawnpoint.sub(2, 0, 2), spawnpoint.add(2, 0, 2), point -> {
|
GeneratorUtils.iterateArea(spawnpoint.sub(5, 0, 5), spawnpoint.add(5, 0, 5), point -> setBlock(point, BlockPallet.STREET.rnd()));
|
||||||
setBlock(point, BlockPallet.STREET.rnd());
|
|
||||||
});
|
|
||||||
|
|
||||||
generateRing(ringSpacing);
|
generateRing(ringSpacing);
|
||||||
generateRing(ringSpacing * 2);
|
generateRing(ringSpacing * 2);
|
||||||
@@ -118,9 +117,9 @@ public class ElytraRace extends StatelessGame {
|
|||||||
@Override
|
@Override
|
||||||
protected void onStart() {
|
protected void onStart() {
|
||||||
getPlayers().forEach(player -> {
|
getPlayers().forEach(player -> {
|
||||||
player.getInventory().setChestplate(ItemStack.of(Material.ELYTRA));
|
player.getInventory().setEquipment(EquipmentSlot.CHESTPLATE, (byte) 0, ItemStack.of(Material.ELYTRA));
|
||||||
for(int i = 0; i < 3; i++) {
|
for(int i = 0; i < 3; i++) {
|
||||||
player.getInventory().setItemStack(i, ItemStack.builder(boostMaterial).displayName(TranslatedComponent.byId("boost").getAssembled(player)).build());
|
player.getInventory().setItemStack(i, ItemStack.builder(boostMaterial).customName(TranslatedComponent.byId("boost").getAssembled(player)).build());
|
||||||
}
|
}
|
||||||
addResetItemToPlayer(player);
|
addResetItemToPlayer(player);
|
||||||
});
|
});
|
||||||
@@ -131,6 +130,11 @@ public class ElytraRace extends StatelessGame {
|
|||||||
Player player = playerMoveEvent.getPlayer();
|
Player player = playerMoveEvent.getPlayer();
|
||||||
Point newPos = playerMoveEvent.getNewPosition();
|
Point newPos = playerMoveEvent.getNewPosition();
|
||||||
|
|
||||||
|
if(isBeforeBeginning && playerMoveEvent.getNewPosition().y() < getSpawn().y()) {
|
||||||
|
player.teleport(getSpawn());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
playerCheckpoints.putIfAbsent(player, new CheckPointData(ringSpacing, ringSpacing * 2));
|
playerCheckpoints.putIfAbsent(player, new CheckPointData(ringSpacing, ringSpacing * 2));
|
||||||
|
|
||||||
if(newPos.z() > generatedUntil - ringSpacing) {
|
if(newPos.z() > generatedUntil - ringSpacing) {
|
||||||
@@ -144,7 +148,7 @@ public class ElytraRace extends StatelessGame {
|
|||||||
|
|
||||||
if(newPos.y() > gameHeight - 5) {
|
if(newPos.y() > gameHeight - 5) {
|
||||||
Point particlePoint = newPos.withY(gameHeight);
|
Point particlePoint = newPos.withY(gameHeight);
|
||||||
ParticlePacket particle = ParticleCreator.createParticlePacket(
|
ParticlePacket particle = new ParticlePacket(
|
||||||
Particle.WAX_ON,
|
Particle.WAX_ON,
|
||||||
particlePoint.blockX(),
|
particlePoint.blockX(),
|
||||||
particlePoint.blockY(),
|
particlePoint.blockY(),
|
||||||
@@ -152,6 +156,7 @@ public class ElytraRace extends StatelessGame {
|
|||||||
20,
|
20,
|
||||||
0,
|
0,
|
||||||
30,
|
30,
|
||||||
|
1f,
|
||||||
Math.toIntExact((long) NumberUtil.map(newPos.y(), gameHeight - 5, gameHeight, 50, 500))
|
Math.toIntExact((long) NumberUtil.map(newPos.y(), gameHeight - 5, gameHeight, 50, 500))
|
||||||
);
|
);
|
||||||
player.sendPacket(particle);
|
player.sendPacket(particle);
|
||||||
@@ -164,6 +169,7 @@ public class ElytraRace extends StatelessGame {
|
|||||||
if(newPos.z() > ringCount * ringSpacing) {
|
if(newPos.z() > ringCount * ringSpacing) {
|
||||||
getScore().insertResult(player);
|
getScore().insertResult(player);
|
||||||
player.setGameMode(GameMode.SPECTATOR);
|
player.setGameMode(GameMode.SPECTATOR);
|
||||||
|
player.setFlyingWithElytra(false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -173,7 +179,7 @@ public class ElytraRace extends StatelessGame {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void addResetItemToPlayer(Player p) {
|
private void addResetItemToPlayer(Player p) {
|
||||||
p.getInventory().setItemStack(8, ItemStack.builder(resetMaterial).displayName(TranslatedComponent.byId("reset").getAssembled(p)).build());
|
p.getInventory().setItemStack(8, ItemStack.builder(resetMaterial).customName(TranslatedComponent.byId("reset").getAssembled(p)).build());
|
||||||
}
|
}
|
||||||
|
|
||||||
private Point getRingPositionAtZ(int z) {
|
private Point getRingPositionAtZ(int z) {
|
||||||
@@ -189,8 +195,8 @@ public class ElytraRace extends StatelessGame {
|
|||||||
|
|
||||||
Point ringPos = getRingPositionAtZ(zPos);
|
Point ringPos = getRingPositionAtZ(zPos);
|
||||||
GeneratorUtils.iterateArea(
|
GeneratorUtils.iterateArea(
|
||||||
ringPos.sub(100, 0, 0).withY(getDimensionType().getMinY()),
|
ringPos.sub(100, 0, 0).withY(0),
|
||||||
ringPos.add(100, 0, 0).withY(gameHeight),
|
ringPos.add(100, 0, 0).withY(seaLevel),
|
||||||
point -> batch.setBlock(point, Block.BARRIER)
|
point -> batch.setBlock(point, Block.BARRIER)
|
||||||
);
|
);
|
||||||
GeneratorUtils.iterateArea(
|
GeneratorUtils.iterateArea(
|
||||||
|
@@ -14,7 +14,12 @@ import java.util.Map;
|
|||||||
public class ElytraRaceFactory implements GameFactory {
|
public class ElytraRaceFactory implements GameFactory {
|
||||||
@Override
|
@Override
|
||||||
public TranslatedComponent name() {
|
public TranslatedComponent name() {
|
||||||
return TranslatedComponent.byId("");
|
return TranslatedComponent.byId("game_ElytraRace#name");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TranslatedComponent description() {
|
||||||
|
return TranslatedComponent.byId("game_ElytraRace#description");
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -25,7 +30,7 @@ public class ElytraRaceFactory implements GameFactory {
|
|||||||
@Override
|
@Override
|
||||||
public ConfigManager configuration() {
|
public ConfigManager configuration() {
|
||||||
return new ConfigManager()
|
return new ConfigManager()
|
||||||
.addOption(new NumericOption("ringCount", Material.DIAMOND_BLOCK, TranslatedComponent.byId("ringCount"), 5, 10, 20, 30, 40, 50));
|
.addOption(new NumericOption("ringCount", Material.DIAMOND_BLOCK, TranslatedComponent.byId("game_ElytraRace#ringCount"), 5, 10, 20, 30, 40, 50));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@@ -0,0 +1,64 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.fastbridge;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.instance.Dimension;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.StatelessGame;
|
||||||
|
import eu.mhsl.minenet.minigames.score.FirstWinsScore;
|
||||||
|
import net.minestom.server.coordinate.Pos;
|
||||||
|
import net.minestom.server.entity.GameMode;
|
||||||
|
import net.minestom.server.entity.Player;
|
||||||
|
import net.minestom.server.event.player.PlayerBlockPlaceEvent;
|
||||||
|
import net.minestom.server.event.player.PlayerMoveEvent;
|
||||||
|
import net.minestom.server.instance.Chunk;
|
||||||
|
import net.minestom.server.inventory.PlayerInventory;
|
||||||
|
import net.minestom.server.item.ItemStack;
|
||||||
|
import net.minestom.server.item.Material;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
|
public class Fastbridge extends StatelessGame {
|
||||||
|
private int currentSpawn = 0;
|
||||||
|
|
||||||
|
public Fastbridge() {
|
||||||
|
super(Dimension.OVERWORLD.key, "Fastbridge", new FirstWinsScore());
|
||||||
|
this.setGenerator(new FastbridgeChunkgenerator());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onPlayerMove(@NotNull PlayerMoveEvent playerMoveEvent) {
|
||||||
|
Player player = playerMoveEvent.getPlayer();
|
||||||
|
Pos newPos = playerMoveEvent.getNewPosition();
|
||||||
|
if(this.getScore().hasResult(player)) return;
|
||||||
|
if(newPos.y() < 0) {
|
||||||
|
player.teleport(getSpawn());
|
||||||
|
if(!isBeforeBeginning) this.resetPlayer(player);
|
||||||
|
}
|
||||||
|
if(newPos.x() > 53) {
|
||||||
|
this.getScore().insertResult(player);
|
||||||
|
player.setGameMode(GameMode.SPECTATOR);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onStart() {
|
||||||
|
getPlayers().forEach(player -> {
|
||||||
|
player.setGameMode(GameMode.SURVIVAL);
|
||||||
|
resetPlayer(player);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onBlockPlace(@NotNull PlayerBlockPlaceEvent playerBlockPlaceEvent) {
|
||||||
|
if(isBeforeBeginning) playerBlockPlaceEvent.setCancelled(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void resetPlayer(Player player) {
|
||||||
|
if(isBeforeBeginning) return;
|
||||||
|
PlayerInventory inventory = player.getInventory();
|
||||||
|
inventory.clear();
|
||||||
|
inventory.addItemStack(ItemStack.of(Material.WHITE_WOOL, 64));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public synchronized Pos getSpawn() {
|
||||||
|
return new Pos(24, 1, currentSpawn++*Chunk.CHUNK_SIZE_Z*2-8, -90, 0);
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,29 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.fastbridge;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.world.generator.featureEnriched.ValeGenerator;
|
||||||
|
import eu.mhsl.minenet.minigames.world.generator.terrain.BaseGenerator;
|
||||||
|
import net.minestom.server.instance.block.Block;
|
||||||
|
|
||||||
|
public class FastbridgeChunkgenerator extends BaseGenerator {
|
||||||
|
public FastbridgeChunkgenerator() {
|
||||||
|
this.addMixIn(unit -> {
|
||||||
|
if (unit.absoluteStart().chunkZ() % 2 == 0) {
|
||||||
|
unit.modifier().fill(Block.BARRIER);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (unit.absoluteStart().chunkX() != 1 && unit.absoluteStart().chunkX() != 3) return;
|
||||||
|
for (int x = 5; x <= 10; x++){
|
||||||
|
for (int z = 5; z <= 10; z++){
|
||||||
|
unit.modifier().setRelative(x, 64, z, unit.absoluteStart().chunkX() == 3 ? Block.GOLD_BLOCK : Block.GRASS_BLOCK);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
ValeGenerator vale = new ValeGenerator();
|
||||||
|
vale.setXShiftMultiplier(integer -> 0.5d);
|
||||||
|
vale.setHeightNoiseMultiplier(integer -> 2);
|
||||||
|
vale.setXShiftOffset(integer -> 40d);
|
||||||
|
this.addMixIn(vale);
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,33 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.fastbridge;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.Game;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.GameFactory;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.Option;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.room.Room;
|
||||||
|
import eu.mhsl.minenet.minigames.message.component.TranslatedComponent;
|
||||||
|
import net.minestom.server.item.Material;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
public class FastbridgeFactory implements GameFactory {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TranslatedComponent name() {
|
||||||
|
return TranslatedComponent.byId("game_Fastbridge#name");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TranslatedComponent description() {
|
||||||
|
return TranslatedComponent.byId("game_Fastbridge#description");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Material symbol() {
|
||||||
|
return Material.WHITE_WOOL;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Game manufacture(Room parent, Map<String, Option<?>> configuration) throws Exception {
|
||||||
|
return new Fastbridge().setParent(parent);
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,126 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.highGround;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.instance.Dimension;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.StatelessGame;
|
||||||
|
import eu.mhsl.minenet.minigames.score.PointsWinScore;
|
||||||
|
import eu.mhsl.minenet.minigames.world.BlockPallet;
|
||||||
|
import io.github.togar2.pvp.events.EntityKnockbackEvent;
|
||||||
|
import io.github.togar2.pvp.events.FinalAttackEvent;
|
||||||
|
import io.github.togar2.pvp.events.PrepareAttackEvent;
|
||||||
|
import io.github.togar2.pvp.feature.CombatFeatures;
|
||||||
|
import net.minestom.server.coordinate.Pos;
|
||||||
|
import net.minestom.server.entity.Player;
|
||||||
|
import net.minestom.server.event.instance.InstanceTickEvent;
|
||||||
|
import net.minestom.server.event.player.PlayerMoveEvent;
|
||||||
|
import net.minestom.server.instance.block.Block;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
|
import java.util.WeakHashMap;
|
||||||
|
import java.util.concurrent.CompletableFuture;
|
||||||
|
|
||||||
|
class HighGround extends StatelessGame {
|
||||||
|
private final int radius;
|
||||||
|
private final int seconds;
|
||||||
|
private final WeakHashMap<Player, Integer> scoreMap = new WeakHashMap<>();
|
||||||
|
|
||||||
|
HighGround(int radius, int seconds) {
|
||||||
|
super(Dimension.THE_END.key, "highground", new PointsWinScore());
|
||||||
|
this.radius = radius;
|
||||||
|
this.seconds = seconds;
|
||||||
|
|
||||||
|
this.eventNode().addChild(
|
||||||
|
CombatFeatures.empty()
|
||||||
|
.add(CombatFeatures.VANILLA_ATTACK)
|
||||||
|
.add(CombatFeatures.VANILLA_DAMAGE)
|
||||||
|
.add(CombatFeatures.VANILLA_KNOCKBACK)
|
||||||
|
.build()
|
||||||
|
.createNode()
|
||||||
|
);
|
||||||
|
|
||||||
|
this.eventNode().addListener(
|
||||||
|
FinalAttackEvent.class,
|
||||||
|
finalAttackEvent -> finalAttackEvent.setBaseDamage(0)
|
||||||
|
);
|
||||||
|
|
||||||
|
this.eventNode().addListener(PrepareAttackEvent.class, prepareAttackEvent -> {
|
||||||
|
if(this.isBeforeBeginning){
|
||||||
|
prepareAttackEvent.setCancelled(true);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
this.eventNode().addListener(
|
||||||
|
EntityKnockbackEvent.class,
|
||||||
|
entityKnockbackEvent -> entityKnockbackEvent.setStrength(1.1f)
|
||||||
|
);
|
||||||
|
|
||||||
|
this.eventNode().addListener(InstanceTickEvent.class, instanceTickEvent -> {
|
||||||
|
if (this.isBeforeBeginning || !this.isRunning) return;
|
||||||
|
this.getPlayers().forEach(player -> {
|
||||||
|
if((player.isOnGround() && player.getPosition().y() >= 1) || (!player.isOnGround() && player.getPosition().y() >= 1.5)){
|
||||||
|
this.scoreMap.put(player, this.scoreMap.get(player) + 1);
|
||||||
|
player.setLevel(this.scoreMap.get(player) / 20);
|
||||||
|
player.setExp((this.scoreMap.get(player) % 20) / 20.0f);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onLoad(@NotNull CompletableFuture<Void> callback) {
|
||||||
|
for (int y = 0; y >= -3; y--) {
|
||||||
|
int radius = (Math.abs(y) * 5) + this.radius;
|
||||||
|
for (int x = -radius; x <= radius; x++) {
|
||||||
|
for (int z = -radius; z <= radius; z++) {
|
||||||
|
double distance = new Pos(x, 0, z).distance(0, 0, 0);
|
||||||
|
if (distance <= radius) {
|
||||||
|
this.setBlock(x, y, z, y == 0 ? Block.DIAMOND_BLOCK : Block.GRASS_BLOCK);
|
||||||
|
Pos featurePosition = new Pos(x, y + 1, z);
|
||||||
|
|
||||||
|
if(y >= 0 || this.getBlock(featurePosition).isSolid()) continue;
|
||||||
|
if (this.rnd.nextDouble() < 0.1){
|
||||||
|
this.setBlock(featurePosition, Block.SHORT_GRASS);
|
||||||
|
}
|
||||||
|
if (this.rnd.nextDouble() < 0.01){
|
||||||
|
this.setBlock(featurePosition, BlockPallet.FLOWER.rnd());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onPlayerMove(@NotNull PlayerMoveEvent playerMoveEvent) {
|
||||||
|
Player player = playerMoveEvent.getPlayer();
|
||||||
|
if(playerMoveEvent.getNewPosition().y() < -10){
|
||||||
|
player.teleport(this.getSpawn());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void start() {
|
||||||
|
this.getPlayers().forEach(player -> this.scoreMap.put(player, 0));
|
||||||
|
super.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onStart() {
|
||||||
|
this.setTimeLimit(this.seconds);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onStop() {
|
||||||
|
this.getPlayers().forEach(player -> this.getScore().insertResult(player, this.scoreMap.get(player)));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Pos getSpawn() {
|
||||||
|
double theta = this.rnd.nextDouble() * 2 * Math.PI;
|
||||||
|
|
||||||
|
double spawnRadius = this.radius + 5;
|
||||||
|
double x = spawnRadius * Math.cos(theta);
|
||||||
|
double z = spawnRadius * Math.sin(theta);
|
||||||
|
|
||||||
|
return new Pos(x, 0, z).withLookAt(new Pos(0, 0, 0));
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,41 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.highGround;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.Game;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.ConfigManager;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.GameFactory;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.Option;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.common.NumericOption;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.room.Room;
|
||||||
|
import eu.mhsl.minenet.minigames.message.component.TranslatedComponent;
|
||||||
|
import net.minestom.server.item.Material;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
public class HighGroundFactory implements GameFactory {
|
||||||
|
@Override
|
||||||
|
public TranslatedComponent name() {
|
||||||
|
return TranslatedComponent.byId("game_Highground#name");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TranslatedComponent description() {
|
||||||
|
return TranslatedComponent.byId("game_Highground#description");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Material symbol() {
|
||||||
|
return Material.GOLDEN_HELMET;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ConfigManager configuration() {
|
||||||
|
return new ConfigManager()
|
||||||
|
.addOption(new NumericOption("radius", Material.HEART_OF_THE_SEA, TranslatedComponent.byId("optionCommon#radius"), 3, 5, 7, 10))
|
||||||
|
.addOption(new NumericOption("seconds", Material.CLOCK, TranslatedComponent.byId("optionCommon#seconds"), 30, 60, 90, 120));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Game manufacture(Room parent, Map<String, Option<?>> configuration) throws Exception {
|
||||||
|
return new HighGround(configuration.get("radius").getAsInt(), configuration.get("seconds").getAsInt()).setParent(parent);
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,99 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.jumpDive;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.instance.Dimension;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.StatelessGame;
|
||||||
|
import eu.mhsl.minenet.minigames.score.PointsWinScore;
|
||||||
|
import eu.mhsl.minenet.minigames.util.BatchUtil;
|
||||||
|
import eu.mhsl.minenet.minigames.world.BlockPallet;
|
||||||
|
import net.kyori.adventure.sound.Sound;
|
||||||
|
import net.minestom.server.coordinate.Pos;
|
||||||
|
import net.minestom.server.entity.Player;
|
||||||
|
import net.minestom.server.event.player.PlayerMoveEvent;
|
||||||
|
import net.minestom.server.instance.batch.AbsoluteBlockBatch;
|
||||||
|
import net.minestom.server.instance.block.Block;
|
||||||
|
import net.minestom.server.sound.SoundEvent;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
|
import java.util.WeakHashMap;
|
||||||
|
import java.util.concurrent.CompletableFuture;
|
||||||
|
|
||||||
|
public class JumpDive extends StatelessGame {
|
||||||
|
private final int radius;
|
||||||
|
private final int height;
|
||||||
|
private final int timeLimit;
|
||||||
|
|
||||||
|
private final WeakHashMap<Player, Integer> scores = new WeakHashMap<>();
|
||||||
|
|
||||||
|
public JumpDive(int radius, int height, int timeLimit) {
|
||||||
|
super(Dimension.OVERWORLD.key, "jumpDive", new PointsWinScore());
|
||||||
|
this.radius = radius;
|
||||||
|
this.height = height;
|
||||||
|
this.timeLimit = timeLimit;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onLoad(@NotNull CompletableFuture<Void> callback) {
|
||||||
|
AbsoluteBlockBatch batch = new AbsoluteBlockBatch();
|
||||||
|
|
||||||
|
for(int x = -radius*2; x <= radius*2; x++) {
|
||||||
|
for(int z = -radius*2; z <= radius*2; z++) {
|
||||||
|
if(new Pos(x, 0, z).distance(new Pos(0, 0, 0)) > radius) {
|
||||||
|
batch.setBlock(x, height, z, BlockPallet.STONE.rnd());
|
||||||
|
} else {
|
||||||
|
batch.setBlock(x, 0, z, BlockPallet.GROUND.rnd());
|
||||||
|
batch.setBlock(x, 1, z, Block.WATER);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
BatchUtil.loadAndApplyBatch(batch, this, () -> callback.complete(null));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onStart() {
|
||||||
|
setTimeLimit(timeLimit);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onPlayerMove(@NotNull PlayerMoveEvent playerMoveEvent) {
|
||||||
|
Player p = playerMoveEvent.getPlayer();
|
||||||
|
|
||||||
|
if(
|
||||||
|
p.isOnGround() && playerMoveEvent.getNewPosition().y() < height
|
||||||
|
|| playerMoveEvent.getNewPosition().y() < 0
|
||||||
|
|| isBeforeBeginning && playerMoveEvent.getNewPosition().y() < height
|
||||||
|
) {
|
||||||
|
p.teleport(getSpawn());
|
||||||
|
playerMoveEvent.setCancelled(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(
|
||||||
|
playerMoveEvent.getNewPosition().y() <= 1
|
||||||
|
&& playerMoveEvent.getNewPosition().distance(0, 1, 0) < radius + 0.5
|
||||||
|
&& !(!isBeforeBeginning && !isRunning)
|
||||||
|
) {
|
||||||
|
setBlock(playerMoveEvent.getNewPosition().withY(1), Block.REDSTONE_BLOCK);
|
||||||
|
scores.merge(p, 1, Integer::sum);
|
||||||
|
p.teleport(getSpawn());
|
||||||
|
playerMoveEvent.setCancelled(true);
|
||||||
|
p.playSound(Sound.sound(SoundEvent.ENTITY_EXPERIENCE_ORB_PICKUP, Sound.Source.PLAYER, 2f, 2f));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onStop() {
|
||||||
|
getPlayers().forEach(player -> getScore().insertResult(player, scores.getOrDefault(player, 0)));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Pos getSpawn() {
|
||||||
|
double theta = rnd.nextDouble() * 2 * Math.PI;
|
||||||
|
|
||||||
|
double spawnRadius = radius + 2;
|
||||||
|
double x = spawnRadius * Math.cos(theta);
|
||||||
|
double z = spawnRadius * Math.sin(theta);
|
||||||
|
|
||||||
|
return new Pos(x, height + 2, z).withLookAt(new Pos(0, height, 0));
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,43 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.jumpDive;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.Game;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.ConfigManager;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.GameFactory;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.Option;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.common.NumericOption;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.room.Room;
|
||||||
|
import eu.mhsl.minenet.minigames.message.component.TranslatedComponent;
|
||||||
|
import net.minestom.server.item.Material;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
public class JumpDiveFactory implements GameFactory {
|
||||||
|
@Override
|
||||||
|
public TranslatedComponent name() {
|
||||||
|
return TranslatedComponent.byId("game_jumpDive#name");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TranslatedComponent description() {
|
||||||
|
return TranslatedComponent.byId("game_jumpDive#description");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ConfigManager configuration() {
|
||||||
|
return new ConfigManager()
|
||||||
|
.addOption(new NumericOption("radius", Material.HEART_OF_THE_SEA, TranslatedComponent.byId("optionCommon#radius"), 5, 8, 10, 12, 14, 16))
|
||||||
|
.addOption(new NumericOption("height", Material.SCAFFOLDING, TranslatedComponent.byId("optionCommon#height"), 30, 60, 90))
|
||||||
|
.addOption(new NumericOption("timeLimit", Material.CLOCK, TranslatedComponent.byId("optionCommon#seconds"), 60, 120, 180, 240, 300));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Game manufacture(Room parent, Map<String, Option<?>> configuration) throws Exception {
|
||||||
|
return new JumpDive(configuration.get("radius").getAsInt(), configuration.get("height").getAsInt(), configuration.get("timeLimit").getAsInt()).setParent(parent);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Material symbol() {
|
||||||
|
return Material.WATER_BUCKET;
|
||||||
|
}
|
||||||
|
}
|
@@ -10,7 +10,6 @@ import eu.mhsl.minenet.minigames.world.BlockPallet;
|
|||||||
import eu.mhsl.minenet.minigames.world.generator.terrain.SquarePlateTerrainGenerator;
|
import eu.mhsl.minenet.minigames.world.generator.terrain.SquarePlateTerrainGenerator;
|
||||||
import net.kyori.adventure.sound.Sound;
|
import net.kyori.adventure.sound.Sound;
|
||||||
import net.minestom.server.coordinate.Pos;
|
import net.minestom.server.coordinate.Pos;
|
||||||
import net.minestom.server.entity.Entity;
|
|
||||||
import net.minestom.server.entity.GameMode;
|
import net.minestom.server.entity.GameMode;
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.event.player.PlayerMoveEvent;
|
import net.minestom.server.event.player.PlayerMoveEvent;
|
||||||
@@ -33,7 +32,7 @@ class Minerun extends StatelessGame {
|
|||||||
private final int afterFinishLine = 10;
|
private final int afterFinishLine = 10;
|
||||||
|
|
||||||
public Minerun(int width, int length, int minePercentage) {
|
public Minerun(int width, int length, int minePercentage) {
|
||||||
super(Dimension.THE_END.DIMENSION, "Minerun", new FirstWinsScore());
|
super(Dimension.THE_END.key, "Minerun", new FirstWinsScore());
|
||||||
setGenerator(new SquarePlateTerrainGenerator(width, length + preRun + afterFinishLine).setPlateHeight(50).setGenerateBorders(true));
|
setGenerator(new SquarePlateTerrainGenerator(width, length + preRun + afterFinishLine).setPlateHeight(50).setGenerateBorders(true));
|
||||||
|
|
||||||
this.width = width;
|
this.width = width;
|
||||||
@@ -87,7 +86,6 @@ class Minerun extends StatelessGame {
|
|||||||
|
|
||||||
if(Intersect.withPressurePlate(this, BlockPallet.PRESSURE_PLATES, middle)) { //Player died
|
if(Intersect.withPressurePlate(this, BlockPallet.PRESSURE_PLATES, middle)) { //Player died
|
||||||
p.playSound(Sound.sound(SoundEvent.ENTITY_GENERIC_EXPLODE, Sound.Source.PLAYER, 1f, 1f));
|
p.playSound(Sound.sound(SoundEvent.ENTITY_GENERIC_EXPLODE, Sound.Source.PLAYER, 1f, 1f));
|
||||||
p.setPose(Entity.Pose.DYING);
|
|
||||||
p.teleport(new Pos(p.getPosition().x(), getSpawn().y(), getSpawn().z()));
|
p.teleport(new Pos(p.getPosition().x(), getSpawn().y(), getSpawn().z()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -22,7 +22,7 @@ public class MinerunFactory implements GameFactory {
|
|||||||
return new ConfigManager()
|
return new ConfigManager()
|
||||||
.addOption(new NumericOption("width", Material.OAK_FENCE, TranslatedComponent.byId("optionCommon#width"), 10, 30, 50, 100))
|
.addOption(new NumericOption("width", Material.OAK_FENCE, TranslatedComponent.byId("optionCommon#width"), 10, 30, 50, 100))
|
||||||
.addOption(new NumericOption("length", Material.ZOMBIE_HEAD, TranslatedComponent.byId("optionCommon#length"), 50, 100, 150, 200))
|
.addOption(new NumericOption("length", Material.ZOMBIE_HEAD, TranslatedComponent.byId("optionCommon#length"), 50, 100, 150, 200))
|
||||||
.addOption(new NumericOption("percentage", Material.LIGHT_WEIGHTED_PRESSURE_PLATE, TranslatedComponent.byId("game_Minerun#optionPercentageMiens"), 30, 40, 50, 60, 70));
|
.addOption(new NumericOption("percentage", Material.LIGHT_WEIGHTED_PRESSURE_PLATE, TranslatedComponent.byId("game_Minerun#optionPercentageMines"), 30, 40, 50, 60, 70));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@@ -26,7 +26,7 @@ public class Spleef extends StatelessGame {
|
|||||||
final int totalElevation = 50;
|
final int totalElevation = 50;
|
||||||
|
|
||||||
public Spleef(int radius, int stackCount) {
|
public Spleef(int radius, int stackCount) {
|
||||||
super(Dimension.OVERWORLD.DIMENSION, "Spleef", new LastWinsScore());
|
super(Dimension.OVERWORLD.key, "Spleef", new LastWinsScore());
|
||||||
getScore().setIgnoreLastPlayers(1);
|
getScore().setIgnoreLastPlayers(1);
|
||||||
|
|
||||||
this.radius = radius;
|
this.radius = radius;
|
||||||
@@ -61,13 +61,7 @@ public class Spleef extends StatelessGame {
|
|||||||
player.getInventory().addItemStack(
|
player.getInventory().addItemStack(
|
||||||
ItemStack
|
ItemStack
|
||||||
.builder(Material.DIAMOND_SHOVEL)
|
.builder(Material.DIAMOND_SHOVEL)
|
||||||
.displayName(TranslatedComponent.byId("game_Spleef#shovelName").getAssembled(player))
|
.customName(TranslatedComponent.byId("game_Spleef#shovelName").getAssembled(player))
|
||||||
.meta(
|
|
||||||
builder -> builder
|
|
||||||
.enchantment(Enchantment.EFFICIENCY, (short) 99)
|
|
||||||
.hideFlag(ItemHideFlag.HIDE_ENCHANTS)
|
|
||||||
.build()
|
|
||||||
)
|
|
||||||
.build()
|
.build()
|
||||||
);
|
);
|
||||||
player.setHeldItemSlot((byte) 0);
|
player.setHeldItemSlot((byte) 0);
|
||||||
@@ -84,6 +78,7 @@ public class Spleef extends StatelessGame {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void destroyBlock(PlayerStartDiggingEvent event) {
|
private void destroyBlock(PlayerStartDiggingEvent event) {
|
||||||
|
if(!isRunning) return;
|
||||||
setBlock(event.getBlockPosition(), Block.AIR);
|
setBlock(event.getBlockPosition(), Block.AIR);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -33,7 +33,7 @@ public class SpleefFactory implements GameFactory {
|
|||||||
@Override
|
@Override
|
||||||
public ConfigManager configuration() {
|
public ConfigManager configuration() {
|
||||||
return new ConfigManager()
|
return new ConfigManager()
|
||||||
.addOption(new NumericOption("radius", Material.HEART_OF_THE_SEA, TranslatedComponent.byId("game_Spleef#radius"), 10, 20, 30))
|
.addOption(new NumericOption("radius", Material.HEART_OF_THE_SEA, TranslatedComponent.byId("optionCommon#radius"), 10, 20, 30))
|
||||||
.addOption(new NumericOption("stackCount", Material.SCAFFOLDING, TranslatedComponent.byId("game_Spleef#stackCount"), 1, 2, 3, 4, 5));
|
.addOption(new NumericOption("stackCount", Material.SCAFFOLDING, TranslatedComponent.byId("game_Spleef#stackCount"), 1, 2, 3, 4, 5));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -3,28 +3,32 @@ package eu.mhsl.minenet.minigames.instance.game.stateless.types.stickfight;
|
|||||||
import eu.mhsl.minenet.minigames.instance.Dimension;
|
import eu.mhsl.minenet.minigames.instance.Dimension;
|
||||||
import eu.mhsl.minenet.minigames.instance.game.stateless.StatelessGame;
|
import eu.mhsl.minenet.minigames.instance.game.stateless.StatelessGame;
|
||||||
import eu.mhsl.minenet.minigames.score.LastWinsScore;
|
import eu.mhsl.minenet.minigames.score.LastWinsScore;
|
||||||
import eu.mhsl.minenet.minigames.util.BatchUtil;
|
|
||||||
import eu.mhsl.minenet.minigames.world.generator.terrain.CircularPlateTerrainGenerator;
|
import eu.mhsl.minenet.minigames.world.generator.terrain.CircularPlateTerrainGenerator;
|
||||||
import io.github.bloepiloepi.pvp.config.*;
|
import io.github.togar2.pvp.events.FinalAttackEvent;
|
||||||
import io.github.bloepiloepi.pvp.events.FinalAttackEvent;
|
import io.github.togar2.pvp.feature.CombatFeatures;
|
||||||
import net.minestom.server.coordinate.Pos;
|
import net.minestom.server.coordinate.Pos;
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.event.player.PlayerMoveEvent;
|
import net.minestom.server.event.player.PlayerMoveEvent;
|
||||||
import net.minestom.server.instance.batch.AbsoluteBlockBatch;
|
|
||||||
import net.minestom.server.instance.block.Block;
|
import net.minestom.server.instance.block.Block;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.WeakHashMap;
|
||||||
import java.util.concurrent.CompletableFuture;
|
import java.util.concurrent.CompletableFuture;
|
||||||
|
|
||||||
public class Stickfight extends StatelessGame {
|
public class Stickfight extends StatelessGame {
|
||||||
|
private final double radius = 20;
|
||||||
|
private final WeakHashMap<Player, Pos> spawnPoints = new WeakHashMap<>();
|
||||||
|
|
||||||
public Stickfight() {
|
public Stickfight() {
|
||||||
super(Dimension.OVERWORLD.DIMENSION, "Stickfight", new LastWinsScore());
|
super(Dimension.OVERWORLD.key, "Stickfight", new LastWinsScore());
|
||||||
|
|
||||||
eventNode().addChild(
|
eventNode().addChild(
|
||||||
PvPConfig.emptyBuilder()
|
CombatFeatures.empty()
|
||||||
.damage(DamageConfig.legacyBuilder().fallDamage(false))
|
.add(CombatFeatures.VANILLA_ATTACK)
|
||||||
.attack(AttackConfig.legacyBuilder().attackCooldown(true))
|
.add(CombatFeatures.VANILLA_DAMAGE)
|
||||||
.build().createNode()
|
.add(CombatFeatures.VANILLA_KNOCKBACK)
|
||||||
|
.build().createNode()
|
||||||
);
|
);
|
||||||
|
|
||||||
eventNode().addListener(FinalAttackEvent.class, finalAttackEvent -> {
|
eventNode().addListener(FinalAttackEvent.class, finalAttackEvent -> {
|
||||||
@@ -37,21 +41,56 @@ public class Stickfight extends StatelessGame {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onLoad(@NotNull CompletableFuture<Void> callback) {
|
protected void onLoad(@NotNull CompletableFuture<Void> callback) {
|
||||||
AbsoluteBlockBatch batch = new AbsoluteBlockBatch();
|
setBlock(0, 50, 0, Block.DIAMOND_BLOCK);
|
||||||
for (int z = -10; z <= 10; z++) {
|
}
|
||||||
batch.setBlock(0, 50, z, Block.SANDSTONE);
|
|
||||||
}
|
|
||||||
batch.setBlock(0, 50, 0, Block.GOLD_BLOCK);
|
|
||||||
|
|
||||||
BatchUtil.loadAndApplyBatch(batch, this, () -> callback.complete(null));
|
@Override
|
||||||
|
protected void start() {
|
||||||
|
List<Player> players = getPlayers().stream().toList();
|
||||||
|
int numPlayers = players.size();
|
||||||
|
|
||||||
|
for (int i = 0; i < numPlayers; i++) {
|
||||||
|
double angle = (2 * Math.PI / numPlayers) * i;
|
||||||
|
int spawnX = (int) (radius * Math.cos(angle));
|
||||||
|
int spawnZ = (int) (radius * Math.sin(angle));
|
||||||
|
int spawnY = 50;
|
||||||
|
|
||||||
|
Pos spawnpoint = new Pos(spawnX, spawnY + 1, spawnZ).add(0.5);
|
||||||
|
spawnPoints.put(players.get(i), spawnpoint.withLookAt(getSpawn()));
|
||||||
|
players.get(i).teleport(spawnpoint);
|
||||||
|
|
||||||
|
generateBridge(spawnX, spawnY, spawnZ);
|
||||||
|
}
|
||||||
|
|
||||||
|
setBlock(0, 50, 0, Block.GOLD_BLOCK);
|
||||||
|
super.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void generateBridge(int startX, int startY, int startZ) {
|
||||||
|
int steps = (int) (radius * 1.5);
|
||||||
|
for (int i = 0; i < steps; i++) {
|
||||||
|
double t = (double) i / steps;
|
||||||
|
int x = (int) (startX * (1 - t));
|
||||||
|
int z = (int) (startZ * (1 - t));
|
||||||
|
setBlock(x, startY, z, Block.SANDSTONE);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onPlayerMove(@NotNull PlayerMoveEvent playerMoveEvent) {
|
protected void onPlayerMove(@NotNull PlayerMoveEvent playerMoveEvent) {
|
||||||
if(isBeforeBeginning) playerMoveEvent.setCancelled(true);
|
if(!spawnPoints.containsKey(playerMoveEvent.getPlayer())) {
|
||||||
|
playerMoveEvent.setCancelled(true);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(isBeforeBeginning) {
|
||||||
|
if(spawnPoints.get(playerMoveEvent.getPlayer()).distance(playerMoveEvent.getNewPosition()) < 1) return;
|
||||||
|
playerMoveEvent.setCancelled(true);
|
||||||
|
playerMoveEvent.getPlayer().teleport(spawnPoints.get(playerMoveEvent.getPlayer()));
|
||||||
|
}
|
||||||
|
|
||||||
if(playerMoveEvent.getNewPosition().y() < 40) {
|
if(playerMoveEvent.getNewPosition().y() < 40) {
|
||||||
playerMoveEvent.getPlayer().teleport(new Pos(0, 51, 0));
|
playerMoveEvent.getPlayer().teleport(spawnPoints.get(playerMoveEvent.getPlayer()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -0,0 +1,129 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.sumo;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.instance.Dimension;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.StatelessGame;
|
||||||
|
import eu.mhsl.minenet.minigames.score.LastWinsScore;
|
||||||
|
import io.github.togar2.pvp.events.FinalAttackEvent;
|
||||||
|
import io.github.togar2.pvp.events.PrepareAttackEvent;
|
||||||
|
import io.github.togar2.pvp.feature.CombatFeatures;
|
||||||
|
import net.minestom.server.MinecraftServer;
|
||||||
|
import net.minestom.server.coordinate.Pos;
|
||||||
|
import net.minestom.server.entity.GameMode;
|
||||||
|
import net.minestom.server.entity.Player;
|
||||||
|
import net.minestom.server.event.player.PlayerMoveEvent;
|
||||||
|
import net.minestom.server.instance.block.Block;
|
||||||
|
import net.minestom.server.timer.TaskSchedule;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
|
import java.util.WeakHashMap;
|
||||||
|
import java.util.concurrent.CompletableFuture;
|
||||||
|
|
||||||
|
public class Sumo extends StatelessGame {
|
||||||
|
int radius;
|
||||||
|
int health;
|
||||||
|
int seconds;
|
||||||
|
|
||||||
|
int originalRadius;
|
||||||
|
int timer;
|
||||||
|
|
||||||
|
WeakHashMap<Player, Integer> healthMap = new WeakHashMap<>();
|
||||||
|
|
||||||
|
public Sumo(int radius, int health, int seconds) {
|
||||||
|
super(Dimension.OVERWORLD.key, "Sumo", new LastWinsScore());
|
||||||
|
this.getScore().setIgnoreLastPlayers(1);
|
||||||
|
this.setTime(6000);
|
||||||
|
this.setTimeRate(0);
|
||||||
|
|
||||||
|
this.radius = radius;
|
||||||
|
this.health = health;
|
||||||
|
this.seconds = seconds;
|
||||||
|
|
||||||
|
this.originalRadius = radius;
|
||||||
|
this.timer = seconds;
|
||||||
|
|
||||||
|
this.eventNode().addChild(
|
||||||
|
CombatFeatures.empty()
|
||||||
|
.add(CombatFeatures.VANILLA_ATTACK)
|
||||||
|
.add(CombatFeatures.VANILLA_DAMAGE)
|
||||||
|
.add(CombatFeatures.VANILLA_KNOCKBACK)
|
||||||
|
.build()
|
||||||
|
.createNode()
|
||||||
|
);
|
||||||
|
|
||||||
|
this.eventNode().addListener(PrepareAttackEvent.class, prepareAttackEvent -> {
|
||||||
|
if (this.isBeforeBeginning)
|
||||||
|
prepareAttackEvent.setCancelled(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
this.eventNode().addListener(FinalAttackEvent.class, finalAttackEvent -> {
|
||||||
|
finalAttackEvent.setBaseDamage(0);
|
||||||
|
((Player) finalAttackEvent.getTarget()).setHealth(20);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void start() {
|
||||||
|
this.getPlayers().forEach(player -> {
|
||||||
|
this.healthMap.put(player, this.health);
|
||||||
|
player.setLevel(this.healthMap.get(player));
|
||||||
|
});
|
||||||
|
|
||||||
|
MinecraftServer.getSchedulerManager().scheduleTask(
|
||||||
|
() -> {
|
||||||
|
if(this.isBeforeBeginning) return TaskSchedule.seconds(1);
|
||||||
|
this.timer--;
|
||||||
|
double percent = (double) this.timer / this.seconds;
|
||||||
|
int radius = (int) (this.originalRadius * percent);
|
||||||
|
if (this.radius >= 5) {
|
||||||
|
this.radius = radius;
|
||||||
|
this.generatePlatform();
|
||||||
|
return TaskSchedule.seconds(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
return TaskSchedule.stop();
|
||||||
|
},
|
||||||
|
TaskSchedule.seconds(1)
|
||||||
|
);
|
||||||
|
|
||||||
|
super.start();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onLoad(@NotNull CompletableFuture<Void> callback) {
|
||||||
|
this.generatePlatform();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void generatePlatform() {
|
||||||
|
int buffer = 5;
|
||||||
|
for(int x = -this.radius - buffer; x <= this.radius + buffer; x++) {
|
||||||
|
for(int z = -this.radius - buffer; z <= this.radius + buffer; z++) {
|
||||||
|
double distance = new Pos(x, 0, z).distance(new Pos(0, 0, 0));
|
||||||
|
if(distance <= this.radius) {
|
||||||
|
boolean isEdge = this.radius - 1 < distance;
|
||||||
|
this.setBlock(x, 0, z, isEdge ? Block.RED_CONCRETE : Block.WHITE_CONCRETE);
|
||||||
|
} else {
|
||||||
|
this.setBlock(x, 0, z, Block.AIR);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onPlayerMove(@NotNull PlayerMoveEvent playerMoveEvent) {
|
||||||
|
Player player = playerMoveEvent.getPlayer();
|
||||||
|
if(playerMoveEvent.getNewPosition().y() < -10) {
|
||||||
|
player.teleport(this.getSpawn());
|
||||||
|
this.healthMap.put(player, this.healthMap.get(player) - 1);
|
||||||
|
player.setLevel(this.healthMap.get(player));
|
||||||
|
if (this.healthMap.get(player) == 0) {
|
||||||
|
this.getScore().insertResult(player);
|
||||||
|
player.setGameMode(GameMode.SPECTATOR);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Pos getSpawn() {
|
||||||
|
return new Pos(0, 2, 0);
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,41 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.sumo;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.Game;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.ConfigManager;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.GameFactory;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.Option;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.common.NumericOption;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.room.Room;
|
||||||
|
import eu.mhsl.minenet.minigames.message.component.TranslatedComponent;
|
||||||
|
import net.minestom.server.item.Material;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
public class SumoFactory implements GameFactory {
|
||||||
|
@Override
|
||||||
|
public TranslatedComponent name() {
|
||||||
|
return TranslatedComponent.byId("game_Sumo#name");
|
||||||
|
}
|
||||||
|
|
||||||
|
public TranslatedComponent description() {
|
||||||
|
return TranslatedComponent.byId("game_Sumo#description");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Material symbol() {
|
||||||
|
return Material.SLIME_BALL;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ConfigManager configuration() {
|
||||||
|
return new ConfigManager()
|
||||||
|
.addOption(new NumericOption("radius", Material.HEART_OF_THE_SEA, TranslatedComponent.byId("optionCommon#radius"), 10, 20, 30))
|
||||||
|
.addOption(new NumericOption("health", Material.GOLDEN_APPLE, TranslatedComponent.byId("game_Sumo#lives"), 1, 2, 3, 4, 5))
|
||||||
|
.addOption(new NumericOption("seconds", Material.CLOCK, TranslatedComponent.byId("optionCommon#seconds"), 30, 60, 90, 120));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Game manufacture(Room parent, Map<String, Option<?>> configuration) throws Exception {
|
||||||
|
return new Sumo(configuration.get("radius").getAsInt(), configuration.get("health").getAsInt(), configuration.get("seconds").getAsInt()).setParent(parent);
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,162 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.tetris;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.StatelessGame;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.tetris.game.TetrisGame;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.Dimension;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.tetris.game.Tetromino;
|
||||||
|
import eu.mhsl.minenet.minigames.score.PointsWinScore;
|
||||||
|
import net.kyori.adventure.text.Component;
|
||||||
|
import net.minestom.server.coordinate.Pos;
|
||||||
|
import net.minestom.server.entity.Entity;
|
||||||
|
import net.minestom.server.entity.GameMode;
|
||||||
|
import net.minestom.server.entity.Player;
|
||||||
|
import net.minestom.server.event.player.*;
|
||||||
|
import net.minestom.server.item.ItemStack;
|
||||||
|
import net.minestom.server.item.Material;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
|
import java.util.*;
|
||||||
|
|
||||||
|
class Tetris extends StatelessGame {
|
||||||
|
private final Map<Player, TetrisGame> tetrisGames = new WeakHashMap<>();
|
||||||
|
private final int nextTetrominoesCount;
|
||||||
|
private final boolean isFast;
|
||||||
|
private final boolean hasCombat;
|
||||||
|
private boolean setTimeLimit = false;
|
||||||
|
private final long randomSeed;
|
||||||
|
|
||||||
|
public Tetris(int nextTetrominoesCount, boolean isFast, boolean hasCombat) {
|
||||||
|
super(Dimension.THE_END.key, "Tetris", new PointsWinScore());
|
||||||
|
|
||||||
|
this.eventNode()
|
||||||
|
.addListener(PlayerUseItemEvent.class, this::onPlayerInteract)
|
||||||
|
.addListener(PlayerHandAnimationEvent.class, this::onPlayerAttack)
|
||||||
|
.addListener(PlayerTickEvent.class, this::onPlayerTick);
|
||||||
|
|
||||||
|
this.nextTetrominoesCount = nextTetrominoesCount;
|
||||||
|
this.isFast = isFast;
|
||||||
|
this.hasCombat = hasCombat;
|
||||||
|
|
||||||
|
Random random = new Random();
|
||||||
|
this.randomSeed = random.nextLong();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onStart() {
|
||||||
|
this.getEntities().stream()
|
||||||
|
.filter(entity -> entity.getEntityType().equals(Tetromino.getGhostEntityType()))
|
||||||
|
.forEach(Entity::remove);
|
||||||
|
|
||||||
|
if(this.hasCombat) {
|
||||||
|
this.tetrisGames.values().forEach(tetrisGame -> tetrisGame.updateOtherTetrisGames(this.tetrisGames.values()));
|
||||||
|
}
|
||||||
|
|
||||||
|
this.tetrisGames.forEach((player, tetrisGame) -> tetrisGame.start());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onStop() {
|
||||||
|
this.tetrisGames.forEach((player, tetrisGame) -> {
|
||||||
|
tetrisGame.loose();
|
||||||
|
this.getScore().insertResult(player, tetrisGame.getScore());
|
||||||
|
tetrisGame.sidebar.removeViewer(player);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onPlayerLeave(Player p) {
|
||||||
|
this.tetrisGames.get(p).sidebar.removeViewer(p);
|
||||||
|
this.letPlayerLoose(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void onPlayerMove(@NotNull PlayerMoveEvent event) {
|
||||||
|
Player player = event.getPlayer();
|
||||||
|
Pos currentPosition = event.getNewPosition();
|
||||||
|
|
||||||
|
TetrisGame tetrisGame = this.tetrisGames.get(player);
|
||||||
|
|
||||||
|
if(tetrisGame == null) {
|
||||||
|
event.setCancelled(true);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if(tetrisGame.lost) return;
|
||||||
|
if(player.getGameMode() == GameMode.SPECTATOR) return;
|
||||||
|
|
||||||
|
if(player.inputs().forward()) tetrisGame.pressedButton(TetrisGame.Button.W);
|
||||||
|
if(player.inputs().backward()) tetrisGame.pressedButton(TetrisGame.Button.S);
|
||||||
|
if(player.inputs().right()) tetrisGame.pressedButton(TetrisGame.Button.D);
|
||||||
|
if(player.inputs().left()) tetrisGame.pressedButton(TetrisGame.Button.A);
|
||||||
|
if(player.inputs().jump()) tetrisGame.pressedButton(TetrisGame.Button.space);
|
||||||
|
|
||||||
|
event.setNewPosition(tetrisGame.getPlayerSpawnPosition().withView(currentPosition));
|
||||||
|
player.setSprinting(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void onPlayerInteract(@NotNull PlayerUseItemEvent event) {
|
||||||
|
this.tetrisGames.get(event.getPlayer()).pressedButton(TetrisGame.Button.mouseRight);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void onPlayerAttack(@NotNull PlayerHandAnimationEvent event) {
|
||||||
|
this.tetrisGames.get(event.getPlayer()).pressedButton(TetrisGame.Button.mouseLeft);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected void onPlayerTick(PlayerTickEvent event) {
|
||||||
|
Player player = event.getPlayer();
|
||||||
|
TetrisGame tetrisGame = this.tetrisGames.get(player);
|
||||||
|
if(tetrisGame == null) return;
|
||||||
|
if(tetrisGame.lost && player.getGameMode() != GameMode.SPECTATOR) {
|
||||||
|
this.letPlayerLoose(player);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void letPlayerLoose(Player player) {
|
||||||
|
TetrisGame tetrisGame = this.tetrisGames.get(player);
|
||||||
|
if(!this.getScore().hasResult(player)) {
|
||||||
|
player.setGameMode(GameMode.SPECTATOR);
|
||||||
|
player.setInvisible(true);
|
||||||
|
this.getScore().insertResult(player, tetrisGame.getScore());
|
||||||
|
}
|
||||||
|
|
||||||
|
boolean allGamesLost = this.tetrisGames.values().stream()
|
||||||
|
.filter(game -> !game.lost)
|
||||||
|
.toList()
|
||||||
|
.isEmpty();
|
||||||
|
if(!this.setTimeLimit && !allGamesLost) {
|
||||||
|
this.setTimeLimit(90);
|
||||||
|
this.setTimeLimit = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected boolean onPlayerJoin(Player p) {
|
||||||
|
p.getInventory().setItemStack(0, ItemStack.builder(Material.BIRCH_BUTTON).customName(Component.text("Controller")).build());
|
||||||
|
p.setSprinting(false);
|
||||||
|
|
||||||
|
if(this.tetrisGames.get(p) == null) {
|
||||||
|
this.tetrisGames.put(p, new TetrisGame(
|
||||||
|
this,
|
||||||
|
this.getSpawn().sub(6, 8, 15).add(this.tetrisGames.size()*30, 0, 0),
|
||||||
|
Tetromino.Shape.J,
|
||||||
|
this.nextTetrominoesCount,
|
||||||
|
this.isFast,
|
||||||
|
this.hasCombat,
|
||||||
|
this.randomSeed
|
||||||
|
));
|
||||||
|
this.tetrisGames.get(p).generate();
|
||||||
|
this.tetrisGames.values().forEach(tetrisGame -> tetrisGame.updateOtherTetrisGames(this.tetrisGames.values()));
|
||||||
|
}
|
||||||
|
|
||||||
|
TetrisGame tetrisGame = this.tetrisGames.get(p);
|
||||||
|
|
||||||
|
p.teleport(tetrisGame.getPlayerSpawnPosition());
|
||||||
|
tetrisGame.sidebar.addViewer(p);
|
||||||
|
|
||||||
|
return super.onPlayerJoin(p);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Pos getSpawn() {
|
||||||
|
return new Pos(0, 30, 15).withView(180, 0);
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,43 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.tetris;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.Game;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.GameFactory;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.Option;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.ConfigManager;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.common.BoolOption;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.config.common.NumericOption;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.room.Room;
|
||||||
|
import eu.mhsl.minenet.minigames.message.component.TranslatedComponent;
|
||||||
|
import net.minestom.server.item.Material;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
public class TetrisFactory implements GameFactory {
|
||||||
|
@Override
|
||||||
|
public TranslatedComponent name() {
|
||||||
|
return TranslatedComponent.byId("game_Tetris#name");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TranslatedComponent description() {
|
||||||
|
return TranslatedComponent.byId("game_Tetris#description");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ConfigManager configuration() {
|
||||||
|
return new ConfigManager()
|
||||||
|
.addOption(new NumericOption("nextTetrominoesCount", Material.LADDER, TranslatedComponent.byId("game_Tetris#nextTetrominoesCount"), 3, 4, 5, 1, 2))
|
||||||
|
.addOption(new BoolOption("isFast", Material.MINECART, TranslatedComponent.byId("game_Tetris#isFast")))
|
||||||
|
.addOption(new BoolOption("hasCombat", Material.DIAMOND_SWORD, TranslatedComponent.byId("game_Tetris#hasCombat")));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Game manufacture(Room parent, Map<String, Option<?>> configuration) {
|
||||||
|
return new Tetris(configuration.get("nextTetrominoesCount").getAsInt(), configuration.get("isFast").getAsBoolean(), configuration.get("hasCombat").getAsBoolean()).setParent(parent);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Material symbol() {
|
||||||
|
return Material.PURPLE_WOOL;
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,191 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.tetris.game;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.StatelessGame;
|
||||||
|
import eu.mhsl.minenet.minigames.util.BatchUtil;
|
||||||
|
import net.minestom.server.coordinate.Pos;
|
||||||
|
import net.minestom.server.instance.batch.AbsoluteBlockBatch;
|
||||||
|
import net.minestom.server.instance.block.Block;
|
||||||
|
import org.apache.commons.lang3.ArrayUtils;
|
||||||
|
|
||||||
|
import java.util.Random;
|
||||||
|
|
||||||
|
public class Playfield {
|
||||||
|
private final Pos lowerLeftCorner;
|
||||||
|
private final StatelessGame instance;
|
||||||
|
private final static int height = 22;
|
||||||
|
private final static Block scoreBlock = Block.STONE;
|
||||||
|
private final int nextTetrominoesCount;
|
||||||
|
private final Random random;
|
||||||
|
|
||||||
|
public Playfield(Pos lowerLeftCorner, StatelessGame instance, int nextTetrominoesCount) {
|
||||||
|
this.nextTetrominoesCount = nextTetrominoesCount;
|
||||||
|
this.lowerLeftCorner = lowerLeftCorner;
|
||||||
|
this.instance = instance;
|
||||||
|
this.random = new Random();
|
||||||
|
}
|
||||||
|
|
||||||
|
public Pos getPlayerSpawnPosition() {
|
||||||
|
return this.lowerLeftCorner.add(6, 9, 20).withView(180, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Pos getTetrominoSpawnPosition() {
|
||||||
|
return this.lowerLeftCorner.add(5, 21, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Pos getHoldPosition() {
|
||||||
|
return this.lowerLeftCorner.add(-4, 18, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Pos getNextPosition() {
|
||||||
|
return this.lowerLeftCorner.add(15, 18, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Pos getScorePosition() {
|
||||||
|
return this.lowerLeftCorner.add(-5, height+3, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void generate() {
|
||||||
|
AbsoluteBlockBatch batch = new AbsoluteBlockBatch();
|
||||||
|
|
||||||
|
// actual playfield:
|
||||||
|
for(int x=0; x<12; x++) {
|
||||||
|
for(int y = 0; y< height; y++) {
|
||||||
|
batch.setBlock(this.lowerLeftCorner.add(x, y, 0), Block.GLASS);
|
||||||
|
batch.setBlock(this.lowerLeftCorner.add(x, y, -1), Block.BLACK_CONCRETE);
|
||||||
|
|
||||||
|
if(x==0 || x==11 || y==0) {
|
||||||
|
batch.setBlock(this.lowerLeftCorner.add(x, y, 1), Block.GRAY_CONCRETE);
|
||||||
|
batch.setBlock(this.lowerLeftCorner.add(x, y, 0), Block.GRAY_CONCRETE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// hold position:
|
||||||
|
for(int x = 0; x < 4; x++) {
|
||||||
|
for(int y = 0; y < 4; y++) {
|
||||||
|
batch.setBlock(this.getHoldPosition().add(x-1, y-1, -1), Block.QUARTZ_BLOCK);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// next positions:
|
||||||
|
for(int x = 0; x < 4; x++) {
|
||||||
|
for(int y = -4*this.nextTetrominoesCount+4; y < 4; y++) {
|
||||||
|
batch.setBlock(this.getNextPosition().add(x-1, y-1, -1), Block.QUARTZ_BLOCK);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
batch.setBlock(this.getPlayerSpawnPosition().sub(0, 1, 0), Block.STONE);
|
||||||
|
batch.setBlock(this.getPlayerSpawnPosition().sub(1, 1, 0), Block.STONE);
|
||||||
|
batch.setBlock(this.getPlayerSpawnPosition().sub(1, 1, 1), Block.STONE);
|
||||||
|
batch.setBlock(this.getPlayerSpawnPosition().sub(0, 1, 1), Block.STONE);
|
||||||
|
|
||||||
|
BatchUtil.loadAndApplyBatch(batch, this.instance, () -> {});
|
||||||
|
}
|
||||||
|
|
||||||
|
public int removeFullLines() {
|
||||||
|
int removedLinesCounter = 0;
|
||||||
|
for(int y = 1; y< height; y++) {
|
||||||
|
boolean isFullLine = true;
|
||||||
|
for(int x=1; x<11; x++) {
|
||||||
|
if(this.instance.getBlock(this.lowerLeftCorner.add(x, y, 1)) == Block.AIR) isFullLine = false;
|
||||||
|
}
|
||||||
|
if(isFullLine) {
|
||||||
|
this.removeFullLine(y);
|
||||||
|
removedLinesCounter += 1;
|
||||||
|
y -= 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return removedLinesCounter;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void addLines(int lines) {
|
||||||
|
int xPosMissing = this.random.nextInt(1, 10);
|
||||||
|
|
||||||
|
for (int i = 0; i < lines; i++) {
|
||||||
|
this.moveAllLinesUp();
|
||||||
|
for (int x = 1; x < 11; x++) {
|
||||||
|
if(x != xPosMissing) {
|
||||||
|
this.instance.setBlock(this.lowerLeftCorner.add(x, 1, 1), Block.LIGHT_GRAY_CONCRETE);
|
||||||
|
} else {
|
||||||
|
this.instance.setBlock(this.lowerLeftCorner.add(x, 1, 1), Block.AIR);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void updateAttackingLines(int attackingLines) {
|
||||||
|
for (int y = 0; y < height + 5; y++) {
|
||||||
|
if(attackingLines > 0) {
|
||||||
|
this.instance.setBlock(this.lowerLeftCorner.add(12, y, 1), Block.REDSTONE_BLOCK);
|
||||||
|
attackingLines -= 1;
|
||||||
|
} else {
|
||||||
|
this.instance.setBlock(this.lowerLeftCorner.add(12, y, 1), Block.AIR);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void updateScore(int score) {
|
||||||
|
this.removeDigits();
|
||||||
|
String scoreString = String.valueOf(score);
|
||||||
|
char[] characters = scoreString.toCharArray();
|
||||||
|
ArrayUtils.reverse(characters);
|
||||||
|
for(int i = 6; i > 0; i--) {
|
||||||
|
char digit;
|
||||||
|
if(i <= characters.length) {
|
||||||
|
digit = characters[i-1];
|
||||||
|
} else {
|
||||||
|
digit = '0';
|
||||||
|
}
|
||||||
|
this.displayDigit(digit, 6-i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private void displayDigit(char digit, int positionFromLeft) {
|
||||||
|
int[][] digitArray;
|
||||||
|
switch (digit) {
|
||||||
|
case '1' -> digitArray = new int[][]{{0,0,1},{0,1,1},{0,0,1},{0,0,1},{0,0,1}};
|
||||||
|
case '2' -> digitArray = new int[][]{{1,1,1},{0,0,1},{1,1,1},{1,0,0},{1,1,1}};
|
||||||
|
case '3' -> digitArray = new int[][]{{1,1,1},{0,0,1},{0,1,1},{0,0,1},{1,1,1}};
|
||||||
|
case '4' -> digitArray = new int[][]{{1,0,1},{1,0,1},{1,1,1},{0,0,1},{0,0,1}};
|
||||||
|
case '5' -> digitArray = new int[][]{{1,1,1},{1,0,0},{1,1,1},{0,0,1},{1,1,1}};
|
||||||
|
case '6' -> digitArray = new int[][]{{1,1,1},{1,0,0},{1,1,1},{1,0,1},{1,1,1}};
|
||||||
|
case '7' -> digitArray = new int[][]{{1,1,1},{0,0,1},{0,1,0},{0,1,0},{0,1,0}};
|
||||||
|
case '8' -> digitArray = new int[][]{{1,1,1},{1,0,1},{1,1,1},{1,0,1},{1,1,1}};
|
||||||
|
case '9' -> digitArray = new int[][]{{1,1,1},{1,0,1},{1,1,1},{0,0,1},{1,1,1}};
|
||||||
|
default -> digitArray = new int[][]{{1,1,1},{1,0,1},{1,0,1},{1,0,1},{1,1,1}};
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int x = 0; x < 3; x++) {
|
||||||
|
for (int y = 0; y < 5; y++) {
|
||||||
|
if(digitArray[4-y][x] == 1) this.instance.setBlock(this.getScorePosition().add(positionFromLeft*4+x, y, 0), scoreBlock);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void removeDigits() {
|
||||||
|
for (int x = 0; x < 4 * 6; x++) {
|
||||||
|
for (int y = 0; y < 5; y++) {
|
||||||
|
this.instance.setBlock(this.getScorePosition().add(x, y, 0), Block.AIR);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void moveAllLinesUp() {
|
||||||
|
for (int y = height + 3; y > 1; y--) {
|
||||||
|
for (int x = 1; x < 11; x++) {
|
||||||
|
Block blockBeneath = this.instance.getBlock(this.lowerLeftCorner.add(x, y - 1, 1));
|
||||||
|
this.instance.setBlock(this.lowerLeftCorner.add(x, y, 1), blockBeneath);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void removeFullLine(int positionY) {
|
||||||
|
for(int y = positionY; y< height; y++) {
|
||||||
|
for(int x=1; x<11; x++) {
|
||||||
|
Block blockAbove = this.instance.getBlock(this.lowerLeftCorner.add(x, y+1, 1));
|
||||||
|
this.instance.setBlock(this.lowerLeftCorner.add(x, y, 1), blockAbove);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,318 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.tetris.game;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.StatelessGame;
|
||||||
|
import net.kyori.adventure.text.Component;
|
||||||
|
import net.minestom.server.MinecraftServer;
|
||||||
|
import net.minestom.server.coordinate.Pos;
|
||||||
|
import net.minestom.server.scoreboard.Sidebar;
|
||||||
|
import net.minestom.server.timer.Scheduler;
|
||||||
|
import net.minestom.server.timer.TaskSchedule;
|
||||||
|
|
||||||
|
import java.util.*;
|
||||||
|
|
||||||
|
public class TetrisGame {
|
||||||
|
private final StatelessGame instance;
|
||||||
|
private final Playfield playfield;
|
||||||
|
private final boolean isFast;
|
||||||
|
private int level = 1;
|
||||||
|
private int lines = 0;
|
||||||
|
private int score = 0;
|
||||||
|
private int combo = 0;
|
||||||
|
private int attackingLines = 0;
|
||||||
|
public boolean lost = false;
|
||||||
|
public boolean paused = true;
|
||||||
|
private final boolean hasCombat;
|
||||||
|
public Tetromino currentTetromino;
|
||||||
|
private final List<Tetromino> nextTetrominoes = new ArrayList<>();
|
||||||
|
private Tetromino holdTetromino;
|
||||||
|
private final List<Tetromino> tetrominoBag = new ArrayList<>();
|
||||||
|
private boolean holdPossible = true;
|
||||||
|
private final Pos nextPosition;
|
||||||
|
private final Pos holdPosition;
|
||||||
|
private final Pos tetrominoSpawnPosition;
|
||||||
|
public Sidebar sidebar = new Sidebar(Component.text("Info:"));
|
||||||
|
private final Map<Button, Long> lastPresses = new HashMap<>();
|
||||||
|
private final List<TetrisGame> otherTetrisGames = new ArrayList<>();
|
||||||
|
private final Random random;
|
||||||
|
|
||||||
|
public enum Button {
|
||||||
|
W,
|
||||||
|
A,
|
||||||
|
S,
|
||||||
|
D,
|
||||||
|
mouseLeft,
|
||||||
|
mouseRight,
|
||||||
|
space
|
||||||
|
}
|
||||||
|
|
||||||
|
public TetrisGame(StatelessGame instance, Pos lowerLeftCorner, Tetromino.Shape startTetrominoShape, int nextTetrominoesCount, boolean isfast, boolean hasCombat, long randomSeed) {
|
||||||
|
this.isFast = isfast;
|
||||||
|
this.hasCombat = hasCombat;
|
||||||
|
this.instance = instance;
|
||||||
|
this.playfield = new Playfield(lowerLeftCorner, this.instance, nextTetrominoesCount);
|
||||||
|
this.random = new Random(randomSeed);
|
||||||
|
|
||||||
|
this.holdPosition = this.playfield.getHoldPosition();
|
||||||
|
this.nextPosition = this.playfield.getNextPosition();
|
||||||
|
this.tetrominoSpawnPosition = this.playfield.getTetrominoSpawnPosition();
|
||||||
|
this.buildSidebar();
|
||||||
|
|
||||||
|
this.currentTetromino = new Tetromino(this.instance, startTetrominoShape);
|
||||||
|
for (int i = 0; i < nextTetrominoesCount; i++) {
|
||||||
|
this.updateNextTetrominoes();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void pressedButton(Button button) {
|
||||||
|
final int standardButtonDelay = 100;
|
||||||
|
final int buttonDebounce = 70;
|
||||||
|
|
||||||
|
if(this.lastPresses.getOrDefault(button, 0L) >= System.currentTimeMillis()-standardButtonDelay) return;
|
||||||
|
|
||||||
|
this.lastPresses.put(button, System.currentTimeMillis());
|
||||||
|
if(button == Button.W) this.lastPresses.put(button, System.currentTimeMillis()+buttonDebounce);
|
||||||
|
if(button == Button.S) this.lastPresses.put(button, System.currentTimeMillis()-buttonDebounce);
|
||||||
|
|
||||||
|
if(this.lost || this.paused) return;
|
||||||
|
|
||||||
|
switch (button) {
|
||||||
|
case A -> this.currentTetromino.moveLeft();
|
||||||
|
case S -> this.moveDown();
|
||||||
|
case D -> this.currentTetromino.moveRight();
|
||||||
|
case W -> this.hardDrop();
|
||||||
|
case mouseLeft -> this.currentTetromino.rotate(false);
|
||||||
|
case mouseRight -> this.currentTetromino.rotate(true);
|
||||||
|
case space -> this.switchHold();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public Pos getPlayerSpawnPosition() {
|
||||||
|
return this.playfield.getPlayerSpawnPosition();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void start() {
|
||||||
|
this.paused = false;
|
||||||
|
Scheduler scheduler = MinecraftServer.getSchedulerManager();
|
||||||
|
scheduler.submitTask(() -> {
|
||||||
|
if(this.lost) return TaskSchedule.stop();
|
||||||
|
int standardTickDelay = 40;
|
||||||
|
if(this.isFast) standardTickDelay = 20;
|
||||||
|
|
||||||
|
TaskSchedule nextTick = TaskSchedule.tick(Math.round((float) standardTickDelay /this.level));
|
||||||
|
if(this.paused) return nextTick;
|
||||||
|
this.tick();
|
||||||
|
return nextTick;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.updateInfo();
|
||||||
|
this.nextTetrominoes.forEach(tetromino -> {
|
||||||
|
double xChange = -tetromino.getXChange();
|
||||||
|
tetromino.setPosition(this.nextPosition.sub(xChange, 4*this.nextTetrominoes.indexOf(tetromino), 0));
|
||||||
|
tetromino.drawAsEntities();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void generate() {
|
||||||
|
this.playfield.generate();
|
||||||
|
|
||||||
|
this.currentTetromino.setPosition(this.tetrominoSpawnPosition);
|
||||||
|
this.currentTetromino.draw(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void tick() {
|
||||||
|
if(this.lost || this.paused) return;
|
||||||
|
if(!this.currentTetromino.moveDown()) {
|
||||||
|
this.setActiveTetrominoDown();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public int getScore() {
|
||||||
|
return this.score;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void updateOtherTetrisGames(Collection<TetrisGame> tetrisGames) {
|
||||||
|
List<TetrisGame> games = new ArrayList<>(tetrisGames);
|
||||||
|
games.remove(this);
|
||||||
|
games.removeIf(tetrisGame -> tetrisGame.lost);
|
||||||
|
this.otherTetrisGames.clear();
|
||||||
|
this.otherTetrisGames.addAll(games);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void getAttacked(int lines) {
|
||||||
|
if(this.hasCombat && !this.lost) {
|
||||||
|
this.attackingLines += lines;
|
||||||
|
this.playfield.updateAttackingLines(this.attackingLines);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void loose() {
|
||||||
|
this.lost = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private boolean moveDown() {
|
||||||
|
if(!this.currentTetromino.moveDown()) {
|
||||||
|
this.setActiveTetrominoDown();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
this.score += 1;
|
||||||
|
this.updateInfo();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean hardDrop() {
|
||||||
|
if(!this.currentTetromino.moveDown()) {
|
||||||
|
this.setActiveTetrominoDown();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
this.score += 2;
|
||||||
|
this.updateInfo();
|
||||||
|
while(this.currentTetromino.moveDown()) {
|
||||||
|
this.score += 2;
|
||||||
|
this.updateInfo();
|
||||||
|
}
|
||||||
|
this.setActiveTetrominoDown();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean switchHold() {
|
||||||
|
if(!this.holdPossible) return false;
|
||||||
|
|
||||||
|
Tetromino newCurrentTetromino;
|
||||||
|
if(this.holdTetromino == null) {
|
||||||
|
newCurrentTetromino = this.nextTetrominoes.removeFirst();
|
||||||
|
newCurrentTetromino.remove();
|
||||||
|
this.updateNextTetrominoes();
|
||||||
|
} else {
|
||||||
|
newCurrentTetromino = this.holdTetromino;
|
||||||
|
this.holdTetromino.remove();
|
||||||
|
}
|
||||||
|
|
||||||
|
this.currentTetromino.remove();
|
||||||
|
this.holdTetromino = new Tetromino(this.instance, this.currentTetromino.getShape());
|
||||||
|
this.currentTetromino = newCurrentTetromino;
|
||||||
|
|
||||||
|
this.currentTetromino.setPosition(this.tetrominoSpawnPosition);
|
||||||
|
this.currentTetromino.draw();
|
||||||
|
if(!this.currentTetromino.moveDown()) this.loose();
|
||||||
|
|
||||||
|
double xChange = this.holdTetromino.getXChange();
|
||||||
|
this.holdTetromino.setPosition(this.holdPosition.add(xChange, 0, 0));
|
||||||
|
this.holdTetromino.drawAsEntities();
|
||||||
|
this.holdPossible = false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private void updateNextTetrominoes() {
|
||||||
|
if(this.tetrominoBag.isEmpty()) {
|
||||||
|
for(Tetromino.Shape shape : Tetromino.Shape.values()) {
|
||||||
|
this.tetrominoBag.add(new Tetromino(this.instance, shape));
|
||||||
|
}
|
||||||
|
Collections.shuffle(this.tetrominoBag, this.random);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(!this.nextTetrominoes.isEmpty()) this.nextTetrominoes.forEach(Tetromino::remove);
|
||||||
|
Tetromino newTetromino = this.tetrominoBag.removeFirst();
|
||||||
|
this.nextTetrominoes.add(newTetromino);
|
||||||
|
this.nextTetrominoes.forEach(tetromino -> {
|
||||||
|
double xChange = -tetromino.getXChange();
|
||||||
|
tetromino.setPosition(this.nextPosition.sub(xChange, 4*this.nextTetrominoes.indexOf(tetromino), 0));
|
||||||
|
tetromino.drawAsEntities();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private void buildSidebar() {
|
||||||
|
this.sidebar.createLine(new Sidebar.ScoreboardLine(
|
||||||
|
"0",
|
||||||
|
Component.text("Score: "),
|
||||||
|
0
|
||||||
|
));
|
||||||
|
this.sidebar.createLine(new Sidebar.ScoreboardLine(
|
||||||
|
"1",
|
||||||
|
Component.text("Lines: "),
|
||||||
|
0
|
||||||
|
));
|
||||||
|
this.sidebar.createLine(new Sidebar.ScoreboardLine(
|
||||||
|
"2",
|
||||||
|
Component.text("Level: "),
|
||||||
|
1
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
private void updateInfo() {
|
||||||
|
this.playfield.updateScore(this.score);
|
||||||
|
this.sidebar.updateLineScore("0", this.score);
|
||||||
|
this.sidebar.updateLineScore("1", this.lines);
|
||||||
|
this.sidebar.updateLineScore("2", this.level);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setActiveTetrominoDown() {
|
||||||
|
this.currentTetromino.removeOwnEntities();
|
||||||
|
this.currentTetromino = this.nextTetrominoes.removeFirst();
|
||||||
|
this.currentTetromino.remove();
|
||||||
|
this.updateNextTetrominoes();
|
||||||
|
|
||||||
|
int removedLines = this.playfield.removeFullLines();
|
||||||
|
int combatLines = 0;
|
||||||
|
this.combo += 1;
|
||||||
|
switch (removedLines) {
|
||||||
|
case 0 -> this.combo = 0;
|
||||||
|
case 1 -> {
|
||||||
|
this.lines += 1;
|
||||||
|
this.score += 40 * this.level;
|
||||||
|
}
|
||||||
|
case 2 -> {
|
||||||
|
combatLines = 1;
|
||||||
|
this.lines += 2;
|
||||||
|
this.score += 100 * this.level;
|
||||||
|
}
|
||||||
|
case 3 -> {
|
||||||
|
combatLines = 2;
|
||||||
|
this.lines += 3;
|
||||||
|
this.score += 300 * this.level;
|
||||||
|
}
|
||||||
|
case 4 -> {
|
||||||
|
combatLines = 4;
|
||||||
|
this.lines += 4;
|
||||||
|
this.score += 1200 * this.level;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.score += 50 * this.combo * this.level;
|
||||||
|
if(this.combo >= 2) {
|
||||||
|
combatLines += (int) Math.floor((double) this.combo /2);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(this.hasCombat && this.attackingLines > 0) {
|
||||||
|
if(combatLines > 0 && this.attackingLines >= combatLines) {
|
||||||
|
this.attackingLines -= combatLines;
|
||||||
|
combatLines = 0;
|
||||||
|
} else if(combatLines > 0) {
|
||||||
|
combatLines -= this.attackingLines;
|
||||||
|
this.attackingLines = 0;
|
||||||
|
} else {
|
||||||
|
this.playfield.addLines(this.attackingLines);
|
||||||
|
this.attackingLines = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.playfield.updateAttackingLines(this.attackingLines);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(this.hasCombat && !this.otherTetrisGames.isEmpty()) {
|
||||||
|
Collections.shuffle(this.otherTetrisGames);
|
||||||
|
this.otherTetrisGames.getFirst().getAttacked(combatLines);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.level = (int) Math.floor((double) this.lines / 10) + 1;
|
||||||
|
this.holdPossible = true;
|
||||||
|
|
||||||
|
this.updateInfo();
|
||||||
|
|
||||||
|
this.currentTetromino.setPosition(this.tetrominoSpawnPosition);
|
||||||
|
this.currentTetromino.draw();
|
||||||
|
if(!this.currentTetromino.moveDown()) {
|
||||||
|
this.loose();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@@ -0,0 +1,246 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.tetris.game;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.StatelessGame;
|
||||||
|
import net.minestom.server.coordinate.Pos;
|
||||||
|
import net.minestom.server.entity.Entity;
|
||||||
|
import net.minestom.server.entity.EntityType;
|
||||||
|
import net.minestom.server.entity.metadata.other.FallingBlockMeta;
|
||||||
|
import net.minestom.server.instance.block.Block;
|
||||||
|
import net.minestom.server.tag.Tag;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
public class Tetromino {
|
||||||
|
private final Shape shape;
|
||||||
|
private final StatelessGame instance;
|
||||||
|
private Pos position;
|
||||||
|
private int[][] shapeArray;
|
||||||
|
private final static EntityType ghostEntityType = EntityType.FALLING_BLOCK;
|
||||||
|
private final UUID uuid;
|
||||||
|
private final static Tag<String> uuidTag = Tag.String("uuid");
|
||||||
|
|
||||||
|
public enum Shape {
|
||||||
|
I,
|
||||||
|
J,
|
||||||
|
L,
|
||||||
|
O,
|
||||||
|
S,
|
||||||
|
T,
|
||||||
|
Z
|
||||||
|
}
|
||||||
|
|
||||||
|
public Tetromino(StatelessGame instance, Shape shape) {
|
||||||
|
this.instance = instance;
|
||||||
|
this.shape = shape;
|
||||||
|
this.uuid = UUID.randomUUID();
|
||||||
|
|
||||||
|
switch (this.shape) {
|
||||||
|
case I -> this.shapeArray = new int[][]{{0, 0, 0, 0}, {1, 1, 1, 1}, {0, 0, 0, 0}, {0, 0, 0, 0}};
|
||||||
|
case J -> this.shapeArray = new int[][]{{1,0,0}, {1,1,1}, {0,0,0}};
|
||||||
|
case L -> this.shapeArray = new int[][]{{0,0,1}, {1,1,1}, {0,0,0}};
|
||||||
|
case O -> this.shapeArray = new int[][]{{1,1}, {1,1}};
|
||||||
|
case S -> this.shapeArray = new int[][]{{0,1,1}, {1,1,0}, {0,0,0}};
|
||||||
|
case T -> this.shapeArray = new int[][]{{0,1,0}, {1,1,1}, {0,0,0}};
|
||||||
|
case Z -> this.shapeArray = new int[][]{{1,1,0}, {0,1,1}, {0,0,0}};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static EntityType getGhostEntityType() {
|
||||||
|
return ghostEntityType;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setPosition(Pos newPosition) {
|
||||||
|
this.position = newPosition;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean rotate(boolean clockwise) {
|
||||||
|
int[][] newShapeArray = this.getTurnedShapeArray(clockwise);
|
||||||
|
return this.checkCollisionAndMove(this.position, newShapeArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean moveDown() {
|
||||||
|
Pos newPosition = this.position.sub(0, 1, 0);
|
||||||
|
return this.checkCollisionAndMove(newPosition, this.shapeArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean moveLeft() {
|
||||||
|
Pos newPosition = this.position.sub(1, 0, 0);
|
||||||
|
return this.checkCollisionAndMove(newPosition, this.shapeArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean moveRight() {
|
||||||
|
Pos newPosition = this.position.add(1, 0, 0);
|
||||||
|
return this.checkCollisionAndMove(newPosition, this.shapeArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void draw() {
|
||||||
|
this.draw(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void draw(boolean withGhost) {
|
||||||
|
if(withGhost) {
|
||||||
|
Pos ghostPos = this.position;
|
||||||
|
while (!this.checkCollision(ghostPos.sub(0, 1, 0), this.shapeArray)) {
|
||||||
|
ghostPos = ghostPos.sub(0, 1, 0);
|
||||||
|
}
|
||||||
|
Pos positionChange = this.position.sub(ghostPos);
|
||||||
|
this.getBlockPositions().forEach(pos -> {
|
||||||
|
Entity ghostBlock = new Entity(ghostEntityType);
|
||||||
|
((FallingBlockMeta) ghostBlock.getEntityMeta()).setBlock(this.getGhostBlock());
|
||||||
|
ghostBlock.setNoGravity(true);
|
||||||
|
ghostBlock.setGlowing(true);
|
||||||
|
ghostBlock.setTag(uuidTag, this.uuid.toString());
|
||||||
|
ghostBlock.setInstance(this.instance, pos.sub(positionChange).add(0.5, 0, 0.5));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
this.getBlockPositions().forEach(pos -> this.instance.setBlock(pos, this.getColoredBlock()));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void drawAsEntities() {
|
||||||
|
this.getBlockPositions().forEach(pos -> {
|
||||||
|
Entity ghostBlock = new Entity(ghostEntityType);
|
||||||
|
((FallingBlockMeta) ghostBlock.getEntityMeta()).setBlock(this.getColoredBlock());
|
||||||
|
ghostBlock.setNoGravity(true);
|
||||||
|
ghostBlock.setTag(uuidTag, this.uuid.toString());
|
||||||
|
ghostBlock.setInstance(this.instance, pos.add(0.5, 0, 0.5));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void remove() {
|
||||||
|
this.removeOwnEntities();
|
||||||
|
this.getBlockPositions().forEach(pos -> this.instance.setBlock(pos, Block.AIR));
|
||||||
|
}
|
||||||
|
|
||||||
|
public Block getColoredBlock() {
|
||||||
|
Block returnBlock;
|
||||||
|
switch (this.shape) {
|
||||||
|
case I -> returnBlock = Block.LIGHT_BLUE_CONCRETE;
|
||||||
|
case J -> returnBlock = Block.BLUE_CONCRETE;
|
||||||
|
case L -> returnBlock = Block.ORANGE_CONCRETE;
|
||||||
|
case O -> returnBlock = Block.YELLOW_CONCRETE;
|
||||||
|
case S -> returnBlock = Block.GREEN_CONCRETE;
|
||||||
|
case T -> returnBlock = Block.PURPLE_CONCRETE;
|
||||||
|
case Z -> returnBlock = Block.RED_CONCRETE;
|
||||||
|
default -> returnBlock = Block.WHITE_CONCRETE;
|
||||||
|
}
|
||||||
|
return returnBlock;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void removeOwnEntities() {
|
||||||
|
this.instance.getEntities().stream()
|
||||||
|
.filter(entity -> {
|
||||||
|
String tagValue = entity.getTag(uuidTag);
|
||||||
|
if(tagValue == null) return false;
|
||||||
|
return entity.getTag(uuidTag).equals(this.uuid.toString());
|
||||||
|
})
|
||||||
|
.forEach(Entity::remove);
|
||||||
|
}
|
||||||
|
|
||||||
|
public double getXChange() {
|
||||||
|
switch (this.shape) {
|
||||||
|
case O, I -> {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
case null, default -> {
|
||||||
|
return 0.5;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public Shape getShape() {
|
||||||
|
return this.shape;
|
||||||
|
}
|
||||||
|
|
||||||
|
private Block getGhostBlock() {
|
||||||
|
Block returnBlock;
|
||||||
|
switch (this.shape) {
|
||||||
|
case I -> returnBlock = Block.LIGHT_BLUE_STAINED_GLASS;
|
||||||
|
case J -> returnBlock = Block.BLUE_STAINED_GLASS;
|
||||||
|
case L -> returnBlock = Block.ORANGE_STAINED_GLASS;
|
||||||
|
case O -> returnBlock = Block.YELLOW_STAINED_GLASS;
|
||||||
|
case S -> returnBlock = Block.GREEN_STAINED_GLASS;
|
||||||
|
case T -> returnBlock = Block.PURPLE_STAINED_GLASS;
|
||||||
|
case Z -> returnBlock = Block.RED_STAINED_GLASS;
|
||||||
|
default -> returnBlock = Block.WHITE_STAINED_GLASS;
|
||||||
|
}
|
||||||
|
return returnBlock;
|
||||||
|
}
|
||||||
|
|
||||||
|
private int[][] getTurnedShapeArray(boolean clockwise) {
|
||||||
|
int iterations = 1;
|
||||||
|
if(!clockwise) iterations = 3;
|
||||||
|
|
||||||
|
int arrayLength = this.shapeArray.length;
|
||||||
|
int[][] startArray = Arrays.stream(this.shapeArray).map(int[]::clone).toArray(int[][]::new);
|
||||||
|
int[][] returnArray = new int[arrayLength][arrayLength];
|
||||||
|
|
||||||
|
for(int k=0; k<iterations; k++) {
|
||||||
|
for(int i=0; i<arrayLength; i++) {
|
||||||
|
for(int j=0; j<arrayLength; j++) {
|
||||||
|
returnArray[i][arrayLength-1-j] = startArray[j][i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
startArray = Arrays.stream(returnArray).map(int[]::clone).toArray(int[][]::new);
|
||||||
|
}
|
||||||
|
|
||||||
|
return returnArray;
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean isPartOfTetromino(Pos position) {
|
||||||
|
return this.getBlockPositions().stream()
|
||||||
|
.anyMatch(pos -> pos.equals(position));
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<Pos> getBlockPositions() {
|
||||||
|
return this.getBlockPositions(this.position, this.shapeArray);
|
||||||
|
}
|
||||||
|
|
||||||
|
private List<Pos> getBlockPositions(Pos position, int[][] shapeArray) {
|
||||||
|
List<Pos> returnList = new ArrayList<>();
|
||||||
|
if(position == null) return returnList;
|
||||||
|
|
||||||
|
int arrayLength = shapeArray.length;
|
||||||
|
|
||||||
|
for(int x=0; x<arrayLength; x++) {
|
||||||
|
for(int y=0; y<arrayLength; y++) {
|
||||||
|
if(shapeArray[arrayLength-1-y][x] == 1) {
|
||||||
|
switch (this.shape) {
|
||||||
|
case I -> returnList.add(position.add(x-1, y-2, 0));
|
||||||
|
case O -> returnList.add(position.add(x, y, 0));
|
||||||
|
default -> returnList.add(position.add(x-1, y-1, 0));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return returnList;
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean checkCollision(Pos newPosition, int[][] newShapeArray) {
|
||||||
|
List<Pos> newBlockPositions = this.getBlockPositions(newPosition, newShapeArray);
|
||||||
|
|
||||||
|
for(Pos pos : newBlockPositions) {
|
||||||
|
if(this.isPartOfTetromino(pos)) continue;
|
||||||
|
if(this.instance.getBlock(pos) == this.getGhostBlock()) continue;
|
||||||
|
if(this.instance.getBlock(pos) != Block.AIR) return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean checkCollisionAndMove(Pos newPosition, int[][] newShapeArray) {
|
||||||
|
if(!this.checkCollision(newPosition, newShapeArray)) {
|
||||||
|
this.remove();
|
||||||
|
this.shapeArray = Arrays.stream(newShapeArray).map(int[]::clone).toArray(int[][]::new);
|
||||||
|
this.setPosition(newPosition);
|
||||||
|
this.draw();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
@@ -7,10 +7,7 @@ import eu.mhsl.minenet.minigames.util.BatchUtil;
|
|||||||
import eu.mhsl.minenet.minigames.world.generator.terrain.CircularPlateTerrainGenerator;
|
import eu.mhsl.minenet.minigames.world.generator.terrain.CircularPlateTerrainGenerator;
|
||||||
import net.minestom.server.MinecraftServer;
|
import net.minestom.server.MinecraftServer;
|
||||||
import net.minestom.server.coordinate.Pos;
|
import net.minestom.server.coordinate.Pos;
|
||||||
import net.minestom.server.entity.Entity;
|
|
||||||
import net.minestom.server.entity.EntityType;
|
|
||||||
import net.minestom.server.entity.GameMode;
|
import net.minestom.server.entity.GameMode;
|
||||||
import net.minestom.server.entity.metadata.other.PrimedTntMeta;
|
|
||||||
import net.minestom.server.event.player.PlayerMoveEvent;
|
import net.minestom.server.event.player.PlayerMoveEvent;
|
||||||
import net.minestom.server.instance.batch.AbsoluteBlockBatch;
|
import net.minestom.server.instance.batch.AbsoluteBlockBatch;
|
||||||
import net.minestom.server.instance.block.Block;
|
import net.minestom.server.instance.block.Block;
|
||||||
@@ -27,7 +24,7 @@ public class TntRun extends StatelessGame {
|
|||||||
final int radius;
|
final int radius;
|
||||||
final int stackCount;
|
final int stackCount;
|
||||||
public TntRun(int radius, int stackCount) {
|
public TntRun(int radius, int stackCount) {
|
||||||
super(Dimension.OVERWORLD.DIMENSION, "tntRun", new LastWinsScore());
|
super(Dimension.OVERWORLD.key, "tntRun", new LastWinsScore());
|
||||||
this.radius = radius;
|
this.radius = radius;
|
||||||
this.stackCount = stackCount;
|
this.stackCount = stackCount;
|
||||||
setGenerator(new CircularPlateTerrainGenerator(radius));
|
setGenerator(new CircularPlateTerrainGenerator(radius));
|
||||||
@@ -51,7 +48,13 @@ public class TntRun extends StatelessGame {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void onPlayerMove(@NotNull PlayerMoveEvent playerMoveEvent) {
|
protected void onPlayerMove(@NotNull PlayerMoveEvent playerMoveEvent) {
|
||||||
|
|
||||||
|
|
||||||
if(playerMoveEvent.getNewPosition().y() < totalElevation) {
|
if(playerMoveEvent.getNewPosition().y() < totalElevation) {
|
||||||
|
if(isBeforeBeginning) {
|
||||||
|
playerMoveEvent.getPlayer().teleport(getSpawn());
|
||||||
|
return;
|
||||||
|
}
|
||||||
playerMoveEvent.getPlayer().setGameMode(GameMode.SPECTATOR);
|
playerMoveEvent.getPlayer().setGameMode(GameMode.SPECTATOR);
|
||||||
getScore().insertResult(playerMoveEvent.getPlayer());
|
getScore().insertResult(playerMoveEvent.getPlayer());
|
||||||
}
|
}
|
||||||
@@ -67,14 +70,14 @@ public class TntRun extends StatelessGame {
|
|||||||
setBlock(firstLocation, Block.AIR);
|
setBlock(firstLocation, Block.AIR);
|
||||||
setBlock(secondLocation, Block.AIR);
|
setBlock(secondLocation, Block.AIR);
|
||||||
|
|
||||||
Entity fallingTnt = new Entity(EntityType.TNT);
|
// Entity fallingTnt = new Entity(EntityType.TNT);
|
||||||
PrimedTntMeta fallingTntMeta = (PrimedTntMeta) fallingTnt.getEntityMeta();
|
// PrimedTntMeta fallingTntMeta = (PrimedTntMeta) fallingTnt.getEntityMeta();
|
||||||
fallingTntMeta.setFuseTime(20);
|
// fallingTntMeta.setFuseTime(20);
|
||||||
fallingTnt.setInstance(this, secondLocation);
|
// fallingTnt.setInstance(this, secondLocation);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}, TaskSchedule.millis(500), TaskSchedule.stop(), ExecutionType.ASYNC);
|
}, TaskSchedule.millis(500), TaskSchedule.stop(), ExecutionType.TICK_END);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -14,7 +14,12 @@ import java.util.Map;
|
|||||||
public class TntRunFactory implements GameFactory {
|
public class TntRunFactory implements GameFactory {
|
||||||
@Override
|
@Override
|
||||||
public TranslatedComponent name() {
|
public TranslatedComponent name() {
|
||||||
return TranslatedComponent.byId("game_tntRun#name");
|
return TranslatedComponent.byId("game_TntRun#name");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TranslatedComponent description() {
|
||||||
|
return TranslatedComponent.byId("game_TntRun#description");
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@@ -25,8 +30,8 @@ public class TntRunFactory implements GameFactory {
|
|||||||
@Override
|
@Override
|
||||||
public ConfigManager configuration() {
|
public ConfigManager configuration() {
|
||||||
return new ConfigManager()
|
return new ConfigManager()
|
||||||
.addOption(new NumericOption("radius", Material.STICK, TranslatedComponent.byId("game_tntRun#radius"), 20, 30, 50, 60))
|
.addOption(new NumericOption("radius", Material.STICK, TranslatedComponent.byId("optionCommon#radius"), 20, 30, 50, 60))
|
||||||
.addOption(new NumericOption("levels", Material.SCAFFOLDING, TranslatedComponent.byId("game_tntRun#levels"), 1, 2, 3, 4, 5));
|
.addOption(new NumericOption("levels", Material.SCAFFOLDING, TranslatedComponent.byId("game_TntRun#levels"), 1, 2, 3, 4, 5));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@@ -3,23 +3,74 @@ package eu.mhsl.minenet.minigames.instance.game.stateless.types.towerdefense;
|
|||||||
import eu.mhsl.minenet.minigames.instance.Dimension;
|
import eu.mhsl.minenet.minigames.instance.Dimension;
|
||||||
import eu.mhsl.minenet.minigames.instance.game.stateless.StatelessGame;
|
import eu.mhsl.minenet.minigames.instance.game.stateless.StatelessGame;
|
||||||
import eu.mhsl.minenet.minigames.instance.game.stateless.types.towerdefense.generator.MazeGenerator;
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.towerdefense.generator.MazeGenerator;
|
||||||
import eu.mhsl.minenet.minigames.score.NoScore;
|
import eu.mhsl.minenet.minigames.score.LastWinsScore;
|
||||||
|
import net.minestom.server.coordinate.Pos;
|
||||||
|
import net.minestom.server.entity.EntityType;
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.item.ItemStack;
|
import net.minestom.server.instance.batch.AbsoluteBlockBatch;
|
||||||
import net.minestom.server.item.Material;
|
import net.minestom.server.instance.block.Block;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Random;
|
||||||
|
|
||||||
public class Towerdefense extends StatelessGame {
|
public class Towerdefense extends StatelessGame {
|
||||||
|
private final Random random = new Random();
|
||||||
|
private final AbsoluteBlockBatch mazeBatch = new AbsoluteBlockBatch();
|
||||||
|
private final List<Pos> mazePath = new ArrayList<>();
|
||||||
|
private List<TowerdefenseRoom> instances = new ArrayList<>();
|
||||||
|
|
||||||
public Towerdefense() {
|
public Towerdefense() {
|
||||||
super(Dimension.NETHER.DIMENSION, "Towerdefense", new NoScore());
|
super(Dimension.NETHER.key, "Towerdefense", new LastWinsScore());
|
||||||
|
|
||||||
setGenerator(new MazeGenerator());
|
setGenerator(new MazeGenerator());
|
||||||
|
this.generateMaze();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void generateMaze() {
|
||||||
|
Pos position = new Pos(0, 0, 0);
|
||||||
|
this.addMazePosition(position, Block.GREEN_WOOL);
|
||||||
|
|
||||||
|
List<Integer> previousDirections = new ArrayList<>();
|
||||||
|
int direction = 1; // 0 -> right; 1 -> straight; 2 -> left
|
||||||
|
for (int i = 0; i < 9; i++) {
|
||||||
|
for (int j = 0; j < 3; j++) {
|
||||||
|
position = position.add(direction-1,0,direction%2);
|
||||||
|
this.addMazePosition(position, Block.WHITE_WOOL);
|
||||||
|
}
|
||||||
|
|
||||||
|
int origin = 0;
|
||||||
|
int bound = 3;
|
||||||
|
long rightLeftDifference = previousDirections.stream().filter(integer -> integer == 0).count() - previousDirections.stream().filter(integer -> integer == 2).count();
|
||||||
|
if(rightLeftDifference >= 2 || direction == 2) origin = 1;
|
||||||
|
if(rightLeftDifference <= -2 || direction == 0) bound = 2;
|
||||||
|
direction = this.random.nextInt(origin, bound);
|
||||||
|
previousDirections.add(direction);
|
||||||
|
}
|
||||||
|
this.addMazePosition(position, Block.WHITE_WOOL);
|
||||||
|
this.addMazePosition(position.add(0,0,1), Block.WHITE_WOOL);
|
||||||
|
this.addMazePosition(position.add(0,0,2), Block.RED_WOOL);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void addMazePosition(Pos position, Block pathBlock) {
|
||||||
|
this.mazeBatch.setBlock(position, pathBlock);
|
||||||
|
this.mazePath.add(position.add(0.5,1,0.5));
|
||||||
|
}
|
||||||
|
|
||||||
|
public AbsoluteBlockBatch getMazeBatch() {
|
||||||
|
return this.mazeBatch;
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<Pos> getMazePath() {
|
||||||
|
return this.mazePath;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected boolean onPlayerJoin(Player p) {
|
protected boolean onPlayerJoin(Player p) {
|
||||||
p.getInventory().setItemStack(1, ItemStack.of(Material.ARMOR_STAND));
|
TowerdefenseRoom newRoom = new TowerdefenseRoom(p, this);
|
||||||
|
this.instances.add(newRoom);
|
||||||
|
p.setInstance(newRoom);
|
||||||
|
newRoom.startWave(List.of(EntityType.ENDERMAN, EntityType.BLAZE, EntityType.PLAYER, EntityType.HORSE, EntityType.ARMOR_STAND, EntityType.SKELETON));
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@@ -0,0 +1,58 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.instance.game.stateless.types.towerdefense;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.instance.Dimension;
|
||||||
|
import eu.mhsl.minenet.minigames.instance.game.stateless.types.towerdefense.generator.MazeGenerator;
|
||||||
|
import eu.mhsl.minenet.minigames.util.BatchUtil;
|
||||||
|
import net.minestom.server.MinecraftServer;
|
||||||
|
import net.minestom.server.entity.*;
|
||||||
|
import net.minestom.server.entity.attribute.Attribute;
|
||||||
|
import net.minestom.server.instance.InstanceContainer;
|
||||||
|
import net.minestom.server.item.ItemStack;
|
||||||
|
import net.minestom.server.item.Material;
|
||||||
|
import net.minestom.server.timer.TaskSchedule;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
public class TowerdefenseRoom extends InstanceContainer {
|
||||||
|
private final Player player;
|
||||||
|
private final Towerdefense game;
|
||||||
|
|
||||||
|
public TowerdefenseRoom(Player player, Towerdefense game) {
|
||||||
|
super(UUID.randomUUID(), Dimension.OVERWORLD.key);
|
||||||
|
MinecraftServer.getInstanceManager().registerInstance(this);
|
||||||
|
this.player = player;
|
||||||
|
this.game = game;
|
||||||
|
this.player.setGameMode(GameMode.ADVENTURE);
|
||||||
|
this.player.setAllowFlying(true);
|
||||||
|
this.player.getInventory().setItemStack(0, ItemStack.of(Material.ARMOR_STAND));
|
||||||
|
|
||||||
|
setGenerator(new MazeGenerator());
|
||||||
|
BatchUtil.loadAndApplyBatch(this.game.getMazeBatch(), this, () -> {});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void startWave(List<EntityType> entities) {
|
||||||
|
int counter = 0;
|
||||||
|
for(EntityType entityType : entities) {
|
||||||
|
MinecraftServer.getSchedulerManager().scheduleTask(() -> {
|
||||||
|
this.addEntity(new EntityCreature(entityType));
|
||||||
|
return TaskSchedule.stop();
|
||||||
|
}, TaskSchedule.millis(800L*counter));
|
||||||
|
counter++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void addEntity(EntityCreature entity) {
|
||||||
|
entity.setInstance(this, this.game.getMazePath().getFirst());
|
||||||
|
entity.getAttribute(Attribute.MOVEMENT_SPEED).setBaseValue(0.15);
|
||||||
|
entity.getNavigator().setPathTo(this.game.getMazePath().get(1), 0.7, () -> changeEntityGoal(entity, 1));
|
||||||
|
}
|
||||||
|
|
||||||
|
private void changeEntityGoal(EntityCreature entity, int positionIndex) {
|
||||||
|
if(positionIndex == this.game.getMazePath().size()-1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
entity.getNavigator().setPathTo(this.game.getMazePath().get(positionIndex+1), 0.7, () -> changeEntityGoal(entity, positionIndex+1));
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@@ -37,7 +37,7 @@ class TrafficLightRace extends StatelessGame {
|
|||||||
private final List<Pos> trafficLights = new ArrayList<>();
|
private final List<Pos> trafficLights = new ArrayList<>();
|
||||||
|
|
||||||
public TrafficLightRace(int width, int length) {
|
public TrafficLightRace(int width, int length) {
|
||||||
super(Dimension.THE_END.DIMENSION, "Ampelrennen", new FirstWinsScore());
|
super(Dimension.THE_END.key, "Ampelrennen", new FirstWinsScore());
|
||||||
this.width = width;
|
this.width = width;
|
||||||
this.length = length;
|
this.length = length;
|
||||||
|
|
||||||
@@ -150,7 +150,7 @@ class TrafficLightRace extends StatelessGame {
|
|||||||
BatchUtil.loadAndApplyBatch(changeLightsBatch, this, () -> {});
|
BatchUtil.loadAndApplyBatch(changeLightsBatch, this, () -> {});
|
||||||
|
|
||||||
return phase.taskScheduleRandomDuration();
|
return phase.taskScheduleRandomDuration();
|
||||||
}, ExecutionType.SYNC);
|
}, ExecutionType.TICK_END);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@@ -17,6 +17,11 @@ public class TrafficLightRaceFactory implements GameFactory {
|
|||||||
return TranslatedComponent.byId("game_TrafficlightRace#name");
|
return TranslatedComponent.byId("game_TrafficlightRace#name");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TranslatedComponent description() {
|
||||||
|
return TranslatedComponent.byId("game_TrafficlightRace#description");
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public ConfigManager configuration() {
|
public ConfigManager configuration() {
|
||||||
return new ConfigManager()
|
return new ConfigManager()
|
||||||
|
@@ -9,7 +9,7 @@ import eu.mhsl.minenet.minigames.instance.Dimension;
|
|||||||
import net.minestom.server.MinecraftServer;
|
import net.minestom.server.MinecraftServer;
|
||||||
import net.minestom.server.coordinate.Pos;
|
import net.minestom.server.coordinate.Pos;
|
||||||
import net.minestom.server.event.player.*;
|
import net.minestom.server.event.player.*;
|
||||||
import net.minestom.server.instance.AnvilLoader;
|
import net.minestom.server.instance.anvil.AnvilLoader;
|
||||||
|
|
||||||
import java.nio.file.Path;
|
import java.nio.file.Path;
|
||||||
|
|
||||||
@@ -30,7 +30,7 @@ public class Hub extends MineNetInstance implements Spawnable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private Hub() {
|
private Hub() {
|
||||||
super(Dimension.THE_END.DIMENSION);
|
super(Dimension.THE_END.key);
|
||||||
setChunkLoader(new AnvilLoader(Path.of("maps/hub")));
|
setChunkLoader(new AnvilLoader(Path.of("maps/hub")));
|
||||||
|
|
||||||
setTime(18000);
|
setTime(18000);
|
||||||
|
@@ -5,7 +5,6 @@ import eu.mhsl.minenet.minigames.message.component.TranslatedComponent;
|
|||||||
import eu.mhsl.minenet.minigames.shared.inventory.InteractableInventory;
|
import eu.mhsl.minenet.minigames.shared.inventory.InteractableInventory;
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.inventory.InventoryType;
|
import net.minestom.server.inventory.InventoryType;
|
||||||
import net.minestom.server.item.ItemHideFlag;
|
|
||||||
import net.minestom.server.item.ItemStack;
|
import net.minestom.server.item.ItemStack;
|
||||||
import net.minestom.server.item.Material;
|
import net.minestom.server.item.Material;
|
||||||
|
|
||||||
@@ -16,9 +15,8 @@ public class HubInventory extends InteractableInventory {
|
|||||||
setClickableItem(
|
setClickableItem(
|
||||||
ItemStack
|
ItemStack
|
||||||
.builder(Material.WRITABLE_BOOK)
|
.builder(Material.WRITABLE_BOOK)
|
||||||
.displayName(TranslatedComponent.assemble("hub#create", p))
|
.customName(TranslatedComponent.assemble("hub#create", p))
|
||||||
.lore(TranslatedComponent.assemble("hub#create_description", p))
|
.lore(TranslatedComponent.assemble("hub#create_description", p))
|
||||||
.meta(metaBuilder -> metaBuilder.hideFlag(ItemHideFlag.HIDE_ATTRIBUTES))
|
|
||||||
.build(),
|
.build(),
|
||||||
12,
|
12,
|
||||||
itemClick -> Room.createRoom(itemClick.getPlayer()),
|
itemClick -> Room.createRoom(itemClick.getPlayer()),
|
||||||
@@ -28,7 +26,7 @@ public class HubInventory extends InteractableInventory {
|
|||||||
setClickableItem(
|
setClickableItem(
|
||||||
ItemStack
|
ItemStack
|
||||||
.builder(Material.KNOWLEDGE_BOOK)
|
.builder(Material.KNOWLEDGE_BOOK)
|
||||||
.displayName(TranslatedComponent.assemble("hub#join", p))
|
.customName(TranslatedComponent.assemble("hub#join", p))
|
||||||
.lore(TranslatedComponent.assemble("hub#join_description", p))
|
.lore(TranslatedComponent.assemble("hub#join_description", p))
|
||||||
.build(),
|
.build(),
|
||||||
14,
|
14,
|
||||||
|
@@ -29,7 +29,7 @@ public class JoinInventory extends InteractableInventory {
|
|||||||
|
|
||||||
setClickableItem(
|
setClickableItem(
|
||||||
ItemStack.builder(Material.GREEN_STAINED_GLASS_PANE)
|
ItemStack.builder(Material.GREEN_STAINED_GLASS_PANE)
|
||||||
.displayName(Component.text(prefix))
|
.customName(Component.text(prefix))
|
||||||
.build(),
|
.build(),
|
||||||
0,
|
0,
|
||||||
itemClick -> {}
|
itemClick -> {}
|
||||||
@@ -51,7 +51,7 @@ public class JoinInventory extends InteractableInventory {
|
|||||||
|
|
||||||
typedText = formatInput(typedText);
|
typedText = formatInput(typedText);
|
||||||
|
|
||||||
Optional<Room> target = Room.getRoom(MinecraftServer.getConnectionManager().findPlayer(typedText));
|
Optional<Room> target = Room.getRoom(MinecraftServer.getConnectionManager().findOnlinePlayer(typedText));
|
||||||
if(target.isPresent())
|
if(target.isPresent())
|
||||||
Room.setRoom(player, target.get());
|
Room.setRoom(player, target.get());
|
||||||
else
|
else
|
||||||
|
@@ -17,7 +17,7 @@ import net.minestom.server.entity.GameMode;
|
|||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.event.player.PlayerBlockBreakEvent;
|
import net.minestom.server.event.player.PlayerBlockBreakEvent;
|
||||||
import net.minestom.server.event.player.PlayerDisconnectEvent;
|
import net.minestom.server.event.player.PlayerDisconnectEvent;
|
||||||
import net.minestom.server.instance.AnvilLoader;
|
import net.minestom.server.instance.anvil.AnvilLoader;
|
||||||
|
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
import java.util.logging.Logger;
|
import java.util.logging.Logger;
|
||||||
@@ -41,7 +41,7 @@ public class Room extends MineNetInstance implements Spawnable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public static void deleteRoom(Room room) {
|
public static void deleteRoom(Room room) {
|
||||||
logger.info("Deleting room " + room.uniqueId);
|
logger.info("Deleting room " + room.uuid);
|
||||||
rooms.remove(room);
|
rooms.remove(room);
|
||||||
players.values().removeAll(Collections.singleton(room));
|
players.values().removeAll(Collections.singleton(room));
|
||||||
room.getAllMembers().forEach(player -> MoveInstance.move(player, Hub.INSTANCE));
|
room.getAllMembers().forEach(player -> MoveInstance.move(player, Hub.INSTANCE));
|
||||||
@@ -65,6 +65,9 @@ public class Room extends MineNetInstance implements Spawnable {
|
|||||||
p.clearTitle();
|
p.clearTitle();
|
||||||
p.getInventory().clear();
|
p.getInventory().clear();
|
||||||
p.setGameMode(GameMode.ADVENTURE);
|
p.setGameMode(GameMode.ADVENTURE);
|
||||||
|
p.setInvisible(false);
|
||||||
|
p.setExp(0);
|
||||||
|
p.setLevel(0);
|
||||||
rooms.add(room);
|
rooms.add(room);
|
||||||
players.put(p, room);
|
players.put(p, room);
|
||||||
MoveInstance.move(p, room);
|
MoveInstance.move(p, room);
|
||||||
@@ -85,31 +88,31 @@ public class Room extends MineNetInstance implements Spawnable {
|
|||||||
private GameSelector gameSelector;
|
private GameSelector gameSelector;
|
||||||
private final Tournament tournament = new Tournament();
|
private final Tournament tournament = new Tournament();
|
||||||
private Room(Player owner) {
|
private Room(Player owner) {
|
||||||
super(Dimension.THE_END.DIMENSION);
|
super(Dimension.THE_END.key);
|
||||||
this.apiDriven = false;
|
this.apiDriven = false;
|
||||||
construct();
|
this.construct();
|
||||||
setOwner(owner);
|
this.setOwner(owner);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected Room() {
|
protected Room() {
|
||||||
super(Dimension.THE_END.DIMENSION);
|
super(Dimension.THE_END.key);
|
||||||
this.apiDriven = true;
|
this.apiDriven = true;
|
||||||
construct();
|
this.construct();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void construct() {
|
private void construct() {
|
||||||
MinecraftServer.getInstanceManager().registerInstance(this);
|
MinecraftServer.getInstanceManager().registerInstance(this);
|
||||||
setChunkLoader(new AnvilLoader(Resource.LOBBY_MAP.getPath()));
|
this.setChunkLoader(new AnvilLoader(Resource.LOBBY_MAP.getPath()));
|
||||||
|
|
||||||
this.gameSelector = new GameSelector();
|
this.gameSelector = new GameSelector();
|
||||||
this.gameSelector.setInstance(this, new Pos(0.5, 50, 19.5));
|
this.gameSelector.setInstance(this, new Pos(0.5, 50, 19.5));
|
||||||
|
|
||||||
eventNode().addListener(PlayerBlockBreakEvent.class, CommonEventHandles::cancel);
|
this.eventNode().addListener(PlayerBlockBreakEvent.class, CommonEventHandles::cancel);
|
||||||
eventNode().addListener(PlayerDisconnectEvent.class, playerDisconnectEvent -> unsetRoom(playerDisconnectEvent.getPlayer()));
|
this.eventNode().addListener(PlayerDisconnectEvent.class, playerDisconnectEvent -> unsetRoom(playerDisconnectEvent.getPlayer()));
|
||||||
}
|
}
|
||||||
|
|
||||||
public Player getOwner() {
|
public Player getOwner() {
|
||||||
return owner;
|
return this.owner;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setOwner(Player newOwner) {
|
public void setOwner(Player newOwner) {
|
||||||
@@ -122,7 +125,7 @@ public class Room extends MineNetInstance implements Spawnable {
|
|||||||
|
|
||||||
if(p != this.owner) return;
|
if(p != this.owner) return;
|
||||||
|
|
||||||
getAllMembers().stream()
|
this.getAllMembers().stream()
|
||||||
.filter(player -> player != p) // exclude the current leaving owner
|
.filter(player -> player != p) // exclude the current leaving owner
|
||||||
.findFirst()
|
.findFirst()
|
||||||
.ifPresentOrElse(
|
.ifPresentOrElse(
|
||||||
@@ -132,8 +135,8 @@ public class Room extends MineNetInstance implements Spawnable {
|
|||||||
|
|
||||||
Room.unsetRoom(p);
|
Room.unsetRoom(p);
|
||||||
|
|
||||||
new ChatMessage(Icon.ERROR).appendStatic("The room leader left!").send(getAllMembers());
|
new ChatMessage(Icon.ERROR).appendStatic("The room leader left!").send(this.getAllMembers());
|
||||||
new ChatMessage(Icon.SCIENCE).appendStatic(this.owner.getUsername()).appendStatic(" is the new Leader!").send(getAllMembers().stream().filter(player -> player != this.owner).collect(Collectors.toSet()));
|
new ChatMessage(Icon.SCIENCE).appendStatic(this.owner.getUsername()).appendStatic(" is the new Leader!").send(this.getAllMembers().stream().filter(player -> player != this.owner).collect(Collectors.toSet()));
|
||||||
new ChatMessage(Icon.SUCCESS).appendStatic("You are now the leader.").send(this.owner);
|
new ChatMessage(Icon.SUCCESS).appendStatic("You are now the leader.").send(this.owner);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@@ -10,7 +10,6 @@ import eu.mhsl.minenet.minigames.shared.inventory.InteractableInventory;
|
|||||||
import eu.mhsl.minenet.minigames.util.InventoryItemAlignment;
|
import eu.mhsl.minenet.minigames.util.InventoryItemAlignment;
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.inventory.InventoryType;
|
import net.minestom.server.inventory.InventoryType;
|
||||||
import net.minestom.server.item.ItemHideFlag;
|
|
||||||
import net.minestom.server.item.ItemStack;
|
import net.minestom.server.item.ItemStack;
|
||||||
import net.minestom.server.item.Material;
|
import net.minestom.server.item.Material;
|
||||||
|
|
||||||
@@ -29,7 +28,7 @@ public class MinigameSelectInventory extends InteractableInventory {
|
|||||||
for (GameType type : GameType.values()) {
|
for (GameType type : GameType.values()) {
|
||||||
setClickableItem(
|
setClickableItem(
|
||||||
ItemStack.builder(type.getIcon())
|
ItemStack.builder(type.getIcon())
|
||||||
.displayName(type.getTitle().getAssembled(p))
|
.customName(type.getTitle().getAssembled(p))
|
||||||
.lore(type.getDescription().addWrap().getWrappedAssembled(p))
|
.lore(type.getDescription().addWrap().getWrappedAssembled(p))
|
||||||
.build(),
|
.build(),
|
||||||
itemAlignment.next().get(),
|
itemAlignment.next().get(),
|
||||||
@@ -58,9 +57,8 @@ public class MinigameSelectInventory extends InteractableInventory {
|
|||||||
|
|
||||||
setClickableItem(
|
setClickableItem(
|
||||||
ItemStack.builder(gameFactory.symbol())
|
ItemStack.builder(gameFactory.symbol())
|
||||||
.displayName(gameFactory.name().getAssembled(p))
|
.customName(gameFactory.name().getAssembled(p))
|
||||||
.lore(gameFactory.description().addWrap().getWrappedAssembled(p))
|
.lore(gameFactory.description().addWrap().getWrappedAssembled(p))
|
||||||
.meta(metaBuilder -> metaBuilder.hideFlag(ItemHideFlag.HIDE_ATTRIBUTES))
|
|
||||||
.build(),
|
.build(),
|
||||||
offset + itemAlignment.next().get(),
|
offset + itemAlignment.next().get(),
|
||||||
itemClick -> itemClick.getPlayer().openInventory(new GameConfigurationInventory(room, itemClick.getPlayer(), gameFactory))
|
itemClick -> itemClick.getPlayer().openInventory(new GameConfigurationInventory(room, itemClick.getPlayer(), gameFactory))
|
||||||
|
@@ -10,7 +10,7 @@ import net.minestom.server.instance.block.Block;
|
|||||||
|
|
||||||
public class Transfer extends MineNetInstance implements Spawnable {
|
public class Transfer extends MineNetInstance implements Spawnable {
|
||||||
public Transfer() {
|
public Transfer() {
|
||||||
super(Dimension.THE_END.DIMENSION);
|
super(Dimension.THE_END.key);
|
||||||
eventNode().addListener(PlayerMoveEvent.class, CommonEventHandles::cancel);
|
eventNode().addListener(PlayerMoveEvent.class, CommonEventHandles::cancel);
|
||||||
setBlock(0, 0, 0, Block.BARRIER);
|
setBlock(0, 0, 0, Block.BARRIER);
|
||||||
}
|
}
|
||||||
|
@@ -28,10 +28,10 @@ public class Languages {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public Lang getLanguage(Player p) {
|
public Lang getLanguage(Player p) {
|
||||||
return getLanguage(p.getSettings().getLocale());
|
return getLanguage(p.getSettings().locale().toString()); // TODO funktioniert die locale noch?
|
||||||
}
|
}
|
||||||
public Lang getLanguage(String mapId) {
|
public Lang getLanguage(String mapId) {
|
||||||
return languages.computeIfAbsent(mapId, unused -> languages.computeIfAbsent(defaultLanguage, (key) -> new DummyLang()));
|
return languages.computeIfAbsent(mapId.toLowerCase(), unused -> languages.computeIfAbsent(defaultLanguage, (key) -> new DummyLang()));
|
||||||
}
|
}
|
||||||
|
|
||||||
private void readAll() {
|
private void readAll() {
|
||||||
|
@@ -10,7 +10,7 @@ import java.util.Set;
|
|||||||
public class LastWinsScore extends Score {
|
public class LastWinsScore extends Score {
|
||||||
@Override
|
@Override
|
||||||
public void insertResultImplementation(Set<Player> p) {
|
public void insertResultImplementation(Set<Player> p) {
|
||||||
getScores().add(0, p);
|
getScores().addFirst(p);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@@ -0,0 +1,65 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.score;
|
||||||
|
|
||||||
|
import eu.mhsl.minenet.minigames.message.Icon;
|
||||||
|
import eu.mhsl.minenet.minigames.message.TranslatableMessage;
|
||||||
|
import eu.mhsl.minenet.minigames.message.type.ChatMessage;
|
||||||
|
import eu.mhsl.minenet.minigames.message.type.TitleMessage;
|
||||||
|
import eu.mhsl.minenet.minigames.util.MapUtil;
|
||||||
|
import net.minestom.server.entity.Player;
|
||||||
|
|
||||||
|
import java.time.Duration;
|
||||||
|
import java.util.*;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
public class PointsWinScore extends Score {
|
||||||
|
private Map<Set<Player>, Integer> scoreOrder = new HashMap<>();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void insertResultImplementation(Set<Player> p, int currentPoints) {
|
||||||
|
this.scoreOrder.put(p, currentPoints);
|
||||||
|
this.scoreOrder = MapUtil.sortByValue(this.scoreOrder);
|
||||||
|
getScores().clear();
|
||||||
|
this.scoreOrder.forEach((player, integer) -> getScores().addFirst(player));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected void insertResultImplementation(Set<Player> p) {
|
||||||
|
this.insertResultImplementation(p, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setDone() {
|
||||||
|
if(isClosed()) return;
|
||||||
|
close();
|
||||||
|
new ChatMessage(Icon.STAR, true)
|
||||||
|
.appendTranslated("score#result")
|
||||||
|
.newLine()
|
||||||
|
.indent()
|
||||||
|
.numberedList(
|
||||||
|
getScores()
|
||||||
|
.stream()
|
||||||
|
.filter(players -> !players.stream()
|
||||||
|
.filter(player -> !player.getUsername().isBlank())
|
||||||
|
.toList()
|
||||||
|
.isEmpty())
|
||||||
|
.map(players -> players
|
||||||
|
.stream()
|
||||||
|
.filter(player -> scoreOrder.get(Set.of(player)) != null)
|
||||||
|
.map(player -> player.getUsername()+" : "+scoreOrder.get(Set.of(player)).toString())
|
||||||
|
.collect(Collectors.joining(", "))
|
||||||
|
)
|
||||||
|
.toList()
|
||||||
|
)
|
||||||
|
.undent()
|
||||||
|
.newLine()
|
||||||
|
.appendTranslated("score#thanks")
|
||||||
|
.send(instance.getPlayers());
|
||||||
|
|
||||||
|
instance.stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected TranslatableMessage scoreMessage() {
|
||||||
|
return new TitleMessage(Duration.ofMillis(1000), Duration.ofSeconds(1)).appendTranslated("score#death");
|
||||||
|
}
|
||||||
|
}
|
@@ -8,6 +8,7 @@ import net.minestom.server.entity.Player;
|
|||||||
import net.minestom.server.event.instance.AddEntityToInstanceEvent;
|
import net.minestom.server.event.instance.AddEntityToInstanceEvent;
|
||||||
import net.minestom.server.event.instance.RemoveEntityFromInstanceEvent;
|
import net.minestom.server.event.instance.RemoveEntityFromInstanceEvent;
|
||||||
import net.minestom.server.event.player.PlayerDisconnectEvent;
|
import net.minestom.server.event.player.PlayerDisconnectEvent;
|
||||||
|
import org.apache.commons.lang3.NotImplementedException;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
@@ -22,10 +23,19 @@ public abstract class Score {
|
|||||||
|
|
||||||
public Score() {}
|
public Score() {}
|
||||||
|
|
||||||
public Score(int ignoreLastPlayers) {
|
protected abstract void insertResultImplementation(Set<Player> p);
|
||||||
this.ignoreLastPlayers = ignoreLastPlayers;
|
|
||||||
|
protected void insertResultImplementation(Set<Player> p, int points) {
|
||||||
|
throw new NotImplementedException("This Score type is not able to process points");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void insertResult(Player p) {
|
||||||
|
this.insertResultProcessor(p, () -> this.insertResultImplementation(Set.of(p)));
|
||||||
|
}
|
||||||
|
|
||||||
|
public void insertResult(Player p, int points) {
|
||||||
|
this.insertResultProcessor(p, () -> this.insertResultImplementation(Set.of(p), points));
|
||||||
|
}
|
||||||
|
|
||||||
public void attachListeners() {
|
public void attachListeners() {
|
||||||
this.instance.eventNode()
|
this.instance.eventNode()
|
||||||
@@ -45,15 +55,8 @@ public abstract class Score {
|
|||||||
setDone();
|
setDone();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
protected abstract void insertResultImplementation(Set<Player> p);
|
|
||||||
protected abstract TranslatableMessage scoreMessage();
|
|
||||||
|
|
||||||
public void insertResult(Player p) {
|
protected abstract TranslatableMessage scoreMessage();
|
||||||
if(hasResult(p)) return;
|
|
||||||
this.scoreMessage().send(p);
|
|
||||||
this.insertResultImplementation(Set.of(p));
|
|
||||||
this.checkGameEnd();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void insertRemainingPlayers(Set<Player> players) {
|
public void insertRemainingPlayers(Set<Player> players) {
|
||||||
this.insertResultImplementation(players.stream().filter(p -> !hasResult(p)).collect(Collectors.toSet()));
|
this.insertResultImplementation(players.stream().filter(p -> !hasResult(p)).collect(Collectors.toSet()));
|
||||||
@@ -93,10 +96,19 @@ public abstract class Score {
|
|||||||
instance.stop();
|
instance.stop();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void insertResultProcessor(Player p, Runnable callback) {
|
||||||
|
if(hasResult(p)) return;
|
||||||
|
this.scoreMessage().send(p);
|
||||||
|
callback.run();
|
||||||
|
this.checkGameEnd();
|
||||||
|
}
|
||||||
|
|
||||||
public boolean isClosed() {
|
public boolean isClosed() {
|
||||||
return isClosed;
|
return isClosed;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void close() { isClosed = true; }
|
||||||
|
|
||||||
protected void onGameEnd() {
|
protected void onGameEnd() {
|
||||||
this.instance.stop();
|
this.instance.stop();
|
||||||
}
|
}
|
||||||
|
@@ -10,7 +10,7 @@ import net.minestom.server.MinecraftServer;
|
|||||||
import net.minestom.server.coordinate.Pos;
|
import net.minestom.server.coordinate.Pos;
|
||||||
import net.minestom.server.entity.*;
|
import net.minestom.server.entity.*;
|
||||||
import net.minestom.server.event.player.PlayerMoveEvent;
|
import net.minestom.server.event.player.PlayerMoveEvent;
|
||||||
import net.minestom.server.instance.AnvilLoader;
|
import net.minestom.server.instance.anvil.AnvilLoader;
|
||||||
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
@@ -19,17 +19,19 @@ public class TournamentDisplay extends MineNetInstance implements Spawnable {
|
|||||||
private final Tournament tournament;
|
private final Tournament tournament;
|
||||||
|
|
||||||
private final Pos[] placePositions = new Pos[] {
|
private final Pos[] placePositions = new Pos[] {
|
||||||
new Pos(8.5, -57, 21.5, 180, 0),
|
new Pos(8.5, -56, 22.5, 180, 0),
|
||||||
new Pos(9.5, -58, 21.5, 180, 0),
|
new Pos(12.5, -57, 20.5, 180, 0),
|
||||||
new Pos(7.5, -59, 21.5, 180, 0)
|
new Pos(4.5, -58, 20.5, 180, 0)
|
||||||
};
|
};
|
||||||
|
|
||||||
public TournamentDisplay(Tournament tournament) {
|
public TournamentDisplay(Tournament tournament) {
|
||||||
super(Dimension.OVERWORLD.DIMENSION);
|
super(Dimension.OVERWORLD.key);
|
||||||
setChunkLoader(new AnvilLoader(Resource.RESULT_DISPLAY.getPath()));
|
setChunkLoader(new AnvilLoader(Resource.RESULT_DISPLAY.getPath()));
|
||||||
this.places = tournament.getPlaces();
|
this.places = tournament.getPlaces();
|
||||||
this.tournament = tournament;
|
this.tournament = tournament;
|
||||||
|
|
||||||
|
this.places.forEach(player -> player.setGameMode(GameMode.ADVENTURE));
|
||||||
|
|
||||||
eventNode().addListener(PlayerMoveEvent.class, playerMoveEvent -> {
|
eventNode().addListener(PlayerMoveEvent.class, playerMoveEvent -> {
|
||||||
if(isOnDisplay(playerMoveEvent.getPlayer()) && !playerMoveEvent.getNewPosition().sameBlock(placePositions[getRankPosition(playerMoveEvent.getPlayer())])) {
|
if(isOnDisplay(playerMoveEvent.getPlayer()) && !playerMoveEvent.getNewPosition().sameBlock(placePositions[getRankPosition(playerMoveEvent.getPlayer())])) {
|
||||||
playerMoveEvent.setCancelled(true);
|
playerMoveEvent.setCancelled(true);
|
||||||
@@ -62,6 +64,6 @@ public class TournamentDisplay extends MineNetInstance implements Spawnable {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public Pos getSpawn() {
|
public Pos getSpawn() {
|
||||||
return new Pos(8.5, -55, 8.5);
|
return new Pos(8.5, -59, 11.5);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -1,30 +0,0 @@
|
|||||||
package eu.mhsl.minenet.minigames.server.provider;
|
|
||||||
|
|
||||||
import eu.mhsl.minenet.minigames.util.UuidUtil;
|
|
||||||
import net.minestom.server.MinecraftServer;
|
|
||||||
import net.minestom.server.network.UuidProvider;
|
|
||||||
import net.minestom.server.network.player.PlayerConnection;
|
|
||||||
import net.minestom.server.utils.mojang.MojangUtils;
|
|
||||||
|
|
||||||
import java.util.UUID;
|
|
||||||
import java.util.logging.Logger;
|
|
||||||
|
|
||||||
public class ByPlayerNameUuidProvider implements UuidProvider {
|
|
||||||
@Override
|
|
||||||
public UUID provide(PlayerConnection playerConnection, String username) {
|
|
||||||
try {
|
|
||||||
|
|
||||||
if(MinecraftServer.getConnectionManager().getPlayer(username) != null) throw new IllegalStateException();
|
|
||||||
String client = MojangUtils.fromUsername(username).get("id").getAsString();
|
|
||||||
return UuidUtil.unTrimm(client);
|
|
||||||
|
|
||||||
} catch (NullPointerException e) {
|
|
||||||
Logger.getGlobal().info("Player " + username + " is unknown by Mojang! (Using random UUID)");
|
|
||||||
} catch (IllegalStateException e) {
|
|
||||||
Logger.getGlobal().info("Player with the username " + username + " is already online.");
|
|
||||||
playerConnection.disconnect();
|
|
||||||
}
|
|
||||||
|
|
||||||
return UUID.randomUUID();
|
|
||||||
}
|
|
||||||
}
|
|
@@ -28,10 +28,6 @@ public class InteractableEntity extends EntityCreature {
|
|||||||
.addListener(PlayerEntityInteractEvent.class, this::onInteract) //TODO Why are some of these listeners not working?
|
.addListener(PlayerEntityInteractEvent.class, this::onInteract) //TODO Why are some of these listeners not working?
|
||||||
|
|
||||||
.addListener(EntityAttackEvent.class, this::onAttack);
|
.addListener(EntityAttackEvent.class, this::onAttack);
|
||||||
|
|
||||||
System.out.println(eventNode().getChildren());
|
|
||||||
|
|
||||||
System.out.println(eventNode().getParent().getChildren());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void setInstanceEvent(@NotNull AddEntityToInstanceEvent addEntityToInstanceEvent) {
|
private void setInstanceEvent(@NotNull AddEntityToInstanceEvent addEntityToInstanceEvent) {
|
||||||
|
@@ -60,7 +60,7 @@ public class InteractableInventory extends Inventory {
|
|||||||
|
|
||||||
protected void setDummyItem(Material material, int slot) {
|
protected void setDummyItem(Material material, int slot) {
|
||||||
this.setDummyItem(
|
this.setDummyItem(
|
||||||
ItemStack.builder(material).displayName(Component.text("")).build(),
|
ItemStack.builder(material).customName(Component.text("")).build(),
|
||||||
slot
|
slot
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@@ -3,7 +3,6 @@ package eu.mhsl.minenet.minigames.skin;
|
|||||||
import net.minestom.server.MinecraftServer;
|
import net.minestom.server.MinecraftServer;
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.entity.PlayerSkin;
|
import net.minestom.server.entity.PlayerSkin;
|
||||||
import net.minestom.server.timer.ExecutionType;
|
|
||||||
import net.minestom.server.timer.TaskSchedule;
|
import net.minestom.server.timer.TaskSchedule;
|
||||||
|
|
||||||
import java.util.HashMap;
|
import java.util.HashMap;
|
||||||
@@ -12,19 +11,15 @@ import java.util.Map;
|
|||||||
public class SkinCache {
|
public class SkinCache {
|
||||||
private static final Map<String, PlayerSkin> skins = new HashMap<>();
|
private static final Map<String, PlayerSkin> skins = new HashMap<>();
|
||||||
|
|
||||||
public static PlayerSkin getSkin(Player p) {
|
|
||||||
return SkinCache.getSkin(p.getUsername());
|
|
||||||
}
|
|
||||||
|
|
||||||
public static PlayerSkin getSkin(String p) {
|
public static PlayerSkin getSkin(String p) {
|
||||||
if(!skins.containsKey(p)) skins.put(p, PlayerSkin.fromUsername(p));
|
if(!skins.containsKey(p)) skins.put(p, PlayerSkin.fromUsername(p));
|
||||||
return skins.get(p);
|
return skins.get(p);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void applySkin(Player p) {
|
public static void applySkin(Player p) {
|
||||||
MinecraftServer.getSchedulerManager().submitTask(() -> {
|
MinecraftServer.getSchedulerManager().scheduleTask(() -> {
|
||||||
p.setSkin(SkinCache.getSkin(p.getUsername()));
|
p.setSkin(SkinCache.getSkin(p.getUsername()));
|
||||||
return TaskSchedule.stop();
|
return TaskSchedule.stop();
|
||||||
}, ExecutionType.ASYNC);
|
}, TaskSchedule.millis(500));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -130,25 +130,10 @@ public class InventoryItemAlignment {
|
|||||||
}.get(count);
|
}.get(count);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class ItemOffset {
|
public record ItemOffset(int x, int z) {
|
||||||
private final int x;
|
|
||||||
private final int z;
|
|
||||||
|
|
||||||
public ItemOffset(int x, int z) {
|
|
||||||
this.x = x;
|
|
||||||
this.z = z;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getX() {
|
|
||||||
return x;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getZ() {
|
|
||||||
return z;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int get() {
|
public int get() {
|
||||||
return x + (z * 9);
|
return x + (z * 9);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
20
src/main/java/eu/mhsl/minenet/minigames/util/MapUtil.java
Normal file
20
src/main/java/eu/mhsl/minenet/minigames/util/MapUtil.java
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
package eu.mhsl.minenet.minigames.util;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.LinkedHashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
public class MapUtil {
|
||||||
|
public static <K, V extends Comparable<? super V>> Map<K, V> sortByValue(Map<K, V> map) {
|
||||||
|
List<Map.Entry<K, V>> list = new ArrayList<>(map.entrySet());
|
||||||
|
list.sort(Map.Entry.comparingByValue());
|
||||||
|
|
||||||
|
Map<K, V> result = new LinkedHashMap<>();
|
||||||
|
for (Map.Entry<K, V> entry : list) {
|
||||||
|
result.put(entry.getKey(), entry.getValue());
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
@@ -1,40 +0,0 @@
|
|||||||
package eu.mhsl.minenet.minigames.util;
|
|
||||||
|
|
||||||
import net.minestom.server.MinecraftServer;
|
|
||||||
import net.minestom.server.entity.GameMode;
|
|
||||||
import net.minestom.server.entity.Player;
|
|
||||||
import net.minestom.server.entity.PlayerSkin;
|
|
||||||
import net.minestom.server.network.packet.server.play.PlayerInfoUpdatePacket;
|
|
||||||
import net.minestom.server.network.packet.server.play.SpawnPlayerPacket;
|
|
||||||
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
public class PacketUtil {
|
|
||||||
public static void resendPlayerList(Player p) {
|
|
||||||
MinecraftServer.getConnectionManager().getOnlinePlayers().forEach(player -> {
|
|
||||||
if(player.getUuid().equals(p.getUuid())) return;
|
|
||||||
|
|
||||||
final PlayerSkin skin = player.getSkin();
|
|
||||||
List<PlayerInfoUpdatePacket.Property> properties =
|
|
||||||
skin != null ? List.of(new PlayerInfoUpdatePacket.Property("textures", skin.textures(), skin.signature())) : List.of();
|
|
||||||
|
|
||||||
p.sendPacket(
|
|
||||||
new PlayerInfoUpdatePacket(PlayerInfoUpdatePacket.Action.ADD_PLAYER,
|
|
||||||
new PlayerInfoUpdatePacket.Entry(
|
|
||||||
player.getUuid(),
|
|
||||||
player.getUsername(),
|
|
||||||
properties,
|
|
||||||
true,
|
|
||||||
0,
|
|
||||||
GameMode.SURVIVAL,
|
|
||||||
null,
|
|
||||||
null
|
|
||||||
)
|
|
||||||
)
|
|
||||||
);
|
|
||||||
p.sendPacket(
|
|
||||||
new SpawnPlayerPacket(player.getEntityId(), player.getUuid(), player.getPosition())
|
|
||||||
);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
@@ -3,7 +3,7 @@ package eu.mhsl.minenet.minigames.util;
|
|||||||
import com.google.common.io.ByteArrayDataOutput;
|
import com.google.common.io.ByteArrayDataOutput;
|
||||||
import com.google.common.io.ByteStreams;
|
import com.google.common.io.ByteStreams;
|
||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.network.packet.server.play.PluginMessagePacket;
|
import net.minestom.server.network.packet.server.common.PluginMessagePacket;
|
||||||
|
|
||||||
public class PluginMessageUtil {
|
public class PluginMessageUtil {
|
||||||
private static final String bungeeTargetSelector = "bungeecord:main";
|
private static final String bungeeTargetSelector = "bungeecord:main";
|
||||||
|
@@ -1,5 +0,0 @@
|
|||||||
package eu.mhsl.minenet.minigames.util;
|
|
||||||
|
|
||||||
public abstract class Static {
|
|
||||||
public abstract void load(); //TODO REMOVE
|
|
||||||
}
|
|
@@ -24,6 +24,6 @@ public class WeatherUtils {
|
|||||||
} else {
|
} else {
|
||||||
return TaskSchedule.stop();
|
return TaskSchedule.stop();
|
||||||
}
|
}
|
||||||
}, ExecutionType.ASYNC);
|
}, ExecutionType.TICK_END);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -12,7 +12,7 @@ public enum BlockPallet {
|
|||||||
STONE(new Block[] {Block.CHISELED_STONE_BRICKS, Block.STONE_BRICKS, Block.POLISHED_ANDESITE, Block.POLISHED_BLACKSTONE, Block.POLISHED_DIORITE}),
|
STONE(new Block[] {Block.CHISELED_STONE_BRICKS, Block.STONE_BRICKS, Block.POLISHED_ANDESITE, Block.POLISHED_BLACKSTONE, Block.POLISHED_DIORITE}),
|
||||||
WINTER(new Block[] {Block.SNOW_BLOCK, Block.ICE, Block.PACKED_ICE, Block.BLUE_CONCRETE, Block.SEA_LANTERN}),
|
WINTER(new Block[] {Block.SNOW_BLOCK, Block.ICE, Block.PACKED_ICE, Block.BLUE_CONCRETE, Block.SEA_LANTERN}),
|
||||||
STREET(new Block[] {Block.BLACK_CONCRETE_POWDER, Block.GRAY_CONCRETE_POWDER, Block.GRAVEL, Block.BLACK_CONCRETE, Block.GRAY_CONCRETE}),
|
STREET(new Block[] {Block.BLACK_CONCRETE_POWDER, Block.GRAY_CONCRETE_POWDER, Block.GRAVEL, Block.BLACK_CONCRETE, Block.GRAY_CONCRETE}),
|
||||||
|
FLOWER(new Block[] {Block.ORANGE_TULIP, Block.PINK_TULIP, Block.RED_TULIP, Block.WHITE_TULIP}),
|
||||||
PRESSURE_PLATES(new Block[] {Block.ACACIA_PRESSURE_PLATE, Block.BIRCH_PRESSURE_PLATE, Block.CRIMSON_PRESSURE_PLATE, Block.JUNGLE_PRESSURE_PLATE, Block.OAK_PRESSURE_PLATE, Block.DARK_OAK_PRESSURE_PLATE, Block.HEAVY_WEIGHTED_PRESSURE_PLATE, Block.HEAVY_WEIGHTED_PRESSURE_PLATE, Block.POLISHED_BLACKSTONE_PRESSURE_PLATE, Block.SPRUCE_PRESSURE_PLATE, Block.STONE_PRESSURE_PLATE, Block.WARPED_PRESSURE_PLATE});
|
PRESSURE_PLATES(new Block[] {Block.ACACIA_PRESSURE_PLATE, Block.BIRCH_PRESSURE_PLATE, Block.CRIMSON_PRESSURE_PLATE, Block.JUNGLE_PRESSURE_PLATE, Block.OAK_PRESSURE_PLATE, Block.DARK_OAK_PRESSURE_PLATE, Block.HEAVY_WEIGHTED_PRESSURE_PLATE, Block.HEAVY_WEIGHTED_PRESSURE_PLATE, Block.POLISHED_BLACKSTONE_PRESSURE_PLATE, Block.SPRUCE_PRESSURE_PLATE, Block.STONE_PRESSURE_PLATE, Block.WARPED_PRESSURE_PLATE});
|
||||||
|
|
||||||
final List<Block> list;
|
final List<Block> list;
|
||||||
|
@@ -21,6 +21,7 @@ public class ValeGenerator extends HeightTerrainGenerator {
|
|||||||
.build();
|
.build();
|
||||||
|
|
||||||
private Function<Integer, Double> xShiftMultiplier = multiplier -> 1d;
|
private Function<Integer, Double> xShiftMultiplier = multiplier -> 1d;
|
||||||
|
private Function<Integer, Double> xShiftOffset = z -> 0d;
|
||||||
|
|
||||||
public ValeGenerator() {
|
public ValeGenerator() {
|
||||||
setCalculateHeight(this::calculateY);
|
setCalculateHeight(this::calculateY);
|
||||||
@@ -32,10 +33,14 @@ public class ValeGenerator extends HeightTerrainGenerator {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public int getXShiftAtZ(int z) {
|
public int getXShiftAtZ(int z) {
|
||||||
return (int) ((curves.getNoise(z) * 32 + largeCurves.getNoise(z) * 64) * xShiftMultiplier.apply(z));
|
return (int) ((curves.getNoise(z) * 32 + largeCurves.getNoise(z) * 64) * xShiftMultiplier.apply(z) + xShiftOffset.apply(z));
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setXShiftMultiplier(Function<Integer, Double> xShiftMultiplier) {
|
public void setXShiftMultiplier(Function<Integer, Double> xShiftMultiplier) {
|
||||||
this.xShiftMultiplier = xShiftMultiplier;
|
this.xShiftMultiplier = xShiftMultiplier;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void setXShiftOffset(Function<Integer, Double> xShiftOffset) {
|
||||||
|
this.xShiftOffset = xShiftOffset;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@@ -53,7 +53,7 @@ public class CircularPlateTerrainGenerator extends PlateTerrainGenerator {
|
|||||||
double distance = bottom.distance(new Pos(0, 0, 0));
|
double distance = bottom.distance(new Pos(0, 0, 0));
|
||||||
|
|
||||||
if(distance <= this.size && generatePlate()) {
|
if(distance <= this.size && generatePlate()) {
|
||||||
unit.modifier().fill(bottom, bottom.add(1, 50, 1), platePallet.rnd());
|
unit.modifier().fill(bottom, bottom.add(1, plateHeight, 1), platePallet.rnd());
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -67,7 +67,7 @@ public class SquarePlateTerrainGenerator extends PlateTerrainGenerator {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if(generateBorders) {
|
if(generateBorders) {
|
||||||
Runnable generateBorder = () -> unit.modifier().fill(bottom, bottom.add(1, DimensionType.OVERWORLD.getMaxY(), 1), Block.BARRIER);
|
Runnable generateBorder = () -> unit.modifier().fill(bottom, bottom.add(1, DimensionType.VANILLA_MAX_Y, 1), Block.BARRIER);
|
||||||
|
|
||||||
if(bottom.z() <= length+1 && bottom.z() >= -1 && bottom.x() >= -1 && bottom.x() <= width+1) {
|
if(bottom.z() <= length+1 && bottom.z() >= -1 && bottom.x() >= -1 && bottom.x() <= width+1) {
|
||||||
if(bottom.x() == -1 || bottom.x() == width+1) {
|
if(bottom.x() == -1 || bottom.x() == width+1) {
|
||||||
|
@@ -18,3 +18,4 @@ api:
|
|||||||
|
|
||||||
admins:
|
admins:
|
||||||
- minetec
|
- minetec
|
||||||
|
- 28Pupsi28
|
@@ -23,7 +23,11 @@ other_description;Games which does not fit into other Categories;Spiele welche n
|
|||||||
pvp;Player vs Player;Spieler gegen Spieler
|
pvp;Player vs Player;Spieler gegen Spieler
|
||||||
pvp_description;Fight against other Players;Kämpfe gegen andere Spieler
|
pvp_description;Fight against other Players;Kämpfe gegen andere Spieler
|
||||||
pve;Player vs Enviroment;Spieler gegen Umwelt
|
pve;Player vs Enviroment;Spieler gegen Umwelt
|
||||||
pve_description;Surivie the world or fight entities;Überlebe die Welt oder kämpfe gegen Mobs
|
pve_description;Survive the world or fight entities;Überlebe die Welt oder kämpfe gegen Mobs
|
||||||
|
jumpnrun;jump and run;jump and run
|
||||||
|
jumpnrun_description;Proof your jump-and-run skills;Stelle deine jump-and-run Fähigkeiten unter Beweis
|
||||||
|
prototype;Prototype;Prototyp
|
||||||
|
prototype_description;Prototype games;Prototyp Spiele
|
||||||
;;
|
;;
|
||||||
ns:hub#;;
|
ns:hub#;;
|
||||||
invTitle;MineNet Servernetwork;MineNet Servernetzwerk
|
invTitle;MineNet Servernetwork;MineNet Servernetzwerk
|
||||||
@@ -36,7 +40,7 @@ join_notFound;Lobby not found: ;Lobby konnte nicht gefunden werden:
|
|||||||
;;
|
;;
|
||||||
ns:score#;;
|
ns:score#;;
|
||||||
result;Results;Ergebnisse
|
result;Results;Ergebnisse
|
||||||
thanks;Thank you for Playing;Danke für's spielen
|
thanks;Thank you for Playing;Danke für's Spielen
|
||||||
finish;You did it;Du hast es geschafft
|
finish;You did it;Du hast es geschafft
|
||||||
death;You are out;Du hast verloren
|
death;You are out;Du hast verloren
|
||||||
done;Finish;Fertig
|
done;Finish;Fertig
|
||||||
@@ -55,6 +59,7 @@ width;Width;Breite
|
|||||||
length;Length;Länge
|
length;Length;Länge
|
||||||
height;Height;Höhe
|
height;Height;Höhe
|
||||||
radius;Radius;Radius
|
radius;Radius;Radius
|
||||||
|
seconds;Seconds;Sekunden
|
||||||
;;
|
;;
|
||||||
ns:room#;;
|
ns:room#;;
|
||||||
invTitle;Select a Minigame;Wähle einen Spielmodus
|
invTitle;Select a Minigame;Wähle einen Spielmodus
|
||||||
@@ -64,15 +69,26 @@ noOptionDescription;There are no options for this Game;Es gibt keine Einstellung
|
|||||||
ns:GameFactory#;;
|
ns:GameFactory#;;
|
||||||
missingDescription;No description;Keine Beschreibung
|
missingDescription;No description;Keine Beschreibung
|
||||||
;;
|
;;
|
||||||
|
ns:game_TntRun#;;
|
||||||
|
name;TNT run;TNT Rennen
|
||||||
|
description;The tnt at the ground disappears, don't fall;Das TNT am Boden verschwindet, falle nicht herunter
|
||||||
|
levels;levels;Ebenen
|
||||||
|
;;
|
||||||
|
ns:game_ElytraRace#;;
|
||||||
|
name;Elytra race;Elytra Rennen
|
||||||
|
description;Be fast while flying through the rings;Sei schnell während du durch die Ringe fliegst
|
||||||
|
ringCount;ring count;Anzahl der Ringe
|
||||||
|
;;
|
||||||
ns:game_Minerun#;;
|
ns:game_Minerun#;;
|
||||||
name;Minerun;Minenrennen
|
name;Minerun;Minenrennen
|
||||||
description;Ditch deadly Mines in the ground an be the first in the goal;Weiche den tödlichen Bodenmienen aus und sei der erste im Ziel
|
description;Ditch deadly Mines in the ground an be the first in the goal;Weiche den tödlichen Bodenmienen aus und sei der erste im Ziel
|
||||||
optionPercentageMines;Percentage of Miens;Prozentsatz der Minen
|
optionPercentageMines;Percentage of mines;Prozentsatz der Minen
|
||||||
;;
|
;;
|
||||||
ns:game_Deathcube#;;
|
ns:game_Deathcube#;;
|
||||||
name;Deathcube;Todeswürfel
|
name;Deathcube;Todeswürfel
|
||||||
description;Find a way to jump higher and be the first on the top;Finde einen weg nach oben und sei der erste im Ziel
|
description;Find a way to jump higher and be the first on the top;Finde einen weg nach oben und sei der erste im Ziel
|
||||||
optionPercentageBlocks;Percentage of Blocks;Prozentsatz der Blöcke
|
optionPercentageBlocks;Percentage of Blocks;Prozentsatz der Blöcke
|
||||||
|
optionPvpEnabled;pvp enabled;PvP aktiviert
|
||||||
;;
|
;;
|
||||||
ns:game_Stickfight#;;
|
ns:game_Stickfight#;;
|
||||||
name;Stickfight;Stockschlacht
|
name;Stickfight;Stockschlacht
|
||||||
@@ -82,6 +98,14 @@ ns:game_TrafficlightRace#;;
|
|||||||
name;Red light green light;Rotes licht, Grünes licht
|
name;Red light green light;Rotes licht, Grünes licht
|
||||||
description;Only go forward if the Trafficlights show green;Gehe nur bei Grün vorran
|
description;Only go forward if the Trafficlights show green;Gehe nur bei Grün vorran
|
||||||
;;
|
;;
|
||||||
|
ns:game_AcidRain#;;
|
||||||
|
name;Acid rain;Säureregen
|
||||||
|
description;Stay under the holey roof to dodge acid rain;Bleib unter dem löchrigen Dach um dem sauren Regen auszuweichen
|
||||||
|
;;
|
||||||
|
ns:game_BowSpleef#;;
|
||||||
|
name;Bow spleef;Bogen Spleef
|
||||||
|
description;Spleef other players and be the last survivor;Zerstöre Blöcke unter anderen Spielern und sei der letzte im Feld
|
||||||
|
;;
|
||||||
ns:game_Towerdefense#;;
|
ns:game_Towerdefense#;;
|
||||||
name;Towerdefense;Towerdefense
|
name;Towerdefense;Towerdefense
|
||||||
description;Protect the path ????;??????
|
description;Protect the path ????;??????
|
||||||
@@ -90,3 +114,32 @@ ns:game_Spleef#;;
|
|||||||
name;Spleef;Spleef;
|
name;Spleef;Spleef;
|
||||||
description;Spleef other players and be the last survivor;Zerstöre Blöcke unter anderen Spielern und sei der letzte im Feld
|
description;Spleef other players and be the last survivor;Zerstöre Blöcke unter anderen Spielern und sei der letzte im Feld
|
||||||
shovelName;Snow thrower;Schneeflug
|
shovelName;Snow thrower;Schneeflug
|
||||||
|
stackCount;levels;Ebenen
|
||||||
|
;;
|
||||||
|
ns:game_Tetris#;;
|
||||||
|
name;Tetris;Tetris
|
||||||
|
description;Sort falling blocks and clear lines;Sortiere fallende Blöcke und kläre Linien
|
||||||
|
nextTetrominoesCount;Number of upcoming Tetrominos displayed;Anzahl der angezeigten kommenden Tetrominos
|
||||||
|
isFast;Fast mode;Schneller Modus
|
||||||
|
hasCombat;Competitive mode;Kompetitiver Modus
|
||||||
|
;;
|
||||||
|
ns:game_AnvilRun#;;
|
||||||
|
name;Anvil run;Anvil run
|
||||||
|
description;Run away from falling anvils;Renne von fallenden Ambossen davon
|
||||||
|
;;
|
||||||
|
ns:game_jumpDive#;;
|
||||||
|
name;Jump dive;Wassersprung
|
||||||
|
description;Jump into the water, avoiding already used spots!;Springe ins wasser an stellen, in denen noch niemand zuvor gelandet ist!
|
||||||
|
;;
|
||||||
|
ns:game_Sumo#;;
|
||||||
|
name;Sumo;Sumo
|
||||||
|
lives;Lives;Leben
|
||||||
|
description;Knock your enemies off and stay on top!;Versuche deinen Gegner von der Plattform zu schubsen!
|
||||||
|
;;
|
||||||
|
ns:game_Highground#;;
|
||||||
|
name;Highground;Hochburg
|
||||||
|
description;Stay on the high ground to win!;Bleibe solange wie möglich auf der Hochburg, um zu gewinnen!
|
||||||
|
;;
|
||||||
|
ns:game_Fastbridge#;;
|
||||||
|
name;Fastbridge;Fastbridge
|
||||||
|
description;Speedbridge to the other platform. The first one there wins!;Baue dich so schnell wie möglich zur anderen Plattform. Wer zuerst dort ist, gewinnt!
|
||||||
|
Can't render this file because it has a wrong number of fields in line 114.
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user